Я пытаюсь привести короткий пример IDynamicMetaObjectProvider
для второго выпуска С# в Depth, и я запускаю вопросы.
Я хочу иметь возможность выразить недействительный вызов, и я терплю неудачу. Я уверен, что это возможно, потому что если я динамически вызываю метод void с помощью связующего, то все в порядке. Вот короткий, но полный пример:
using System;
using System.Dynamic;
using System.Linq.Expressions;
class DynamicDemo : IDynamicMetaObjectProvider
{
public DynamicMetaObject GetMetaObject(Expression expression)
{
return new MetaDemo(expression, this);
}
public void TestMethod(string name)
{
Console.WriteLine(name);
}
}
class MetaDemo : DynamicMetaObject
{
internal MetaDemo(Expression expression, DynamicDemo demo)
: base(expression, BindingRestrictions.Empty, demo)
{
}
public override DynamicMetaObject BindInvokeMember
(InvokeMemberBinder binder, DynamicMetaObject[] args)
{
Expression self = this.Expression;
Expression target = Expression.Call
(Expression.Convert(self, typeof(DynamicDemo)),
typeof(DynamicDemo).GetMethod("TestMethod"),
Expression.Constant(binder.Name));
var restrictions = BindingRestrictions.GetTypeRestriction
(self, typeof(DynamicDemo));
return new DynamicMetaObject(target, restrictions);
}
}
class Test
{
public void Foo()
{
}
static void Main()
{
dynamic x = new Test();
x.Foo(); // Works fine!
x = new DynamicDemo();
x.Foo(); // Throws
}
}
Это генерирует исключение:
Необработанное исключение: System.InvalidCastException: тип результата "System.Void" динамическое связывание, созданное объектом с типом "DynamicDemo" для связующего 'Microsoft.CSharp.RuntimeBinder.CSharpInvokeMemberBinder' не совместим с типом результата "System.Object", ожидаемым вызывать сайт.
Если я изменил метод на возврат объекта и вернул null, он отлично работает... но я не хочу, чтобы результат был нулевым, я хочу, чтобы он был недействительным. Это отлично подходит для связующего сигнала отражения (см. Первый вызов в Main), но он не работает для моего динамического объекта. Я хочу, чтобы он работал как связующее отражение - это прекрасно, чтобы вызвать метод, пока вы не пытаетесь использовать результат.
Я пропустил конкретное выражение, которое я могу использовать в качестве цели?