.NET 简易方法拦截器

        伟大的无产阶级Willaim曾说过:"无论你觉得自己多么的了不起,也永远有人比你更强"。对,我说过!我就是william。

       今天想记录一下在项目中遇到的一个比较有意思的东西,异常拦截器(也不能完全说只是异常拦截,准确的说应该叫方法拦截),那有的人可能会说,异常拦截器不就是用Try……Catch就好了吗?没错,Try……Catch是能拦截到异常。如果只是简单拦截下,这种方法简单可行。但是我们如果扩展下,所有的异常都要统一处理,如果你是架构师,你的团队需要你提供一个公共的异常拦截处理组件,你会怎么处理。

        其实在做这个demo之前,我也是想了很多种处理方式,例如在方法前加特性头使用AOP的方式,这种看的比较高大上,也需要配置大量config文件,比较复杂。也不适合我的项目现状(实则是懒,由于框架已经基本形成,采用这种方式需要改动大量的已经完成的框架代码,而且时间紧迫),所以放弃了。开始需求给到我时,一时之间无从下手,在网上搜罗很多处理方式,但大多数都是采用AOP加特性头。当然我也这个demo也是借助于博客园中各位大神的经验。站在巨人的肩膀上,我可以看得更远!谢谢博客园的各位博主!矫情的话不多说了直接开始我的代码之旅。

       首先我们看看效果图,俗话说没图说个J8,有图有文的才是好博文。

       

                                        (这是一张执行正常的截图)

              

                                             (这是一张执行带有异常的截图)

         

                  (demo的结构图,只是简单的模拟)

        解释一下工程结构图每隔类文件的作用。其中红框标记的是本次的重要组件,你的拦截器中需要用他中间的接口。Castle.Core我们可以在NuGet中进行安装。

       1.【MyIntercept.cs

        这个文件就是自定义的拦截器。代码很简单。        

 1 using Castle.DynamicProxy; //必须的
 2 
 3 
 4 public class MyIntercept : IInterceptor //需要实现这个接口(翻译拦截)
 5 {
 6     public void Intercept(IInvocation invocation)
 7     {
 8        Console.WriteLine("【进入拦截器】");
 9        MethodInfo method = invocation.GetConcreteMethod();//得到被拦截的方法
10             var parameter=invocation.Arguments[0].ToString();//获取被拦截的方法参数
11             if (!invocation.MethodInvocationTarget.IsAbstract)
12             {
13                 Console.WriteLine("【被拦截的方法执行前】"+method.Name+"的参数"+ parameter);
14                
15                 try
16                 {
17                     invocation.Proceed();
18                 }
19                 catch (Exception ex)
20                 {
21 
22                     Console.WriteLine("【拦截到异常】"+ex.Message);
23                 }
24                 Console.WriteLine("【被拦截的方法执结果】"+invocation.ReturnValue);
25                
26             }
27             Console.WriteLine("【被拦截的方法执完毕】");
28         }
29     }
View Code

        2.ITestIntercept.cs

       定义一个接口,接口中定义需要实现的方法,也就是需要被拦截的方法     

1 public interface ITestIntercept
2  {
3         string Test(string p);
4 }
View Code

       3.【TestIntercept.cs

        实现上面的接口     

1  public class TestIntercept : ITestIntercept
2 {
3     public string Test(string p)
4     {
5         throw new Exception("异常了"); //演示抛出异常,拦截器是否能捕捉到异常信息
6             //return p;
7     }
8 }
View Code

      基本的异常拦截准备工作已经完毕,我们看看如何使用拦截器对方法进行拦截。

 1 using Castle.DynamicProxy; //必须的
 2 class Program
 3     {
 4         static void Main(string[] args)
 5         {
 6             MyIntercept myIntercept = new MyIntercept();//实例化拦截器
 7             ProxyGenerator proxy = new ProxyGenerator(); //实例化代理
 8             ITestIntercept intercept = proxy.CreateInterfaceProxyWithTarget<ITestIntercept>(new TestIntercept(),myIntercept);
 9             intercept.Test("william");
10             Console.ReadLine();
11         }
12     }
View Code

好了,一个简单的方法拦截demo完成,他可以应用到很多场景,比如:权限验证,异常统计等等。

遗留问题,有兴趣的同学可以试试:

1.如果接口中有多个方法,拦截器会全部拦截吗?

2.ProxyGenerator,我才用的是CreateInterfaceProxyWithTarget对接口和实现类进行mapping。是否还有其他的方式进行mapping工作,例如配置文件?

3.本次demo中采用的是实现接口的方式对方法进行拦截,如果不用接口,拦截器是否会起作用。因为我发现在ProxyGenerator中有个CreateClassProxyWithTarget()函数。大家可以试试。

       

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

                              

posted @ 2018-05-23 13:46  stonewl  阅读(1081)  评论(0编辑  收藏  举报