简介

代理模式出场率真的相当的高,几乎所有框架中无一例外都用到了代理模式,所以了解一下收益还是很高的。

代理模式是什么

如果用一句话来描述代理模式:

代理模式就是为其他对象提供一种代理以控制对被代理对象的访问,也就是我们常说的中介

在开发以及生活中经常听到正向代理,反向代理这样的词,举例说明

  • 正向代理

    由于网络原因我们访问不了谷歌,这时候我们就需要找个梯子,替我们去访问谷歌,并且把我们需要的信息返回,这个梯子代理

  • 反向代理

    作为服务端为了安全,我们不想把实际服务器的信息暴露出去,已防止不法分子的攻击,这时候我们我需要一个代理统一接受用户的请求,并且帮助用户请求后端用户返回给用户

代理模式的作用

一言以蔽之就是解耦合,创建一个没法访问对象的代理供我们使用,同时我们又可以在代理对象中加入一些补充的功能,这样完全不会破坏封装,满足开闭原则

UML

动物有一个睡觉行为,大多数人都没法见到北极熊(RealSubject),我们只能通过动物世界节目组的摄影师(Proxy)去北极拍摄,从传回的画面中我们看到一只北极熊在洞里睡觉,并且画面上还加上了字幕“快看这里有只冬眠的北极熊!”

实践

代理模式的实现有多种方式主要分为静态代理和动态代理

静态代理

  • Subject
public interface LifeService {
String sleep();
}
  • RealSubject
public class WhiteBear implements LifeService {
@Override
public String sleep() {
return "Zzzzzzz";
}
}
  • Proxy
public class Proxy implements LifeService {

    // 被代理对象
private LifeService target; public Proxy(LifeService target) {
this.target = target;
} @Override
public String sleep() {
// 拿到被代理对象行为的返回值,加上辅助功能,一起返回
return "快看这里有只冬眠的北极熊! \n" + this.target.sleep();
}
}
  • Factory,也可以不用工厂客户端直接new
public class ProxyFactory {

    public static Proxy getLifeServiceProxy(Class clz) throws IllegalAccessException, InstantiationException {
LifeService target = (LifeService) clz.newInstance();
return new Proxy(target);
}
}
  • Client
public class Test {
public static void main(String[] args) throws IllegalAccessException, InstantiationException {
Proxy proxy = ProxyFactory.getLifeServiceProxy(WhiteBear.class);
System.out.println(proxy.sleep());
} /**
* 输出:
* 快看这里有只冬眠的北极熊!
* Zzzzzzz
*/
}

可以看到静态代理其实挺好理解的,就是我们把被代理和代理类都写好,生成两个class字节码文件, 所谓静态也就是在程序运行前就已经存在代理类的字节码文件

不足

  • 代理对象的一个接口只服务于一种类型的对象,如果要代理的方法很多,势必要为每一种方法都进行代理,静态代理在程序规模稍大时就无法胜任了。
  • 如果接口增加一个方法,除了所有实现类需要实现这个方法外,所有代理类也需要实现此方法。增加了代码维护的复杂度

动态代理

JDK自带

大致思路是在运行过程中JVM进行监控,发生指定行为时动态的创建的代理,通过反射去访问被代理对象

  • Subject
public interface LifeService {
String sleep();
String wake();
}
  • RealSubject
public class Person implements LifeService {
@Override
public String sleep() {
return "晚安晚安";
} @Override
public String wake() {
return "早鸭";
}
}
  • Proxy

    我们实现InvocationHandler这个接口,通过invoke方法去执行代理行为

public class InvocationProxy implements InvocationHandler {

    // 被监控的对象(此例中为Person类实例)
private LifeService lifeService; // 监控启动拿到需要被监控的对象
public InvocationProxy(LifeService lifeService) {
this.lifeService = lifeService;
} /**
* 监控的行为发生时,JVM会拦截到行为执行invoke
*
* @param proxy 监控对象:监控行为是否发生
* @param method 被监控的行为方法
* @param args 被监控行为方法的参数
* @return
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 因为我们拦截了行为,并且加了一些辅助行为,完成之后我们要替被拦截行为把值返回
Object result = null;
String methodName = method.getName();
if ("sleep".equals(methodName)) {
result = getTime();
result += (String) method.invoke(this.lifeService, args);
retrun result;
} else if ("wake".equals(methodName)) {
result = getTime();
result += (String) method.invoke(this.lifeService, args);
retrun result;
}
result = method.invoke(this.lifeService, args);
return result;
} // 辅助方法
private String getTime() {
return Clock.systemDefaultZone().instant().toString() + "\n";
} }
  • Factory,也可以不用工厂客户端直接new
public class ProxyFactory {

    public static LifeService getLifeServiceProxyInstance(Class clz) throws IllegalAccessException, InstantiationException {
// 创建被代理对象
LifeService target = (LifeService) clz.newInstance();
// 绑定到代理执行器中
InvocationHandler handler = new InvocationProxy(target);
// JVM层面对被代理对象进行监控,行为发生就动态创建代理对象处理
LifeService $proxy = (LifeService) Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
handler);
return $proxy;
}
}
  • Client
public class Test {
public static void main(String[] args) throws InstantiationException, IllegalAccessException, InterruptedException {
LifeService zhang = ProxyFactory.getLifeServiceProxyInstance(Person.class);
System.out.println(zhang.sleep());
System.out.println(zhang.wake());
} /**
* 输出:
* 2019-11-10T05:24:16.932Z
* 晚安晚安
* 2019-11-10T05:24:16.942Z
* 早鸭
*/
}

