SmtpClient変数に割り当てられる動的オブジェクトを作成する次のコードがあります。
public class TranferManager
{
public void Tranfer(Account from, Account to, Money amount)
{
// Perform the required actions
var smtpClient = New.SmtpClient();
smtpClient.Send("[email protected]", "from.Email", "Tranfer", "?");
// In the previous line I get a Microsoft.CSharp.RuntimeBinder.RuntimeBinderException
// with the description = "'object' does not contain a definition for 'Send'"
}
}
public static class New
{
public static dynamic SmtpClient(params object[] parameters)
{
return typeof(SmtpClient).New(parameters);
}
}
public static class CreationExtensions
{
private static Dictionary<Type, Func<object, dynamic>> builders =
new Dictionary<Type, Func<object, dynamic>>();
public static dynamic New(this Type type, params object[] parameters)
{
if(builders.ContainsKey(type))
return builders[type](parameters);
return Activator.CreateInstance(type, parameters);
}
public static void RegisterBuilder(this Type type, Func<object, dynamic> builder)
{
builders.Add(type, builder);
}
}
テストするには、UT(下記)を使用しています。
[TestMethod()]
public void TranferTest()
{
typeof(SmtpClient).RegisterBuilder(p =>
new
{
Send = new Action<string, string, string, string>(
(from, to, subject, body) => { })
}
);
var tm = new TranferManager();
tm.Tranfer(new Account(), new Account(), new Money());
// Assert
}
中間ウィンドウを使用して、取得したsmtpClientタイプを尋ねます。
smtpClient.GetType()
{<>f__AnonymousType0`1[System.Action`4[System.String,System.String,System.String,System.String]]}
そして、私がそのメンバーを求めるとき、私は得ます:
smtpClient.GetType().GetMembers()
{System.Reflection.MemberInfo[7]}
[0]: {System.Action`4[System.String,System.String,System.String,System.String] get_Send()}
[1]: {System.String ToString()}
[2]: {Boolean Equals(System.Object)}
[3]: {Int32 GetHashCode()}
[4]: {System.Type GetType()}
[5]: {Void .ctor(System.Action`4[System.String,System.String,System.String,System.String])}
[6]: {System.Action`4[System.String,System.String,System.String,System.String] Send}
だから、私の質問は、なぜその例外が発生するのですか?
匿名の型は内部であり、アセンブリの境界を越える場合、dynamic
はプロパティを解決できません。
匿名型を使用するのではなく、実際の型またはExpandoオブジェクトを使用してみてください。
AssemblyInfo.csに以下を追加してみます。
[Assembly: InternalsVisibleTo("NameSpace1.SubNameSpace1")]
ここで、NamsSpace1はプロジェクト名、SubNameSpaceは動的/匿名オブジェクトの名前空間です。