程序员麦兜【编程笔记】-Java静态代理与动态代理
2019-06-13 16:22
323 查看
------代理模式
为其他对象提供一种代理以控制对这个对象的访问。比如A对象要做一件事情,在没有代理前,自己来做。在对A代理后,由A的代理类B来做。
------静态代理的原理
静态代理说白了,就是在程序运行前就已经存在代理类的字节码文件、代理类和原始类的关系在运行前就已经确定。
实例:
package test.staticProxy; // 接口 public interface IUserDao { void save(); void find(); } //目标对象 class UserDao implements IUserDao{ @Override public void save() { System.out.println("模拟:保存用户!"); } @Override public void find() { System.out.println("模拟:查询用户"); } } /** * 静态代理 * 特点: * 2. 目标对象必须要实现接口 * 2. 代理对象 4000 ,要实现与目标对象一样的接口 */ class UserDaoProxy implements IUserDao{ // 代理对象,需要维护一个目标对象 private IUserDao target = new UserDao(); @Override public void save() { System.out.println("代理操作: 开启事务..."); target.save(); // 执行目标对象的方法 System.out.println("代理操作:提交事务..."); } @Override public void find() { target.find(); } }
------动态代理的原理
静态代理虽然保证了业务类只需要关注逻辑本身,代理对象的一个接口只服务于一种类型的对象.如果要代理的方法很多,势必要为每一种方法都进行代理。再者,如果增加一个方法,除了实现类需要实现这个方法外,所有的代理类也要实现此方法。增加了代码的维护成本。那么要如何解决呢?答案是使用动态代理。动态代理类的源码是在程序运行期间,通过 JVM 反射等机制动态生成。代理类和委托类的关系是运行时才确定的.
实例:
package test.dynamicProxy; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; // 接口 public interface IUserDao { void save(); void find(); } //目标对象 class UserDao implements IUserDao{ @Override public void save() { System.out.println("模拟: 保存用户!"); } @Override public void find() { System.out.println("查询"); } } /** * 动态代理: * 代理工厂,给多个目标对象生成代理对象! * */ class ProxyFactory { // 接收一个目标对象 private Object target; public ProxyFactory(Object target) { this.target = target; } // 返回对目标对象(target)代理后的对象(proxy) public Object getProxyInstance() { Object proxy = Proxy.newProxyInstance( target.getClass().getClassLoader(), // 目标对象使用的类加载器 target.getClass().getInterfaces(), // 目标对象实现的所有接口 new InvocationHandler() { // 执行代理对象方法时候触发 @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // 获取当前执行的方法的方法名 String methodName = method.getName(); // 方法返回值 Object result = null; if ("find".equals(methodName)) { // 直接调用目标对象方法 result = method.invoke(target, args); } else { System.out.println("开启事务..."); // 执行目标对象方法 result = method.invoke(target, args); System.out.println("提交事务..."); } return result; } } ); return proxy; } }
------动态代理的两种方式
JDK代理:
使用 JDK 生成的动态代理的前提是目标类必须有实现的接口。
CGLIB代理:
JDK代理引入一个问题,如果某个类没有实现接口,就不能使用 JDK 动态代理,所以 CGLIB 代理就是解决这个问题的。CGLIB 是以动态生成的子类继承目标的方式实现,在运行期动态的在内存中构建一个子类。CGLIB 使用的前提是目标类不能为 final 修饰。因为 final 修饰的类不能被继承。
相关文章推荐
- java静态代理与动态代理笔记
- java的静态代理和动态代理实现笔记
- 程序员麦兜【编程笔记】-java并发编程:CountDownLatch、CyclicBarrier、Semaphore
- Java代理学习笔记(静态代理&动态代理)
- 【Java反射学习笔记系列之jdk动态代理】静态代理和动态代理的区别以及动态代理的作用和实现
- Java学习笔记:反射与代理机制(静态、动态)
- java静态和动态代理学习笔记
- java 笔记(3) —— 动态代理,静态代理,cglib代理
- Java 静态代理和动态代理
- java 静态代理与动态代理(代理模式)
- Spring静态代理与动态代理机制笔记
- java静态代理和动态代理
- java动态代理学习笔记
- java学习之路----静态代理---动态代理-----AOP的前奏(AOP也是动态代理)
- Java的静态代理和动态代理(网页排版不好,附上pdf附件)
- Java之代理(jdk静态代理,jdk动态代理,cglib动态代理,aop,aspectj)
- Java静态代理、动态代理实例
- java动态代理学习笔记
- 深入研究java中的静态代理和动态代理
- JAVA的静态代理与动态代理比较