用了动态代理我们把所有代理需要实现的行为集中到了invoke这一个方法去执行,不要再写大量模板代码了,并且我们实际上可以在一个InvocationHandler代理多个接口

不足

  • 如果InvocationHandler中代理了两个接口,两个接口中有完全一模一样的两个方法,就没法去区分了
  • 代理必须基于接口,没有实现接口的类没法被代理

三方库Cglib

Cglib 基于 ASM 框架操作字节码帮我们生成需要的代理对象,并且不要求实现接口

加入依赖

<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.3.0</version>
</dependency>
  • RealSubject

    我们不需要实现特定接口了

public class Person {

    public String sleep() {
return "晚安晚安";
} public String wake() {
return "早鸭";
}
}
  • Proxy

    我们的逻辑和JDK自带的动态代理是一样的

public class CglibProxy implements MethodInterceptor {
//需要代理的目标对象
private Object target; public CglibProxy(Object target) {
this.target = target;
} /**
*
* @param o 监控对象:监控行为是否发生
* @param method 被监控的行为方法
* @param objects 被监控行为方法的参数
* @param methodProxy 代理中生成的方法
* @return
* @throws Throwable
*/
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
Object result = null; String methodName = method.getName();
if ("sleep".equals(methodName)) {
result = getTime();
result += (String) method.invoke(this.target, objects);
} else if ("wake".equals(methodName)) {
result = getTime();
result += (String) method.invoke(this.target, objects);
}
return result;
} // 辅助行为
private String getTime() {
return Clock.systemDefaultZone().instant().toString() + "\n";
}
}
  • Factory,也可以不用工厂客户端直接new
public class ProxyFactory {

    public static Object getCglibProxyInstance(Class clz) throws IllegalAccessException, InstantiationException {
// Enhancer类是CGLib中的一个字节码增强器
Enhancer enhancer=new Enhancer();
// 设置被代理类的字节码文件,这里我们关注的不再是接口
enhancer.setSuperclass(clz);
// 创建被代理对象
Object target = clz.newInstance();
// 绑定到代理执行器中
CglibProxy proxy = new CglibProxy(target);
// 设置回调这个代理对象
enhancer.setCallback(proxy);
// 生成返回代理对象
return enhancer.create();
}
}
  • Client
public class Test {

    public static void main(String[] args) throws InstantiationException, IllegalAccessException {
Person zhang = (Person) ProxyFactory.getCglibProxyInstance(Person.class);
System.out.println(zhang.sleep());
System.out.println(zhang.wake());
} /**
* 输出:
* 2019-11-10T06:01:13.105Z
* 晚安晚安
* 2019-11-10T06:01:13.115Z
* 早鸭
*/
}

不需要实现接口也可以动态代理啦,真的很了不起

不足

  • 依赖三方库
  • 对于final的类和方法不能代理, 因为Cglib 生成的代理类需要重写代理类中所有的方法

