Java学习笔记——设计模式:动态代理

2022-06-23 12:28:06 浏览数 (1)

引入

代理设计模式的原理:使用一个代理将对象包装起来,然后用该代理对象取代原始对象。任何对原始对象的调用都要通过代理。代理对象决定是否以及何时将方法调用转到原始对象上。

上一篇设计模式讲到的代理模式属于静态代理,特征是代理类和目标对象的类都是在编译期间确定下来,不利于程序的扩展。同时,每一个代理类只能为一个接口服务,这样一来程序开发中必然产生过多的代理。最好可以通过一个代理类完成全部的代理功能。

简介

动态代理是指客户通过代理类来调用其它对象的方法,并且是在程序运行时根据需要动态创建目标类的代理对象。

动态代理使用场合:

  • 调试
  • 远程方法调用

步骤

AOP

Aspect Oriented Programming(AOP),面向切面编程。

使用Proxy生成一个动态代理时,往往并不会凭空产生一个动态代理,这样没有太大的意义。通常都是为指定的目标对象生成动态代理。

这种动态代理在AOP中被称为AOP代理,AOP代理可代替目标对象,AOP代理包含了目标对象的全部方法。但AOP代理中的方法与目标对象的方法存在差异:AOP代理里的方法可以在执行目标方法之前、之后插入一些通用处理。

代码语言:javascript复制
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

// 接口
interface StudentInfoService{
	void findInfo(String studentName);
	void showRole();
}

// 被代理类
class StudentInfoServiceImpl implements StudentInfoService{

	@Override
	public void findInfo(String studentName) {
		System.out.println("The name is "   studentName);
	}

	@Override
	public void showRole() {
		System.out.println("班干部");
	}
	
}

class MyHandler implements InvocationHandler {

	private Object obj;
	
	public void bind(Object obj) {
		this.obj = obj;
	}
	
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		System.out.println("======调用日志方法"   method.getName()   "======");
		Object returnVal = method.invoke(obj, args);
		System.out.println("======日志方法"   method.getName()   "调用结束======");
		return returnVal;
	}
	
}

class AOPFactory{
	// 动态创建创建一个代理类的对象
	public static Object getProxyInstance(Object obj) {
		MyHandler handler = new MyHandler();
		handler.bind(obj);
		return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), handler);
	}
}

public class TestAOP {
	public static void main(String[] args) {
		// 创建一个被代理类的对象
		StudentInfoServiceImpl serviceImpl = new StudentInfoServiceImpl();
		// 动态获取一个代理类的对象
		StudentInfoService studentInfoService = (StudentInfoService) AOPFactory.getProxyInstance(serviceImpl);
		// 通过代理类的对象调用重写的抽象方法
		studentInfoService.findInfo("runner");
		studentInfoService.showRole();
	}
}
代码语言:javascript复制
运行结果:
======调用日志方法findInfo======
The name is runner
======日志方法findInfo调用结束======
======调用日志方法showRole======
班干部
======日志方法showRole调用结束======

总结

核心在于实现InvocationHandler接口和Proxy类的静态方法newProxyInstance,前者负责与被代理类关联(重写的invoke方法是关键),后者负责动态生成一个代理类。这两者再通过newProxyInstance的第三个参数关联起来,从而实现了动态代理。

AOP是动态代理的一种应用,其核心在于实现InvocationHandler的invoke方法,method.invoke(obj, args)是动态的,其前后可能有其他的硬编码。

0 人点赞