博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
利用Proxy.newProxyInstance实现AOP
阅读量:6246 次
发布时间:2019-06-22

本文共 3959 字,大约阅读时间需要 13 分钟。

newProxyInstance()方法介绍

Proxy类的newInstance()方法有三个参数:

ClassLoader loader:
它是类加载器类型,你不用去理睬它,你只需要知道怎么可以获得它就可以了:MyInterface.class.getClassLoader()就可以获取到ClassLoader对象,没错,只要你有一个Class对象就可以获取到ClassLoader对象;
Class[] interfaces:
指定newProxyInstance()方法返回的对象要实现哪些接口,没错,可以指定多个接口,例如上面例子只我们只指定了一个接口:Class[] cs = {MyInterface.class};
InvocationHandler h:
它是最重要的一个参数!它是一个接口!它的名字叫调用处理器!无论你调用代理对象的什么方法,它都是在调用InvocationHandler的invoke()方法!

AOP介绍

Aspect Oriented Programming(AOP)是较为热门的一个话题。AOP,国内大致译作“面向切面编程”。

  • “面向切面编程”,这样的名字并不是非常容易理解,且容易产生一些误导。
  • 笔者不止一次听到类似“OOP/OOD11即将落伍,AOP是新一代软件开发方式”这样的发言。而在AOP中,Aspect的含义,可能更多的理解为“切面”比较合适。所以笔者更倾向于“面向切面编程”的译法。可以通过预编译方式和运行期动态代理实现在不修改源代码的情况下给程序动态统一添加功能的一种技术。
  • AOP实际是GoF设计模式的延续,设计模式孜孜不倦追求的是调用者和被调用者之间的解耦,提高代码的灵活性和可扩展性,AOP可以说也是这种目标的一种实现。
    应用对象只实现它们应该做的——完成业务逻辑——仅此而已。它们并不负责(甚至是意识)其它的系统级关注点,例如日志或事务支持。

AOP主要功能

日志记录,性能统计,安全控制,事务处理,异常处理等等wn及扩展

AOP主要意图

将日志记录,性能统计,安全控制,事务处理,异常处理等代码从业务逻辑代码中划分出来,通过对这些行为的分离,我们希望可以将它们独立到非指导业务逻辑的方法中,进而改变这些行为的时候不影响业务逻辑的代码。

AOP实现

package org.java.base.proxy;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;public class MainApp1 { public static void main(String[] args) { ClassLoader loader = MainApp1.class.getClassLoader(); Class[] cs = {Waiter.class}; Waiter target = new MyWaiter(); MyInvocationHandler h = new MyInvocationHandler(target); Waiter waiter = (Waiter)Proxy.newProxyInstance(loader, cs, h); waiter.serve(); }}class MyInvocationHandler implements InvocationHandler { public Waiter target; public MyInvocationHandler(Waiter target) { this.target = target; } public Object invoke(Object proxy, Method method, Object[] args)throws Throwable { System.out.println("您好!"); Object result = method.invoke(target, args); System.out.println("很高兴为您服务!"); return result; }}复制代码
package org.java.base.proxy;/** * 后置增强 * @author Liuhaihua * */public interface AfterAdvice { public void after();}复制代码
package org.java.base.proxy;/** * 前置增强 * @author Liuhaihua * */public interface BeforeAdvice { public void before();}复制代码
package org.java.base.proxy;public class MyWaiter implements Waiter { public void serve() { System.out.println("服务..."); }}复制代码
package org.java.base.proxy;public interface Waiter { // 服务 public void serve(); }复制代码
package org.java.base.proxy;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;/** * 它用来生成代理对象 * 它需要所有的参数 * * 目标对象 * * 增强 * @author cxf *//** * 1. 创建代理工厂 * 2. 给工厂设置三样东西: * * 目标对象:setTargetObject(xxx); * * 前置增强:setBeforeAdvice(该接口的实现) * * 后置增强:setAfterAdvice(该接口的实现) * 3. 调用createProxy()得到代理对象 * * 执行代理对象方法时: * > 执行BeforeAdvice的before() * > 目标对象的目标方法 * > 执行AfterAdvice的after() * @author cxf * */public class ProxyFactory { private Object targetObject;//目标对象 private BeforeAdvice beforeAdvice;//前置增强 private AfterAdvice afterAdvice;//后置增强 /** * 用来生成代理对象 * @return */ public Object createProxy() { /* * 1. 给出三大参数 */ ClassLoader loader = this.getClass().getClassLoader(); Class[] interfaces = targetObject.getClass().getInterfaces(); InvocationHandler h = new InvocationHandler() { public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { /* * 在调用代理对象的方法时会执行这里的内容 */ // 执行前置增强 if(beforeAdvice != null) { beforeAdvice.before(); }  Object result = method.invoke(targetObject, args);//执行目标对象的目标方法 // 执行后置增强 if(afterAdvice != null) { afterAdvice.after(); }  // 返回目标对象的返回值 return result; } }; /* * 2. 得到代理对象 */ Object proxyObject = Proxy.newProxyInstance(loader, interfaces, h); return proxyObject; }   public Object getTargetObject() { return targetObject; } public void setTargetObject(Object targetObject) { this.targetObject = targetObject; } public BeforeAdvice getBeforeAdvice() { return beforeAdvice; } public void setBeforeAdvice(BeforeAdvice beforeAdvice) { this.beforeAdvice = beforeAdvice; } public AfterAdvice getAfterAdvice() { return afterAdvice; } public void setAfterAdvice(AfterAdvice afterAdvice) { this.afterAdvice = afterAdvice; }}复制代码

代码下载:

转载地址:http://udlia.baihongyu.com/

你可能感兴趣的文章
UMeditor上传图片配置
查看>>
Homestead小结
查看>>
2015年iOS开发总结
查看>>
CocoaPods 安装与使用
查看>>
学习笔记:查最大内存
查看>>
javascript设计模式与开发实践(二)- 封装和原型模式
查看>>
1475: 方格取数
查看>>
太阳黑子
查看>>
[转载]date命令时间转换
查看>>
适配iOS11新特性代码在Xcode8 上编译通过的方法
查看>>
别再问什么是Java内存模型了,看这里!
查看>>
2014.5.7—20岁这几年
查看>>
javac - Java programming language compiler
查看>>
异常处理
查看>>
Python多线程程序中的MYSQL连接管理研究
查看>>
Prometheus学习系列(七)之名词解析
查看>>
一文彻底搞懂Dart的event队列
查看>>
iOS面试题06-其他
查看>>
JSON和JSONP
查看>>
2019年互联网女皇趋势报告:小程序创新创业商业模式引领全球
查看>>