简述

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

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

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


代码实现

接口

public interface IHome {
void say();
}

被代理类

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

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();
}
}
	public static void main(String[] args) {
ForlanHome forlanHome = new ForlanHome();
HomeProxy homeProxy = new HomeProxy(forlanHome);
homeProxy.say();
}

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;
}
}

测试方法

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();
}

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;
}
}

测试方法

public static void main(String[] args) {
IHome forlanHome = new ForlanHome();
CommonInvocationHandler<IHome> commonInvocationHandler = new CommonInvocationHandler<>();
IHome proxyObj = commonInvocationHandler.getProxyObj(forlanHome);
proxyObj.say();
}

2.2、CGLIB动态代理

第三方,需要导入jar包

 <dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>2.2.2</version>
</dependency>

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;
}
}

测试方法

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();
}

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;
}
}

测试方法

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

总结

1、静态代理

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

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

2、动态代理

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

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

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

3、区别

  • 静态代理
    a、自己写生成代理类
    b、代理类和委托类的关系在运行前就确定了
  • 动态代理
    a、在程序运行期间动态生成代理类
    b、代理类和委托类的关系是在程序运行时确定
 

最新文章

  1. RTSP协议转换RTMP直播协议
  2. 在Linux最小系统上编译运行第一个helloworld程序
  3. vlan与交换机端口模式Access,Hybrid,Trunk
  4. jQuery 判断表单中多个 input text 中至少有一个不为空
  5. lib和dll文件的区别和联系
  6. 意外发现,VC断点可加在构造函数的左括号上
  7. UGUI 过渡动画插件,模仿NGUI的Tween (转载)
  8. Unity该插件NGUI学习(1)—— 环境结构
  9. IOS开发-UI学习-使用UIImageView控件制作动画
  10. Mac 上所有的命令行相关问题的总结
  11. 使用cocos creator的过程中碰到的问题记录
  12. python爬虫scrapy之downloader_middleware设置proxy代理
  13. input修改placeholder文字颜色
  14. Python local error
  15. python08内置函数
  16. ThinkPHP 3.1.2 CURD特性 -3
  17. 利用Palette库来取得图片中的主要色彩
  18. xss实现获取网站源码
  19. lua的local问题
  20. Maximum Gap——桶排序

热门文章

  1. Dockerfile文件全面详解
  2. 【可视化大屏教程】用Python开发智慧城市数据分析大屏!
  3. Mysql 数据库SQL脚本导入
  4. Allure的简单使用
  5. Leetcode刷题笔记(双指针)
  6. NOIP2003 普及组 洛谷P1045 麦森数 (快速幂+高精度)
  7. 【YOLOv5】LabVIEW+YOLOv5快速实现实时物体识别(Object Detection)含源码
  8. 驱动开发:内核监控Register注册表回调
  9. 关于Object.keys()和Object.values()的使用
  10. Java 编码那些事(二)