Java 动态代理的多种实现方式

目录一、动态代理简介二、动态代理的多种实现1. 基于JDK的实现2. 基于cglib的实现三、为什么要有基于cglib的实现四、两种方式的适用场景JDK动态代理优点缺点适用场景cglib优点缺点适用场景

一、动态代理简介

优势:在不修改源码的情况下,对目标方法进行相应的增强。

作用:完成程序功能之间的松耦合。

二、动态代理的多种实现 JDK代理:基于接口的动态代理技术(缺点,目标对象必须有接口,如果没有接口,则无法完成动态代理的实现) cglib代理:基于父类的动态代理技术

两者的区别如图所示:

1. 基于JDK的实现

目标接口类:

public interface TargetInterface {public void save();public void print(String str);}

目标类:

public class Target implements TargetInterface{public void save() {System.out.println("save running...");}public void print(String str) {System.out.println(str);}}

增强类:

public class Advice {public void before() {System.out.println("前置增强");}public void after() {System.out.println("后置增强");}}

测试类:

import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;public class ProxyTest {public static void main(String[] args) {//目标对象final Target target = new Target();//增强对象final Advice advice = new Advice();TargetInterface proxyInstance = (TargetInterface)Proxy.newProxyInstance(target.getClass().getClassLoader(), //目标对象类加载器target.getClass().getInterfaces(), //目标对象相同的接口字节码对象数组new InvocationHandler() {//调用代理对象的任何方法,实质执行的都是invoke方法public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{advice.before();//前置增强Object invoke = method.invoke(target, args);//执行目标方法advice.after();//后置增强System.out.println();return invoke;}});//代理对象的方法测试proxyInstance.save();proxyInstance.print("JDK动态代理");}}

运行截图:

2. 基于cglib的实现

需要导入Jar包,如果是maven项目,则在pom.xml文件加入如下配置:

<dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>4.2.4.RELEASE</version></dependency>

目标类:

public class Target {public void save() {System.out.println("save running...");}public void print(String str) {System.out.println(str);}}

增强类:

public class Advice {public void before() {System.out.println("前置增强");}public void after() {System.out.println("后置增强");}}

测试类:

import java.lang.reflect.Method;import org.springframework.cglib.proxy.Enhancer;import org.springframework.cglib.proxy.MethodInterceptor;import org.springframework.cglib.proxy.MethodProxy;public class ProxyTest {public static void main(String[] args) {final Target target = new Target();final Advice advice = new Advice();//返回值就是动态生成的代理对象,基于cglib//创建增强器Enhancer enhancer = new Enhancer();//设置父类(目标)enhancer.setSuperclass(Target.class);//设置回调enhancer.setCallback(new MethodInterceptor() {public Object intercept(Object o, Method method, Object[] obj, MethodProxy methodProxy) throws Throwable{advice.before();Object invoke = method.invoke(target, obj);advice.after();System.out.println();return invoke;}});//创建代理对象Target proxy = (Target)enhancer.create();//测试代理方法proxy.save();proxy.print("基于cglib实现动态规划");}}

运行截图:

三、为什么要有基于cglib的实现

使用JDK动态代理实现时,最大限制是被增强对象必须实现接口,并且增强的方法只能是接口中声明的方法。但在实际的项目中,可能总是存在对不实现业务接口的对象进行增强的需求,这时JDK动态代理将无能为力。

四、两种方式的适用场景

JDK动态代理

优点 不依赖第三方jar包, 使用方便 随着JDK的升级,JDK动态代理的性能在稳步提升

缺点 只能代理实现了接口的类 执行速度较慢

适用场景 如果你的程序需要频繁、反复地创建代理对象,则JDK动态代理在性能上更占优。

cglib优点

由于是动态生成字节码实现代理,因此代理对象的执行速度较快, 约为JDK动态代理的1.5 ~ 2倍可以代理没有实现接口的对象

缺点 不能代理final类 动态生成字节码虽然执行较快,但是生成速度很慢,根据网上一些人的测试结果,cglib创建代理对象的速度要比JDK慢10 ~ 15倍。适用场景 不需要频繁创建代理对象的应用,如Spring中默认的单例bean,只需要在容器启动时生成一次代理对象。

以上就是Java 动态代理的多种实现方式的详细内容,更多关于Java 动态代理的实现的资料请关注其它相关文章!

如果困难是堵砖墙,拍拍它说你还不够高。

Java 动态代理的多种实现方式

相关文章:

你感兴趣的文章:

标签云: