FormatMessage 関数 (winbase.h)

メッセージ文字列を書式設定します。 関数には、入力としてメッセージ定義が必要です。 メッセージ定義は、関数に渡されるバッファーから取得できます。 これは、既に読み込まれているモジュール内のメッセージ テーブル リソースから取得できます。 または、呼び出し元は、システムのメッセージ テーブル リソースでメッセージ定義を検索するように関数に依頼できます。 この関数は、メッセージ識別子と言語識別子に基づいて、メッセージ テーブル リソース内のメッセージ定義を検索します。 この関数は、書式設定されたメッセージ テキストを出力バッファーにコピーし、要求された場合は埋め込み挿入シーケンスを処理します。

構文

DWORD FormatMessage(
  [in]           DWORD   dwFlags,
  [in, optional] LPCVOID lpSource,
  [in]           DWORD   dwMessageId,
  [in]           DWORD   dwLanguageId,
  [out]          LPTSTR  lpBuffer,
  [in]           DWORD   nSize,
  [in, optional] va_list *Arguments
);

パラメーター

[in] dwFlags

書式設定オプションと lpSource パラメーターの解釈方法。 dwFlags の下位バイトは、関数が出力バッファー内の改行を処理する方法を指定します。 下位バイトでは、書式設定された出力行の最大幅を指定することもできます。

このパラメーターには、次の 1 つ以上の値を指定できます。

説明
FORMAT_MESSAGE_ALLOCATE_BUFFER
0x00000100
この関数は、書式設定されたメッセージを保持するのに十分な大きさのバッファーを割り当て、割り当てられたバッファーへのポインターを lpBuffer で指定されたアドレスに配置します。 lpBuffer パラメーターは、LPTSTR へのポインターです。ポインターを LPTSTR (たとえば) (LPTSTR)&lpBufferにキャストする必要があります。 nSize パラメーターは、出力メッセージ バッファーに割り当てる TCHAR の最小数を指定します。 呼び出し元は 、LocalFree 関数を使用して、不要になったときにバッファーを解放する必要があります。

書式設定されたメッセージの長さが 128K バイトを超える場合、 FormatMessage は失敗し、 GetLastError の後続の呼び出しは ERROR_MORE_DATAを返します。

以前のバージョンの Windows では、この値は Windows ストア アプリのコンパイル時に使用できませんでした。 Windows 10時点では、この値を使用できます。

Windows Server 2003 および Windows XP:

書式設定されたメッセージの長さが 128K バイトを超える場合、 FormatMessage は自動的に失敗せず、 ERROR_MORE_DATAというエラーが発生します。

FORMAT_MESSAGE_ARGUMENT_ARRAY
0x00002000
Arguments パラメーターはva_list構造体ではありませんが、引数を表す値の配列へのポインターです。

このフラグは、64 ビット整数値では使用できません。 64 ビット整数を使用している場合は、 va_list 構造体を使用する必要があります。

FORMAT_MESSAGE_FROM_HMODULE
0x00000800
lpSource パラメーターは、検索するメッセージ テーブル リソースを含むモジュール ハンドルです。 この lpSource ハンドルが NULL の場合、現在のプロセスのアプリケーション イメージ ファイルが検索されます。 このフラグは 、FORMAT_MESSAGE_FROM_STRINGでは使用できません。

モジュールにメッセージ テーブル リソースがない場合、 関数はERROR_RESOURCE_TYPE_NOT_FOUNDで失敗します。

FORMAT_MESSAGE_FROM_STRING
0x00000400
lpSource パラメーターは、メッセージ定義を含む null で終わる文字列へのポインターです。 メッセージ定義には、メッセージ テーブル リソースのメッセージ テキストと同様に、挿入シーケンスを含める場合があります。 このフラグは、 FORMAT_MESSAGE_FROM_HMODULE または FORMAT_MESSAGE_FROM_SYSTEMでは使用できません。
FORMAT_MESSAGE_FROM_SYSTEM
0x00001000
この関数は、システム メッセージ テーブル リソースで要求されたメッセージを検索する必要があります。 このフラグが FORMAT_MESSAGE_FROM_HMODULE で指定されている場合、 lpSource で指定されたモジュールにメッセージが見つからない場合、この関数はシステム・メッセージ・テーブルを検索します。 このフラグは 、FORMAT_MESSAGE_FROM_STRINGでは使用できません。

このフラグを指定すると、アプリケーションは GetLastError 関数の結果を渡して、システム定義エラーのメッセージ テキストを取得できます。