设计模式之代理模式(Java)的更多相关文章

  1. 软件设计模式之代理模式(JAVA)

    貌似停笔了近半个月了,实在不该啊,新的一年,时刻让自己归零. Back To Zero,就从这篇文章拉开今年的序幕吧. 这篇文章准备介绍下有关代理模式的基本概念和静态代理.动态代理的优缺点及使用方法( ...

  2. Java设计模式之代理模式(静态代理和JDK、CGLib动态代理)以及应用场景

    我做了个例子 ,需要可以下载源码:代理模式 1.前言: Spring 的AOP 面向切面编程,是通过动态代理实现的, 由两部分组成:(a) 如果有接口的话 通过 JDK 接口级别的代理 (b) 如果没 ...

  3. 设计模式之第16章-代理模式(Java实现)

    设计模式之第16章-代理模式(Java实现) “现在朋友圈真是太让人蛋疼了啊.”“怎么说?”“一堆代理,各种卖东西的,看着好烦人.”“哎,删了呗.”“都是朋友,哪里好意思删啊.”“这倒也是...哎,迫 ...

  4. 夜话JAVA设计模式之代理模式(Proxy)

    代理模式定义:为另一个对象提供一个替身或者占位符以控制对这个对象的访问.---<Head First 设计模式> 代理模式换句话说就是给某一个对象创建一个代理对象,由这个代理对象控制对原对 ...

  5. java设计模式6——代理模式

    java设计模式6--代理模式 1.代理模式介绍: 1.1.为什么要学习代理模式?因为这就是Spring Aop的底层!(SpringAop 和 SpringMvc) 1.2.代理模式的分类: 静态代 ...

  6. 设计模式之代理模式之二(Proxy)

    from://http://www.cnblogs.com/xwdreamer/archive/2012/05/23/2515306.html 设计模式之代理模式之二(Proxy)   0.前言 在前 ...

  7. GOF23设计模式之代理模式

    GOF23设计模式之代理模式 核心作用:通过代理,控制对对象的访问.可以详细控制访问某个(某类)对象的方法,在调用这个方法前做前置处理,调用这个方法后做后置处理(即:AOP的微观实现) AOP(Asp ...

  8. C#设计模式(13)——代理模式(Proxy Pattern)

    一.引言 在软件开发过程中,有些对象有时候会由于网络或其他的障碍,以至于不能够或者不能直接访问到这些对象,如果直接访问对象给系统带来不必要的复杂性,这时候可以在客户端和目标对象之间增加一层中间层,让代 ...

  9. 乐在其中设计模式(C#) - 代理模式(Proxy Pattern)

    原文:乐在其中设计模式(C#) - 代理模式(Proxy Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 代理模式(Proxy Pattern) 作者:webabcd 介绍 为 ...

  10. C#设计模式:代理模式(Proxy Pattern)

    一,什么是C#设计模式? 代理模式(Proxy Pattern):为其他对象提供一种代理以控制对这个对象的访问 二,代码如下: using System; using System.Collectio ...

随机推荐

  1. [Python] Python 学习记录(2)

    1.range(x,y) [x,y) >>> range(0,4) #0,1,2,3 >>> range(1,4) #1,2,3 2.dics dics.get(k ...

  2. Scala Try Catch Finally

    Scala Try Catch Finally: 在Java中返回值优先级顺序:finally最高, try,catch 选其一,try中抛异常,返回catch,不抛异常,返回try,. public ...

  3. IO复用,AIO,BIO,NIO,同步,异步,阻塞和非阻塞 区别

    一.什么是socket?什么是I/O操作? 我们都知道unix(like)世界里,一切皆文件,而文件是什么呢?文件就是一串二进制流而已,不管socket,还是FIFO.管道.终端,对我们来说,一切都是 ...

  4. Nginx常用命令,解决你日常运维的烦恼

    前面,跟大家简单地介绍了负载均衡和Nginx的一些基础配置(Nginx负载均衡配置实例),接下来,跟大家介绍一下Nginx的常用命令,便于日常的运维. 查看原文 停止Nginx的方法 通过之前的学习, ...

  5. 分库分表(5) ---SpringBoot + ShardingSphere 实现分库分表

    分库分表(5)--- ShardingSphere实现分库分表 有关分库分表前面写了四篇博客: 1.分库分表(1) --- 理论 2.分库分表(2) --- ShardingSphere(理论) 3. ...

  6. Windows下如何调试驱动程序

    Windows内核分析索引目录:https://www.cnblogs.com/onetrainee/p/11675224.html 一.配置Windbg使用双机调试 win10中“windbg+vm ...

  7. Java 多线程爬虫及分布式爬虫架构探索

    这是 Java 爬虫系列博文的第五篇,在上一篇 Java 爬虫服务器被屏蔽,不要慌,咱们换一台服务器 中,我们简单的聊反爬虫策略和反反爬虫方法,主要针对的是 IP 被封及其对应办法.前面几篇文章我们把 ...

  8. 02-15 Logistic回归(鸢尾花分类)

    目录 Logistic回归(鸢尾花分类) 一.导入模块 二.获取数据 三.构建决策边界 四.训练模型 4.1 C参数与权重系数的关系 五.可视化 更新.更全的<机器学习>的更新网站,更有p ...

  9. C#通过对象属性名修改值

    摘自:csdn 给一个对象属性赋值可以通过PropertyInfo.SetValue()方式进行赋值,但要注意值的类型要与属性保持一致.    创建对象实例的两种方法: 1. var obj = As ...

  10. [ERROR ImagePull]: failed to pull image [k8s.gcr.io/kube-apiserver-amd64:v1.11.1]: exit status 1

    问题描述 [root@localhost ~]# kubeadm init --kubernetes-version=v1.11.1 --pod-network-cidr=10.244.0.0/16 ...