Java的两大、三类代理模式
简述
代理,是一种设计模式,主要作用是为其他对象提供一种代理,以控制对这个对象的访问。在某些情况下,一个对象不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。
主要分为两大代理:静态代理、动态代理(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、代理类和委托类的关系是在程序运行时确定
Java的两大、三类代理模式的更多相关文章
- Java设计模式(10)代理模式(Proxy模式)
理解并使用设计模式,能够培养我们良好的面向对象编程习惯,同时在实际应用中,可以如鱼得水,享受游刃有余的乐趣. Proxy是比较有用途的一种模式,而且变种较多,应用场合覆盖从小结构到整个系统的大结构,P ...
- Java的两大数据类型
Java的两大数据类型 基本数据类型 byte,short,int,long,float,double,boolean,char byte 类别 内容 类型 byte 简介 byte 数据类型是8位. ...
- JAVA设计模式——第 2 章 代理模式【Proxy Pattern】(转)
什么是代理模式呢?我很忙,忙的没空理你,那你要找我呢就先找我的代理人吧,那代理人总要知道被代理人能做哪些事情不能做哪些事情吧,那就是两个人具备同一个接口,代理人虽然不能干活,但是被代理的人能干活呀. ...
- 浅谈Java五大设计原则之代理模式
我们来定义一下 AOP(面向切面编程) 它是面向对象的一种补充或者是一种增强,它在这基础上增加了一些 而外的功能增强. 它可以在原有的行为不改变的前提,在这之前或者之后完成一些而外 的事情. 而AO ...
- Java设计模式系列之动态代理模式(转载)
代理设计模式 定义:为其他对象提供一种代理以控制对这个对象的访问. 动态代理使用 java动态代理机制以巧妙的方式实现了代理模式的设计理念. 代理模式示例代码 public interface Sub ...
- Java设计模式---(动态)代理模式
代理设计模式 定义:为其他对象提供一种代理以控制对这个对象的访问. 动态代理使用 java动态代理机制以巧妙的方式实现了代理模式的设计理念. 之前虽然会用JDK的动态代理,但是有些问题却一直没有搞明白 ...
- Java设计模式(八)----代理模式
代理模式 1.生活中: 代理就是一个人或者一个组织代表其它人去做一件事的现实生活中的. 在一些情况下,一个客户不想或者不能够直接引用一个对象,而代理对象能够在client和目标对象之间起到中介的作用. ...
- Java内功修炼系列一代理模式
代理模式是JAVA设计模式之一,网上设计模式相关的博文铺天盖地,参考它们有助于自己理解,但是所谓“尽信书不如无书”,在参考的同时也要思考其正确性,写博客也是为了记录自己理解知识点的思路历程和心路历程, ...
- Java学习笔记——设计模式之四.代理模式
To be, or not to be: that is the question. --<哈姆雷特> 代理模式(Proxy),为其他对象提供一种代理以控制对这个对象的访问. 上代码: p ...
- 详解 Java 中的三种代理模式
代理模式 代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式;即通过代理对象访问目标对象.这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能. 这里使用 ...
随机推荐
- Linux恢复误删除的文件或者目录
文章转载自:https://www.jianshu.com/p/662293f12a47 linux不像windows有个回收站,使用rm -rf *基本上文件是找不回来的. 那么问题来了: 对于li ...
- 密码学奇妙之旅、02 混合加密系统、AES、RSA标准、Golang代码
CTR 计数器模式 计数器模式CTR是分组密码模式中的一种.通过将逐次累加的计数器进行加密来生成密钥流的流密码.每次加密时会生成一个不同的值来作为计数器的初始值. 可以事先进行加密.解密的准备. 加密 ...
- Python抖音视频去水印,并打包成exe可执行文件
前言 抖音里面的视频保存之后,会发现全都带有水印,所以如何解决视频去除水印就很有必要,所以教程来了,本次教程不仅会教大家如何去除视频里的水印,并且教大家将程序制作成exe可执行文件,可以发给你的好友使 ...
- 『现学现忘』Git后悔药 — 33、revert撤销(二)
目录 4.一次移除某几次提交 (1)git revert移除某几次提交的修改 (2)git revert 移除某几次连续的提交的修改 5.revert命令常用参数 6.git revert和git r ...
- 常用cmd及bat脚本命令
1.内部命令和外部命令 cmd 命令 :内部命令和外部命令 内部命令 系统自带命令 dir copy 外部命令 调用应用程序,可自由拓展 mstsc.exe(mstsc)远程连接 ping (ping ...
- 后端框架学习-----mybatis(4)
文章目录 4.解决属性名和字段名不一致的问题 4.解决属性名和字段名不一致的问题 1.问题.数据库字段名和属性名不一致,导致查出的数据部分为空 2.resultMap(用于解决数据库表中的字段和属性) ...
- C语言基础--数组
数组 概念:在内存中连续存储的具有相同数据类型的一组数据的集合. 注意: 数组中的数据类型必须都是一致的 数组在内存中必须是连续的存储空间 定义数组时候的注意事项: 定义数组的时候,[]里面的值不能是 ...
- Java多线程-线程关键字(二)
Java中和线程相关的关键字就两:volatile和synchronized. volatile以前用得较少,以后会用得更少(后面解释).它是一种非常轻量级的同步机制,它的三大特性是: 1.保证可见性 ...
- Java使用lamda表达式简化代码
代码,自然写的越简洁越好啦,写的人舒服,看的人也舒服,一切为了高效. 要把有限的时间花到其它有意思的事情上去. 目的 学习简化代码的思路,使用jdk8新特性lamada表达式. 使用 某接口,只有一个 ...
- 如何把Java代码玩出花?JVM Sandbox入门教程与原理浅谈
在日常业务代码开发中,我们经常接触到AOP,比如熟知的Spring AOP.我们用它来做业务切面,比如登录校验,日志记录,性能监控,全局过滤器等.但Spring AOP有一个局限性,并不是所有的类都托 ...