FORMAT_MESSAGE_IGNORE_INSERTS
0x00000200
%1 などのメッセージ定義内の挿入シーケンスは無視され、変更せずに出力バッファーに渡されます。 このフラグは、後で書式設定するためにメッセージをフェッチする場合に便利です。 このフラグが設定されている場合、 Arguments パラメーターは無視されます。
 

dwFlags の下位バイトは、書式設定された出力行の最大幅を指定できます。 下位バイトの使用可能な値を次に示します。

説明
0
出力行幅の制限はありません。 この関数は、メッセージ定義テキスト内の改行を出力バッファーに格納します。
FORMAT_MESSAGE_MAX_WIDTH_MASK
0x000000FF
この関数は、メッセージ定義テキストの通常の改行を無視します。 この関数は、ハードコーディングされた改行をメッセージ定義テキストに出力バッファーに格納します。 この関数は新しい改行を生成しません。
 

下位バイトが FORMAT_MESSAGE_MAX_WIDTH_MASK以外の 0 以外の値の場合は、出力行の最大文字数を指定します。 この関数は、メッセージ定義テキストの通常の改行を無視します。 この関数は、空白で区切られた文字列を改行で分割することはありません。 この関数は、ハードコーディングされた改行をメッセージ定義テキストに出力バッファーに格納します。 ハードコーディングされた改行は、%n エスケープ シーケンスでコード化されます。

[in, optional] lpSource

メッセージ定義の場所。 このパラメーターの型は、 dwFlags パラメーターの設定によって異なります。

Dwflags 設定 説明
FORMAT_MESSAGE_FROM_HMODULE
0x00000800
検索するメッセージ テーブルを含むモジュールへのハンドル。
FORMAT_MESSAGE_FROM_STRING
0x00000400
書式設定されていないメッセージ テキストで構成される文字列へのポインター。 挿入がスキャンされ、それに従って書式設定されます。
 

これらのフラグのどちらも dwFlags で設定されていない場合、 lpSource は無視されます。

[in] dwMessageId

要求されたメッセージのメッセージ識別子。 dwFlagsFORMAT_MESSAGE_FROM_STRINGが含まれている場合、このパラメーターは無視されます。

[in] dwLanguageId

要求されたメッセージの 言語識別子dwFlagsFORMAT_MESSAGE_FROM_STRINGが含まれている場合、このパラメーターは無視されます。

このパラメーターに特定の LANGID を 渡すと、 FormatMessage はその LANGID についてのみメッセージを返します。 関数がその LANGID のメッセージを見つけることができない場合は、Last-ErrorをERROR_RESOURCE_LANG_NOT_FOUNDに設定 します。 0 を渡すと、 FormatMessage は次の順序で LANGID のメッセージを検索します。

  1. 言語に依存しない
  2. スレッドのロケール値に基づくスレッド LANGID
  3. ユーザーの既定のロケール値に基づくユーザーの既定の LANGID
  4. システムの既定のロケール値に基づくシステムの既定の LANGID
  5. 英語 (米国)
FormatMessage が上記の LANGID のメッセージを見つけられない場合は、存在する言語メッセージ文字列を返します。 失敗した場合は、 ERROR_RESOURCE_LANG_NOT_FOUNDが返されます。

[out] lpBuffer

書式設定されたメッセージを指定する null で終わる文字列を受け取るバッファーへのポインター。 dwFlagsFORMAT_MESSAGE_ALLOCATE_BUFFERが含まれている場合、関数は LocalAlloc 関数を使用してバッファーを割り当て、lpBuffer で指定されたアドレスにバッファーへのポインターを配置します。

このバッファーは、64K バイトを超えることはできません。

[in] nSize

FORMAT_MESSAGE_ALLOCATE_BUFFER フラグが設定されていない場合、このパラメーターは出力バッファーのサイズを TCHAR で指定します。 FORMAT_MESSAGE_ALLOCATE_BUFFERが設定されている場合、このパラメーターは、出力バッファーに割り当てる TCHAR の最小数を指定します。

出力バッファーは 64K バイトを超えることはできません。

[in, optional] Arguments

書式設定されたメッセージの挿入値として使用される値の配列。 書式指定文字列の %1 は 、Arguments 配列の最初の値を示します。%2 は 2 番目の引数を示します。などなど。

各値の解釈は、メッセージ定義の挿入に関連付けられている書式設定情報によって異なります。 既定では、各値を null で終わる文字列へのポインターとして扱います。

既定では、 Arguments パラメーターは va_list*型です。これは、可変数の引数を記述するための言語および実装固有のデータ型です。 va_list引数の状態は、関数から戻ると未定義です。 va_listをもう一度使用するには、va_endを使用して変数引数リスト ポインターを破棄し、va_startで再初期化します。

