画面に印刷したいだけの場合はstring
、次の 2 つの方法を使用して実行できます。
printf("abc");
std::cout << "abc" << std::endl;
上記の例では、 を使用する利点はありprintf
ますstd::cout
か、またはその逆ですか?
画面に印刷したいだけの場合はstring
、次の 2 つの方法を使用して実行できます。
printf("abc");
std::cout << "abc" << std::endl;
上記の例では、 を使用する利点はありprintf
ますstd::cout
か、またはその逆ですか?
printf
およびそれに関連する仲間は C 関数です。std::ostream
これらは C++ で動作しますが、C++ s の型安全性はありません。printf
関数を使用してユーザー入力 (またはファイルからの入力) に基づいて出力をフォーマットするプログラムでは、問題が発生する可能性があります。例えば:
int main()
{
char[] a = {'1', '2', '3', '4'}; // a string that isn't 0-terminated
int i = 50;
printf("%s", a); // will continue printing characters until a 0 is found in memory
printf("%s", i); // will attempt to print a string, but this is actually an integer
}
C++ には、std::string
このような問題を防ぐのに役立つ、より強力なタイプ セーフ (およびクラス) があります。
私はまさにこの質問に苦労しています。printf は、一般に書式設定された印刷に使用する方が簡単ですが、C++ の iostreams 機能には、オブジェクトのカスタム フォーマッタを作成できるという大きな利点があります。必要に応じて、コードで両方を使用することになります。
両方を使用して混在させる場合の問題は、printf と cout で使用される出力バッファーが同じではないため、出力をバッファーなしで実行するか明示的にフラッシュしない限り、出力が破損する可能性があることです。
C++ に対する私の主な反対意見は、printf に似た高速な出力フォーマット機能がないため、整数、16 進、および浮動小数点フォーマットの出力を簡単に制御する方法がないことです。
Java にもこれと同じ問題がありました。言語は最終的にprintfを取得しました。
ウィキペディアでは、この問題についてhttp://en.wikipedia.org/wiki/Printf#C.2B.2B_alternatives_to_sprintf_for_numeric_conversionでよく議論されています。
実際、あなたの特定の例では、プットとカウトのどちらが好ましいかを尋ねるべきでした。printf はフォーマットされたテキストを出力しますが、プレーンテキストをコンソールに出力しているだけです。
一般的な用途では、ストリーム (cout が含まれる iostream) はより拡張性が高く (独自の型を出力できます)、関数を生成して任意の型のストリームに出力できるという点でより汎用的です。コンソール (またはリダイレクトされた出力)。FILE* を FILE* として受け取る fprintf を使用して、printf で一般的なストリーム動作を作成することもできますが、これは多くの場合、実際のファイルではありませんが、これはよりトリッキーです。
ストリームは、出力する型でオーバーロードするという点で「タイプセーフ」です。printf は省略記号を使用するとタイプセーフではないため、フォーマット文字列と一致しない間違ったパラメーターの型を入力すると、未定義の結果が得られる可能性がありますが、コンパイラーは文句を言いません。パラメータを見逃したり、不正なパラメータを渡した場合 (たとえば、%s の数値をとにかくポインタとして処理した場合)、seg-fault / undefined 動作が発生することさえあります (ただし、誤って使用すると cout を使用する可能性があります)。
ただし、 printf にはいくつかの利点があります。フォーマット文字列をテンプレート化してから、そのデータが構造体にない場合でも、そのフォーマット文字列を別のデータに再利用できます。また、1 つの変数にフォーマット操作を使用しても、そのフォーマットを後で使用するために「固執」することはありません。各変数の形式を指定します。printf はスレッドセーフであることも知られていますが、cout は実際にはそうではありません。
boost は、それぞれの利点を boost::format ライブラリと組み合わせました。
はprintf
C から借用したもので、いくつかの制限があります。の最も一般的に言及されている制限は、printf
型の安全性です。これは、プログラマーがフォーマット文字列を引数と正しく一致させることに依存しているためです。varargs 環境から再び生じる 2 つ目の制限は、ユーザー定義型で動作を拡張できないことです。はprintf
一連のタイプを印刷する方法を知っており、それで得られるのはそれだけです。それでも、使用できるいくつかのことについては、printf
C++ ストリームよりも文字列をフォーマットする方が高速で簡単です。
最新のコンパイラのほとんどは、タイプ セーフの制限に対処し、少なくとも警告を表示できますが (コンパイラは、フォーマット文字列を解析し、呼び出しで提供された引数をチェックできます)、2 番目の制限を克服することはできません。最初のケースでも、ヌル終了のチェックなど、コンパイラが実際には役に立たないことがありますが、std::cout
それを使用して同じ配列を出力すると、同じ問題が発生します。
一方、ストリーム ( を含むstd::cout
) を拡張しstd::ostream& operator<<( std::ostream&, type const & )
て、特定のユーザー定義型をオーバーロードすることにより、ユーザー定義型を処理できますtype
。それらはそれ自体で型安全です。オーバーロードされていない型を渡すとoperator<<
、コンパイラは文句を言います。一方、フォーマットされた出力を生成するのは面倒です。
それで、あなたは何を使うべきですか?operator<<
私自身の型のオーバーロードは簡単で、すべての型で均一に使用できるため、一般的にストリームを使用することを好みます。
これら 2 つの例は、異なることを行います。後者は改行文字を追加し、出力をフラッシュします (の結果std::endl
)。std::cout
も遅いです。それ以外は、同じことprintf
をstd::cout
達成し、好きな方を選択できます。好みの問題として、std::cout
C++ コードで使用します。より読みやすく、より安全です。
を使用して出力をフォーマットする必要がある場合は、この記事を参照してくださいstd::cout
。
一般に、cout の方がタイプセーフで汎用性が高いため、cout を使用することをお勧めします。printf はタイプセーフではなく、まったくジェネリックでもありません。printf を好む唯一の理由は速度です。メモリから見ると、printf は c よりも何倍も高速です。