ホームページ >バックエンド開発 >C#.Net チュートリアル >C++ 関数のインライン化
1. マクロコードをインライン化に置き換える
C++言語は関数の実行効率(速度)を向上させることを目的とした関数のインライン化をサポートしています。
C プログラムでは、マクロ コードを使用して実行効率を向上させることができます。マクロ コード自体は関数ではありませんが、関数のように動作します。 プリプロセッサはマクロ コードをコピーすることで関数呼び出しを置き換え、パラメータをスタックにプッシュし、アセンブリ言語で CALL 呼び出しを生成し、パラメータを返し、リターンを実行する必要がなくなるため、速度が向上します。マクロ コードを使用する最大の欠点は、マクロ コードをコピーするときに、プリプロセッサによって予期しない副作用が発生することが多いことです。
たとえば、
#define MAX(a, b) (a) > (b) ? (a) : (b)
ステートメント
result = MAX(i, j) + 2 ;
は、プリプロセッサによって
result = (i) > (j) ? (i) : (j) + 2 ;
として解釈されます。 演算子 '+' は演算子 ':' よりも優先順位が高いため、上記のステートメントは予想通り
result = ( (i) > (j) ? (i) : (j) ) + 2 ;
マクロコードを
#define MAX(a, b) ( (a) > (b) ? (a) : (b) )
に書き換えると優先度によるエラーを解決できます。しかし、変更されたマクロ コードを使用しても、確実に実行できるわけではありません。たとえば、ステートメント
result = MAX(i++, j);
はプリプロセッサによって
result = (i++) > (j) ? (i++) : (j);
として解釈されます。C++ の場合、マクロ コードを使用することにはもう 1 つの欠点があります。それは、マクロ コードのプライベート データ メンバーを操作できないことです。クラス、つまりマクロ コードは基本的にパブリックまたはグローバル操作用です。
C++ で「関数のインライン化」がどのように機能するかを見てみましょう。インライン関数の場合、コンパイラは関数の宣言 (名前、パラメーターの型、戻り値の型を含む) をシンボル テーブルに配置します。コンパイラがインライン関数にエラーを検出しなかった場合、その関数のコードもシンボル テーブルに配置されます。インライン関数を呼び出すとき、コンパイラは最初に呼び出しが正しいかどうかをチェックします (型安全性チェックを実行するか、自動型変換を実行します。もちろんすべての関数で同じです)。正しい場合は、インライン関数のコードが関数呼び出しを直接置き換えることになり、関数呼び出しのオーバーヘッドが排除されます。プリプロセッサは型安全性チェックや自動型変換を実行できないため、このプロセスは前処理とは大きく異なります。インライン関数がメンバー関数である場合、オブジェクト (this) のアドレスは適切な場所に配置されますが、これはプリプロセッサでは実行できません。
C++言語の関数インライン化機構は、マクロコードの効率化だけでなく、セキュリティ性も高め、クラスのデータメンバーを自由に操作することができます。したがって、C++ プログラムでは、すべてのマクロ コードをインライン関数に置き換える必要があります。おそらく、「assert」は唯一の例外です。 assert は、デバッグ バージョンでのみ機能するマクロで、「起こるべきではない」状況をチェックするために使用されます。プログラムのデバッグ バージョンとリリース バージョンの間に違いが生じないようにするために、assert には副作用があってはなりません。 Assert が関数の場合、関数呼び出しによってメモリとコードが変更されるため、Debug バージョンと Release バージョンの間に違いが生じます。 したがって、assert は関数ではなくマクロです。
2. インライン関数のプログラミングスタイル
関数をインライン化するには、キーワード inline を関数定義本体と一緒に配置する必要があります。単に関数宣言の前に inline を配置しても効果はありません。次のスタイルの関数 Foo はインライン関数になることはできません:
inline void Foo(int x, int y); // inline 仅与函数声明放在一起,不起任何作用 void Foo(int x, int y) { … }
そして、次のスタイルの関数 Foo はインライン関数になります:
void Foo(int x, int y); inline void Foo(int x, int y) // inline 与函数定义体放在一起 { … }
したがって、inline は「実装のためのキーワード」であり、「キーワード」ではありません。宣言のため」。 一般に、ユーザーは関数の宣言を読むことはできますが、関数の定義を見ることはできません。多くの教科書では inline 関数の宣言や定義本体の前に inline キーワードが追加されていますが、関数の宣言中に inline は登場すべきではないと思います。この詳細は関数の機能には影響しませんが、高品質な C++/C プログラミング スタイルの基本原則を反映しています。つまり、宣言と定義を混同してはならず、ユーザーは関数に必要があるかどうかを知る必要はなく、また知るべきではありません。インライン化する。
クラス宣言で定義したメンバ関数は
class A { public: void Foo(int x, int y) { … } // 自动地成为内联函数 }
のような自動的にインライン関数になります クラス宣言の中にメンバ関数の定義本体を置くと記述が便利になりますが、それは良いことではありません。上の例は次のように変更する必要があります:
// 头文件 class A { public: void Foo(int x, int y); } // 定义文件 inline void A::Foo(int x, int y) { … }
3. インライン化は注意して使用してください
クラスのコンストラクターとデストラクターは、インラインを使用する方が効率的であると誤解されやすいです。コンストラクターとデストラクターには注意してください。関数は、基本クラスまたはメンバー オブジェクトのコンストラクターとデストラクターを「秘密裏に」実行するなど、一部の動作を隠している場合があります。 したがって、クラス宣言にコンストラクターとデストラクターの定義を単に入れないでください。
上記は C++ 関数のインライン化の内容です。さらに関連する内容については、PHP 中国語 Web サイト (www.php.cn) に注目してください。