• Java的两大、三类代理模式


    简述

    代理,是一种设计模式,主要作用是为其他对象提供一种代理,以控制对这个对象的访问。在某些情况下,一个对象不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。

    主要分为两大代理:静态代理、动态代理(JDK动态代理、CGLIB动态代理)

    以下面的例子为例,卖房人、代理方、买房人,买卖人双方不需要打交道,通过和代理方打交道,卖房人提供房子给代理方,不需要自己去找买方,代理方有一堆卖房信息,买方直接找代理方即可


    代码实现

    接口

    public interface IHome {
    	void say();
    }
    
    • 1
    • 2
    • 3

    被代理类

    public class ForlanHome implements IHome{
    	@Override
    	public void say() {
    		System.out.println("卖程序员Forlan房子啦!!!");
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    1、静态代理

    public class HomeProxy implements IHome {
    
    	IHome home;
    
    	public HomeProxy(IHome home) {
    		this.home = home;
    	}
    
    	@Override
    	public void say() {
    		System.out.println("This is HomeProxy");
    		home.say();
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    	public static void main(String[] args) {
    		ForlanHome forlanHome = new ForlanHome();
    		HomeProxy homeProxy = new HomeProxy(forlanHome);
    		homeProxy.say();
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5

    2、动态代理

    2.1、JDK动态代理

    2.1.1、普通版

    实现InvocationHandler接口重写invoke方法

    public class HomeInvocationHandler implements InvocationHandler {
    
    	IHome home;
    
    	public HomeInvocationHandler(IHome home) {
    		this.home = home;
    	}
    
    	/**
    	 * 重写invoke方法
    	 *
    	 * @param proxy  生成的代理对象
    	 * @param method 调用的方法
    	 * @param args 方法入参
    	 * @return
    	 */
    	@Override
    	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    		System.out.println("method " + method.getName() + " start!");
    		Object o = method.invoke(home, args);
    		System.out.println("method " + method.getName() + " end!");
    		return o;
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    测试方法

    public static void main(String[] args) {
    	IHome forlanHome = new ForlanHome();
    	IHome home = (IHome) Proxy.newProxyInstance(IHome.class.getClassLoader(),
    			new Class[]{IHome.class},
    			new HomeInvocationHandler(forlanHome)
    	);
    	home.say();
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    2.1.2、反射版

    引入反射封装,方便调用

    public class CommonInvocationHandler<T> implements InvocationHandler {
    
    	T obj;
    
    	public T getProxyObj(T t) {
    		obj = t;
    		T proxyInstance = (T) Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
    		return proxyInstance;
    	}
    
    	/**
    	 * 重写invoke方法
    	 *
    	 * @param proxy  生成的代理对象
    	 * @param method 调用的方法
    	 * @param args   方法入参
    	 * @return
    	 */
    	@Override
    	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    		System.out.println("method " + method.getName() + " start..");
    		Object o = method.invoke(obj, args);
    		System.out.println("method " + method.getName() + " end!");
    		return o;
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26

    测试方法

    public static void main(String[] args) {
    	IHome forlanHome = new ForlanHome();
    	CommonInvocationHandler<IHome> commonInvocationHandler = new CommonInvocationHandler<>();
    	IHome proxyObj = commonInvocationHandler.getProxyObj(forlanHome);
    	proxyObj.say();
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    2.2、CGLIB动态代理

    第三方,需要导入jar包

     <dependency>
         <groupId>cglibgroupId>
         <artifactId>cglibartifactId>
         <version>2.2.2version>
     dependency>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    2.2.1、普通版

    实现MethodInterceptor,重写intercept方法

    public class HomeMethodInterceptor implements MethodInterceptor {
    
    	@Override
    	public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
    		System.out.println("method " + method.getName() + " start!");
    		Object result = null;
    		result = methodProxy.invokeSuper(o, objects);
    		System.out.println("method " + method.getName() + " end!");
    		return result;
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    测试方法

    public static void main(String[] args) {
    	Enhancer enhancer = new Enhancer();
    	enhancer.setSuperclass(ForlanHome.class); // 被代理对象的class
    	enhancer.setCallback(new HomeMethodInterceptor()); // 设置回调,方法拦截器
    	ForlanHome forlanHome = (ForlanHome)enhancer.create(); // 生成动态代理类
    	forlanHome.say();
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    2.2.1、泛型版

    使用泛型,封装Enhancer代码,方便调用

    public class CommonMethodInterceptor<T> implements MethodInterceptor {
    
    	public T getProxyObj(Class<T> tClass) {
    		Enhancer enhancer = new Enhancer();
    		enhancer.setSuperclass(tClass); // 被代理对象的class
    		enhancer.setCallback(this); // 设置回调,方法拦截器
    		return (T) enhancer.create(); // 生成动态代理类
    	}
    
    	@Override
    	public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
    		System.out.println("method " + method.getName() + " start!");
    		Object result = null;
    		result = methodProxy.invokeSuper(o, objects);
    		System.out.println("method " + method.getName() + " end!");
    		return result;
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    测试方法

    public static void main(String[] args) {
    	CommonMethodInterceptor<ForlanHome> commonMethodInterceptor = new CommonMethodInterceptor<>();
    	ForlanHome proxyObj = commonMethodInterceptor.getProxyObj(ForlanHome.class);
    	proxyObj.say();
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5

    总结

    1、静态代理

    代理对象实现被代理对象的接口,通过调用代理接口的接口,实现代理功能

    • 代理对象/被代理对象共同实现接口
    • 多态的一种应用

    2、动态代理

    分离代理行为和被代理对象,替被代理对象干某件事

    • JDK动态代理
      面向接口的动态代理,代理一个对象去增强面向某个接口中定义的方法
      缺点:被代理对象必须实现接口

    • CGLIB动态代理
      面向类的动态代理,重写父类方法,增强使用
      缺点: final类不能被代理

    3、区别

    • 静态代理
      a、自己写生成代理类
      b、代理类和委托类的关系在运行前就确定了
    • 动态代理
      a、在程序运行期间动态生成代理类
      b、代理类和委托类的关系是在程序运行时确定
  • 相关阅读:
    uniapp开发app——nvue
    有营养的算法笔记(二)
    QRCode.js生成的二维码水平居中的解决方案
    [计算机毕业设计]机器学习的数据驱动股票价格预测
    C++原子操作和互斥锁性能(速度)对比
    本地k8s部署kubesphere及踩坑记录
    SpringMVC源码-不同类型的参数解析
    Python 元类详解
    Design patterns--观察者模式
    基于DAP数仓建设过程说明
  • 原文地址:https://blog.csdn.net/qq_36433289/article/details/128195187