va_list*型のポインターがない場合は、FORMAT_MESSAGE_ARGUMENT_ARRAY フラグを指定し、DWORD_PTR値の配列へのポインターを渡します。これらの値は、挿入値として書式設定されたメッセージに入力されます。 各挿入には、配列内に対応する要素が必要です。

戻り値

関数が成功した場合、戻り値は出力バッファーに格納されている TCHAR の 数であり、終端の null 文字は除きます。

関数が失敗した場合は、0 を返します。 詳細なエラー情報を得るには、GetLastError を呼び出します。

解説

メッセージ テキスト内では、メッセージを動的に書式設定するための複数のエスケープ シーケンスがサポートされています。 これらのエスケープ シーケンスとその意味を次の表に示します。 すべてのエスケープ シーケンスは、パーセント文字 (%)で始まります。

エスケープ シーケンス 説明
%0 末尾の改行文字を含まないメッセージ テキスト行を終了します。 このエスケープ シーケンスは、長い行を作成したり、末尾の改行文字なしでメッセージ自体を終了したりするために使用できます。 プロンプト メッセージに便利です。
%n!書式指定文字列! 挿入シーケンスを識別します。 n の値は、1 から 99 の範囲にすることができます。 書式指定文字列 (感嘆符で囲む必要があります) は省略可能で、既定値は !s! です。 指定されていない場合は 。 詳細については、「 書式指定フィールド」を参照してください。

書式指定文字列には、文字列の幅と有効桁数の指定子、および整数の幅指定子を含めることができます。 アスタリスク () を使用して、幅と有効桁数を指定します。たとえば、%1! です。*S! または %1!*u!.

幅指定子と有効桁数指定子を使用しない場合、挿入番号は入力引数に直接対応します。 たとえば、ソース文字列が "%1 %2 %1" で、入力引数が "Bill" と "Bob" の場合、書式設定された出力文字列は "Bill Bob Bill" になります。

ただし、幅と有効桁数指定子を使用する場合、挿入番号は入力引数に直接対応しません。 たとえば、前の例の挿入番号は "%1!*.*s! %4 %5!*s!"。

挿入番号は、引数配列 (FORMAT_MESSAGE_ARGUMENT_ARRAY) と va_listのどちらを使用するかによって異なります。 引数配列の場合、前の書式指定文字列にアスタリスクが 1 つ含まれている場合、次の挿入番号は n+2、2 つのアスタリスクが指定された場合は n+3 です。 va_listの場合、前の書式指定文字列にアスタリスクが 1 つ含まれている場合、次の挿入番号は n+1 になり、2 つのアスタリスクが指定された場合は n+2 になります。

前の例のように "Bill" を繰り返す場合は、引数に "Bill" を 2 回含める必要があります。 たとえば、ソース文字列が "%1!*.*s! %4 %5!*s!"。引数には、4、2、Bill、Bob、6、Bill ( FORMAT_MESSAGE_ARGUMENT_ARRAY フラグを使用している場合) を指定できます。 書式設定された文字列は "Bi Bob Bill" になります。

ソース文字列に幅と有効桁数の指定子が含まれている場合に、挿入番号を繰り返しても、意図した結果が得られない可能性があります。 %5 を %1 に置き換えた場合、関数はアドレス 6 で文字列を出力しようとします (アクセス違反が発生する可能性があります)。

浮動小数点書式指定子 (e、E、f、g) はサポートされていません。 回避策は、 StringCchPrintf 関数を使用して浮動小数点数を一時バッファーに書式設定し、そのバッファーを挿入文字列として使用することです。

I64 プレフィックスを使用する挿入は、2 つの 32 ビット引数として扱われます。 後続の引数を使用する前に、これらの引数を使用する必要があります。 このプレフィックスの代わりに StringCchPrintf を使用する方が簡単な場合があることに注意してください。

 

パーセント文字の後に続く他の非桁文字は、パーセント文字なしで出力メッセージ内にフォーマットされます。 次は一部の例です。

[書式設定文字列] 結果の出力
%% 1 つのパーセント記号。
%b 1 つのスペース。 この書式指定文字列は、メッセージ テキスト行の末尾のスペースの適切な数を確保するために使用できます。
%. 1 つの期間。 この書式指定文字列を使用すると、メッセージ テキスト定義を終了せずに行の先頭に 1 つのピリオドを含めることができます。
%! 1 つの感嘆符。 この書式指定文字列を使用すると、挿入直後に感嘆符を含めることができます。書式指定文字列の先頭と間違える必要はありません。
%n 書式指定文字列が行末に発生した場合のハード改行。 この書式指定文字列は、 FormatMessage が通常の改行を指定して、メッセージが一定の幅に収まるようにする場合に便利です。
%r 末尾の改行文字を含まないハード キャリッジ リターン。
%t 1 つのタブ。
 

