动态代理的的两种实现方式

动态代理的的两种实现方式

技术杂谈小彩虹2021-06-15 8:24:22680A+A-

一、动态代理简介

动态代理实际上是JVM在运行期动态创建class字节码并加载的过程。

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

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

二、动态代理的两种实现

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

cglib代理:基于父类的动态代理技术

两者的区别如图所示: image.png

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动态代理");
	}

}

运行截图:

image.png

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实现动态规划");
		
	}

}

运行截图:

image.png

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

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

点击这里复制本文地址 以上内容由权冠洲的博客整理呈现,请务必在转载分享时注明本文地址!如对内容有疑问,请联系我们,谢谢!

支持Ctrl+Enter提交

联系我们| 本站介绍| 留言建议 | 交换友链 | 域名展示
本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除

权冠洲的博客 © All Rights Reserved.  Copyright quanguanzhou.top All Rights Reserved
苏公网安备 32030302000848号   苏ICP备20033101号-1

联系我们