Javaweb学习笔记——动态代理

1.什么是代理(中介)
  目标对象/被代理对象 ------ 房主:真正的租房的方法
  代理对象 ------- 黑中介:有租房子的方法(调用房主的租房的方法)
  执行代理对象方法的对象 ---- 租房的人

  流程:我们要租房----->中介(租房的方法)------>房主(租房的方法)
  抽象:调用对象----->代理对象------>目标对象

2.动态代理
  动态代理:不用手动编写一个代理对象,不需要一一编写与目标对象相同的方法,这个过程,在运行时的内存中动态生成代理对象。------字节码对象级别的代理对象
Javaweb学习笔记——动态代理_第1张图片  
  动态代理的API:
  在jdk的API中存在一个Proxy中存在一个生成动态代理的的方法newProxyInstance

static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h) 

返回值:Object就是代理对象
参数:loader:代表与目标对象相同的类加载器-------目标对象.getClass().getClassLoader()
   interfaces:代表与目标对象实现的所有的接口字节码对象数组
   h:具体的代理的操作,InvocationHandler接口
注意:JDK的Proxy方式实现的动态代理 目标对象必须有接口 没有接口不能实现jdk版动态代理

示例1(不带参数):
被代理接口示例1:

public interface TargetInterface {

	public void method1();
	
	public String method2();
}

代理类1:

public class Target implements TargetInterface{

	@Override
	public void method1() {
		System.out.println("method1 running...");
	}

	@Override
	public String method2() {
		System.out.println("method2 running...");
		return "method2";
	}
}

动态代理的实现示例1:

public class ProxyTest {

	@Test
	public void test1(){
		//获得动态的代理对象——在运行时 在内存中动态的为target创建一个虚拟的代理对象
		//objProxy是代理对象 根据参数确定到底是谁的代理对象
		TargetInterface objProxy = (TargetInterface) Proxy.newProxyInstance(
							Target.class.getClassLoader(), //与目标对象相同的类加载器
							new Class[]{TargetInterface.class}, //接口的字节码对象数组
							new InvocationHandler() {
								//此处invoke代表的是执行代理对象的方法
								@Override
								//method代表目标对象的方法的字节码对象
								//args代表目标对象的相应的方法的参数
								public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
									System.out.println("目标方法前的逻辑");
									//执行目标对象的方法,如果有返回值则会被invoke接收,如果没有返回值则直接调用
									Object invoke = method.invoke(new Target(), args);//反射,obj表示代理对象
									System.out.println("目标方法后的逻辑");
									return invoke;
								}
							});
		
		//通过动态代理的对象调用方法
		objProxy.method1();
		String method2 = objProxy.method2();
		System.out.println(method2);
	}
	
}

单元测试的运行结果为:

目标方法前的逻辑
method1 running...
目标方法后的逻辑
目标方法前的逻辑
method2 running...
目标方法后的逻辑
method2

示例2(带参数):
接口示例2:

public interface TargetInterface {

	public void method1();
	
	public String method2();
	
	public int method3(int x);
	
}

目标对象示例2:

public class Target implements TargetInterface{

	@Override
	public void method1() {
		System.out.println("method1 running...");
	}

	@Override
	public String method2() {
		System.out.println("method2 running...");
		return "method2";
	}

	@Override
	public int method3(int x) {
		return x;
	}

}

代理对象实现示例2:

public class ProxyTest2 {

	//获取目标对象
	static Target target = new Target();
	
	public static void main(String[] args) {
		
		//动态代理
		TargetInterface proxyInstance = (TargetInterface) Proxy.newProxyInstance(
				target.getClass().getClassLoader(), //获取目标对象的类加载器
				target.getClass().getInterfaces(), //获取目标对象的所有实现接口的字节码对象数组
				new InvocationHandler() {
					//代理对象调用接口的相应方法 其实都是调用的此处的invoke方法
					/*
					 * proxy:代理对象。即proxyInstance,基本不用
					 * method:代表目标对象的方法的字节码对象,代理对象调用的是目标对象的方法
					 * args:代表的是调用目标方法时的参数数组
					 */
					@Override
					public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
						//此处可以添加对参数的逻辑处理来增强方法
						//反射知识点:调用target对象的method方法,传参为args
						Object invoke = method.invoke(target, args);
						//此处可以添加对返回值的逻辑处理来增强方法
						//return返回的值给代理对象
						return invoke;
					}
				});
		proxyInstance.method1();//调用invoke方法,method:目标对象的method1方法   args:null  返回值是null
		String method2 = proxyInstance.method2();//调用invoke方法,method:目标对象的method2方法    args:null   返回值是"method2"
		int method3 = proxyInstance.method3(100); //调用invoke方法,method:目标对象的method3方法   args:Object[]{100}  返回值是100
		
		System.out.println(method2);
		System.out.println(method3);
	}
}

示例2打印结果:

method1 running...
method2 running...
method2
100

注:Spring的AOP的底层代码原理大概就是如此。

你可能感兴趣的:(web开发)