セキュリティに関する備考

この関数をFORMAT_MESSAGE_IGNORE_INSERTSせずに呼び出す場合、Arguments パラメーターには、メッセージ文字列内のすべての挿入シーケンスを満たすのに十分なパラメーターが含まれている必要があり、正しい型である必要があります。 したがって、挿入が有効になっている信頼されていないメッセージ文字列または不明なメッセージ文字列は、 引数 が提供するよりも多くの挿入シーケンスを含むことができるため、または間違った型である可能性があるため使用しないでください。 特に、API から返された任意のシステム エラー コードを取得し、FORMAT_MESSAGE_IGNORE_INSERTSなしでFORMAT_MESSAGE_FROM_SYSTEMを使用することは安全ではありません。

FormatMessage 関数を使用すると、GetLastError によって返されるシステム エラー コードのエラー メッセージ文字列を取得できます。 例については、「 Last-Error コードの取得」を参照してください。

次の例は、引数配列と幅と有効桁数の指定子を使用する方法を示しています。
#ifndef UNICODE
#define UNICODE
#endif

#include <windows.h>
#include <stdio.h>

void main(void)
{
    LPWSTR pMessage = L"%1!*.*s! %4 %5!*s!";
    DWORD_PTR pArgs[] = { (DWORD_PTR)4, (DWORD_PTR)2, (DWORD_PTR)L"Bill",  // %1!*.*s! refers back to the first insertion string in pMessage
         (DWORD_PTR)L"Bob",                                                // %4 refers back to the second insertion string in pMessage
         (DWORD_PTR)6, (DWORD_PTR)L"Bill" };                               // %5!*s! refers back to the third insertion string in pMessage
    const DWORD size = 100+1;
    WCHAR buffer[size];


    if (!FormatMessage(FORMAT_MESSAGE_FROM_STRING | FORMAT_MESSAGE_ARGUMENT_ARRAY,
                       pMessage, 
                       0,
                       0,
                       buffer, 
                       size, 
                       (va_list*)pArgs))
    {
        wprintf(L"Format message failed with 0x%x\n", GetLastError());
        return;
    }

    // Buffer contains "  Bi Bob   Bill".
    wprintf(L"Formatted message: %s\n", buffer);
}


次の例は、 va_listを使用して前の例を実装する方法を示しています。

#ifndef UNICODE
#define UNICODE
#endif

#include <windows.h>
#include <stdio.h>

LPWSTR GetFormattedMessage(LPWSTR pMessage, ...);

void main(void)
{
    LPWSTR pBuffer = NULL;
    LPWSTR pMessage = L"%1!*.*s! %3 %4!*s!";

    // The variable length arguments correspond directly to the format
    // strings in pMessage.
    pBuffer = GetFormattedMessage(pMessage, 4, 2, L"Bill", L"Bob", 6, L"Bill");
    if (pBuffer)
    {
        // Buffer contains "  Bi Bob   Bill".
        wprintf(L"Formatted message: %s\n", pBuffer);
        LocalFree(pBuffer);
    }
    else
    {
        wprintf(L"Format message failed with 0x%x\n", GetLastError());
    }
}

// Formats a message string using the specified message and variable
// list of arguments.
LPWSTR GetFormattedMessage(LPWSTR pMessage, ...)
{
    LPWSTR pBuffer = NULL;

    va_list args = NULL;
    va_start(args, pMessage);

    FormatMessage(FORMAT_MESSAGE_FROM_STRING |
                  FORMAT_MESSAGE_ALLOCATE_BUFFER,
                  pMessage, 
                  0,
                  0,
                  (LPWSTR)&pBuffer, 
                  0, 
                  &args);

    va_end(args);

    return pBuffer;
}

要件

   
サポートされている最小のクライアント Windows XP [デスクトップ アプリ |UWP アプリ]
サポートされている最小のサーバー Windows Server 2003 [デスクトップ アプリ |UWP アプリ]
対象プラットフォーム Windows
ヘッダー winbase.h (Windows.h を含む)
Library Kernel32.lib
[DLL] Kernel32.dll

関連項目

エラー処理関数

メッセージ コンパイラ

メッセージ テーブル