使用Castle的动态代理时,我遇到了一些(我认为是什么)奇怪的行为.
使用以下代码:
class Program { static void Main(string[] args) { var c = new InterceptedClass(); var i = new Interceptor(); var cp = new ProxyGenerator().CreateClassProxyWithTarget(c,i); cp.Method1(); cp.Method2(); Console.ReadLine(); } } public class Interceptor : IInterceptor { public void Intercept(IInvocation invocation) { Console.WriteLine(string.Format("Intercepted call to: " + invocation.Method.Name)); invocation.Proceed(); } } public class InterceptedClass { public virtual void Method1() { Console.WriteLine("Called Method 1"); Method2(); } public virtual void Method2() { Console.WriteLine("Called Method 2"); } }
我期望得到输出:
>截取的电话:Method1
>被叫方法1
>截取的电话:Method2
>被叫方法2
>截取的电话:Method2
>被叫方法2
但是我得到的是:
>截取的电话:Method1
>被叫方法1
>被叫方法2
>截取的电话:Method2
>被叫方法2
据我所知,动态代理只能在代理方法调用时调用来自于外部类,因为当从程序中调用Method2时被拦截,而不是从InterceptedClass中调用.
我可以明白,当在代理类中进行呼叫时,它将不再通过代理,但只是想检查这是预期的,如果然后看到是否有任何方式让所有呼叫被拦截,不管他们从哪里来?
谢谢
解决方法
编辑:tl; dr – 我刚刚尝试以不同的方式创建代理,如下所述,它产生你以后的输出.我只是改变这个:
var c = new InterceptedClass(); var i = new Interceptor(); var cp = new ProxyGenerator().CreateClassProxyWithTarget(c,i);
为此:
var i = new Interceptor(); var cp = new ProxyGenerator().CreateClassProxy<InterceptedClass>(i);
据了解,代理生成器正在有效地创建一个包装器对象.它们是两个独立的对象 – 一个只是一个包装在另一个对象上,在包装层中有截取等.
很难看到它如何改变InterceptedClass的实例使用自己的方法调用:
> DynamicProxy不能更改现有对象的类型;一旦创建了一个对象,它的类型是固定的
> DynamicProxy不能改变如何绑定现有对象的现有调用
如果您希望Method1通过使用当前代理创建代码的包装器调用Method2,则需要将现有的关于包装器的对象作为其中的字段或方法参数.
或者,可能有不同的方式来创建代理以开始 – 代理在某种意义上是目标对象.我怀疑你可能想看看CreateClassProxy而不是CreateClassProxyWithTarget – 我怀疑这是事实,你提供的目标对象是导致你的问题.
您所看到的行为是否“预期”或不明显取决于您的期望 – 但这肯定是我期望的,而不知道Castle Dynamic Proxy