サマリ
- メソッドに渡す際は、アドレスを渡す。Cのポインタで言う&valのイメージ。
- 引数のpushには ldloca命令を使う。
- メソッド内では、アドレスの解決を行う。Cのポインタで言う*aのイメージ。
- 取得にはldind命令シリーズ(ldobj)、設定にはstind命令シリーズ(stobj)を使う。
- 型は、実際の型に&がついた「ref型」を利用する(例: System.Int32&)
- Type.MakeByRefTypeでref型化、Type.GetElementTypeで非ref型化できる。
- ref型かどうかは、Type.IsByRefで判断できる。
例
具体例として、参照を使うコードにありがちなSwapメソッドを作ってみる。C#コード
void Swap(ref int x, ref int y) { int tmp; tmp = x; x = y; y = tmp; } void SampleFunc() { int x = 10; int y = 20; Swap(ref x, ref y); System.Console.WriteLine("x, y: {0},{1}", x, y); }
これをコンパイルしたものを逆アセンブルしたものが以下。
.method private hidebysig instance void Swap(int32& x, int32& y) cil managed { .maxstack 2 .locals init ([0] int32 tmp) IL_0000: nop IL_0001: ldarg.1 IL_0002: ldind.i4 IL_0003: stloc.0 IL_0004: ldarg.1 IL_0005: ldarg.2 IL_0006: ldind.i4 IL_0007: stind.i4 IL_0008: ldarg.2 IL_0009: ldloc.0 IL_000a: stind.i4 IL_000b: ret } .method private hidebysig instance void SampleFunc() cil managed { .maxstack 3 .locals init ([0] int32 x, [1] int32 y) IL_0000: nop IL_0001: ldc.i4.s 10 IL_0003: stloc.0 IL_0004: ldc.i4.s 20 IL_0006: stloc.1 IL_0007: ldarg.0 IL_0008: ldloca.s x IL_000a: ldloca.s y IL_000c: call instance void ConsoleApplication1.Program::Swap(int32&, int32&) IL_0011: nop IL_0012: ldstr "x, y: {0},{1}" IL_0017: ldloc.0 IL_0018: box [mscorlib]System.Int32 IL_001d: ldloc.1 IL_001e: box [mscorlib]System.Int32 IL_0023: call void [mscorlib]System.Console::WriteLine(string, object, object) IL_0028: nop IL_0029: ret } // end of method Program::SampleFunc
呼ばれるSwapメソッド
適当に端折りながら、上から見ていく。(評価スタックのイメージを忘れてしまった場合は、この絵を思い出してほしい。)
まずメソッドの定義。
.method private hidebysig instance void Swap(int32& x, int32& y) cil managedint32ではなく、int32&型が使われている。これは、
Type t = typeof(Int32).MakeByRefType();てな感じで得られる。
そしてtmp = x;の部分。
IL_0001: ldarg.1 IL_0002: ldind.i4 IL_0003: stloc.0これは、
- 1番目の引数(ref x)を持ってくる
- 持ってきたアドレスからint32の値を持ってくる
- 0番目のローカル変数に格納する
x = y;はいったん飛ばして、y = tmp;部分を見るとこう。
IL_0008: ldarg.2 IL_0009: ldloc.0 IL_000a: stind.i4これは、
- 2番目の引数(ref y)を持ってくる
- 0番目のローカル変数からint32の値を持ってくる
- 持ってきたアドレスに持ってきた値を入れる
x = yは上記の組み合わせ。
IL_0004: ldarg.1 IL_0005: ldarg.2 IL_0006: ldind.i4 IL_0007: stind.i4
ここでは、int32なので、ldind.i4/stind.i4だが、型によって使うべき命令は違う。ただし、
すべての ldind 命令は、対応している組み込み値クラスを指定する Ldobj 命令のショートカットです。(OpCodes.Ldind_I4 フィールド)なので、ldobj [type]でもよい。
ldobj/stobjの例としては、ref int?を使うと、以下の感じで生成される。
IL_0008: stobj valuetype [mscorlib]System.Nullable`1
Swap()を呼ぶメソッド
こちらは実際に呼び出しているところだけ。IL_0001: ldc.i4.s 10 IL_0003: stloc.0 IL_0004: ldc.i4.s 20 IL_0006: stloc.1 IL_0007: ldarg.0 IL_0008: ldloca.s x IL_000a: ldloca.s y IL_000c: call instance void ConsoleApplication1.Program::Swap(int32&, int32&)10、20をそれぞれx、yに入れたあと、this、xのアドレス、yのアドレスの順に持ってきて、関数をコールしている。
リフレクションでの扱い
すでにあるメソッドのMethodInfoをGetMethod()で持ってきて、ParameterTypeを見ると、&のついたref型で入っている。この元の型を得たいときは、こんな感じになる。
if (parameterType.IsByRef) { Type valueType = parameterType.GetElementType(); }(Type.MakeByRefType メソッド)
0 件のコメント:
コメントを投稿