こんばんは。
なんだか妙に暖かい日があったりしましたが、
すっかり冷え込んできて、気づけばもう12月。
今年も残すところあと少しですね。
今回はちょっとリフレクションについて。
そもそもリフレクションでメソッドってどうやって呼ぶ?
軽く確認を。
まずはこんなメソッドを呼んでみましょう。
public class Hoge
{
public void Fuga(string piyo)
{
Debug.WriteLine("Fuga" + piyo);
}
}
こんな感じ。
var hoge = new Hoge();
var type = hoge.GetType();
var method = type.GetMethod("Fuga");
method.Invoke(hoge, new[] { "Piyo" });
タイプからメソッドをとってきて、
第一引数にメソッドを実行するインスタンス、第二引数にメソッドに渡す引数を設定します。
Invokeは戻り値を返しますが、今回はvoidなので取りません。
可変長引数の場合
ちょっと寄り道ですが、可変長引数の場合を軽く確認してみましょう。
public class Hoge
{
public void Fuga(params string[] piyo)
{
Debug.WriteLine("Fuga" + string.Join("", piyo));
}
}
こんな感じ。
~(略)~
method.Invoke(hoge, new[] { new[] { "Piyo", "Piyo", "Piyo" } });
可変長引数は実体としては配列なので、配列としてパラメタを渡してあげればOKです。
省略可能引数の場合
で、本題。
こうして。
public class Hoge
{
public void Fuga(string piyo = default(string))
{
Debug.WriteLine("Fuga" + piyo);
}
}
こうすると、
~(略)~
method.Invoke(hoge, new[] { "Piyo" });
普通に呼べますが、
~(略)~
method.Invoke(hoge, null);
method.Invoke(hoge, new object[0]);
省略しようと思ってこんなふうにしてもエラーになってしまいます。
じゃあどうするかって言うと、
~(略)~
method.Invoke(hoge, new[] { System.Type.Missing });
こんな感じに「System.Type.Missing」を指定してあげます。
せっかくなので拡張メソッドを作ってみましょう。
public static class Extensions
{
public static object InvokeDefault(this MethodBase method, object obj, object[] parameters)
{
var len = method.GetParameters().Length;
var param = (parameters ?? new object[0]).Concat(Enumerable.Repeat(System.Type.Missing, len)).ToArray();
return method.Invoke(obj, param);
}
}
省略可能パラメタは必須パラメタの後ろに書かなくてはならないので、
後ろに足りない分の「System.Type.Missing」を埋めてからInvokeしてあげる感じです。
パラメタを可変長引数にしてあげても良いかもしれないですね。
どうでもいいけどExtensionってtかsか大体いつもどっちが正解か自信がなくて調べてる気が…。