从代理返回异常

时间:2015-02-04 11:11:49

标签: c# castle-dynamicproxy

我使用了大量未记录的Castle动态代理系统。我设法让它做了我想要的几乎所有事情,除了一件事:你如何使代理方法抛出异常而不是返回值?

public sealed class MyInterceptor : IInterceptor
{
  public void Intercept(IInvocation invocation)
  {
    if (CheckArgs(invocation.Arguments))
    {
      invocation.ReturnValue = DoRealWork(invocation.Arguments);
    }
    else
    {
      invocation.Exception = new InvalidOperationException(); // How?
    }
  }
}

1 个答案:

答案 0 :(得分:1)

从代理对象的角度看,拦截器不可见;它只是调用自己的虚方法,DynamicProxy在将ReturnValue返回给调用者之前调用正确的拦截器方法。

因此,如果你想抛出异常,只需从拦截器中抛出它:

if (CheckArgs(invocation.Arguments))
{
  invocation.ReturnValue = DoRealWork(invocation.Arguments);
}
else
{
  throw new InvalidOperationException();
}

从调用者的角度来看,它将是被调用方法中的一个例外。

修改评论

关于生成器中抛出的异常的类型,我有正确的类型,而不是包装器:

public interface IDummy
{
    string DoSomething();
}

public class Dummy: IDummy {
    public virtual string DoSomething()
    {
        return string.Empty;
    }
}

public class MyCustomException : Exception {}

public class CustomIntercept: IInterceptor
{
    public void Intercept(IInvocation invocation)
    {
        throw new MyCustomException();
    }
}

internal class Program
{
    private static void Main(string[] args)
    {
        var pg = new ProxyGenerator();
        GetValue(pg.CreateInterfaceProxyWithoutTarget<IDummy>(new CustomIntercept()));
        GetValue(pg.CreateClassProxy<Dummy>(new CustomIntercept()));
        GetValue(pg.CreateClassProxyWithTarget<Dummy>(new Dummy(), new CustomIntercept()));
        GetValue(pg.CreateInterfaceProxyWithTarget<IDummy>(new Dummy(), new CustomIntercept()));
    }

    private static void GetValue(IDummy dummy)
    {
        try
        {
            dummy.DoSomething();
        }
        catch (Exception e)
        {
            Console.WriteLine(e.GetType().Name);
        }
    }
}

所有四个输出均为MyCustomException

您能否确保TargetInvocationException不是来自您自己的代码?你正在使用什么版本的DynamicProxy(我正在使用Castle.Core 3.2中的那个)