代理模式的定义:

代理模式是一个使用律非常高的模式,定义如下:

为其他对象提供一种代理,以控制对这个对象的访问。

类图:

简单的静态代理:

public interface IRunner{
//这是一个代理类和被代理类都需要实现的接口
//在接口中定义一个抽象函数
public void request();
} //下面是真实的被代理类
public class Runner implements IRunner{
//实现接口中的方法
public void request(){
//实现具体的业务逻辑
}
}
//下面是代理类
public class RunnerAgent implements IRunner{
//要代理哪个实现类
private IRunner runner = null;
//默认被代理者
public RunnerAgent(IRunner runner){
this.runner = runner;
}
//实现接口中定义的方法
public void request(){
this.before();
this.runner.request();
this.after();
}
//预处理
private void before(){
}
//后处理
private void after(){
} }
  • 我们可以看到静态代理只能代理某一个类,每个类都有一个代理类。在对通用性要求更高的场景下(比如Spring AOP)传统的静态代理不能满足需求,会使类的数目非常庞大,因此又有了动态代理。

动态代理

什么是动态代理?动态代理就是在实现阶段不用关心代理谁,而在运行阶段才指定代理哪一个对象。动态代理的一个重要应用就是面向切面编程(AOP)

AOP示意图:

通过示意图可以明显的看出来,AOP抽取出非业务逻辑代码,降低了耦合度。对于日志、事务、权限等都可以在系统设计阶段不用考虑,而在设计后通过AOP的方式织入。

动态代理的通用类图(jdk动态代理,如无特殊声明,以下动态代理均指jdk动态代理)

图中类和接口的代码:

//Subject接口
public interface Subject{
//业务操作
public void doSomething(String str);
} //真实主题RealSubject类
public class RealSubject implements Subject{
//业务操作
public void doSomething(String str){
System.out.println("do something!--->" + str);
}
} //动态代理的Handler类
public class MyInvocationHandler implements InvocationHandler{
//被代理对象
private Object target = null;
//构造函数
public MyInvocationHandler(Object _obj){
this.target = _obj;
}
//代理方法 覆盖接口InvocationHandler中的invoke方法
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable{
//执行被代理的方法
return method.invoke(this.target, args);
} } //动态代理类
public class DynamicProxy<T>{
public static <T> T newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) {
if(true) {
//执行一个前置通知
(new BeforeAdvice()).exec();
}
//执行目标函数
return (T) Proxy.newProxyInstance(loader, interfaces, h);
}
} //通知接口及其实现
public interface IAdvice{
//通知方法
public void exec();
}
public class BeforeAdvice implements IAdvice{
public void exec(){
System.out.println("我是前置通知,我被执行了!");
}
} //动态代理场景类
public class Client{
public static void main(String[] args){
//定义一个主题
Subject subject = new RealSubject();
//定义一个Handler
InvocationHandler handler = new MyInvocationHandler(subject);
//定义主题的代理
Subject proxy = DynamicProxy.newProxyInstance(subject.getClass().
getClassLoader(), subject.getClass.getInterfaces(),handler);
//代理行为
proxy.doSomething("Finish");
}
}

运行结果:

我是前置通知,我被执行了!
do something!--->Finish

其中有这样的一行代码:return (T) Proxy.newProxyInstance(loader, interfaces, h);

这里使用interfaces指出了被代理对象实现的接口中的方法,因此jdk动态代理就要求被代理类必须实现接口。

h是接口InvocationHandler的实现类MyInvocationHandler的实例。由h对象的invoke方法负责接管被代理类实现的接口中的方法。

在运行时,只需通过subject.getClass.getInterfaces()传参,通过反射,达到了解耦的目的。

  • jdk动态代理 vs cglib动态代理

    jdk动态代理是以接口为中心的,相当于添加类一种对于被调用者没有太大意义的限制。我们实例化的是代理对象,而不是真正被调用的类型,这在实践中可能带来不便。

    如果被调用者没有实现接口,而我们还是希望利用动态代理机制,那么可以考虑其他方式。我们知道Spring AOP支持两种模式的动态代理:jdk动态代理和cglib。如果我们选择cglib方式,就不再依赖接口。

    cglib动态代理采取的方式是创建被代理类的子类。因此要求被代理类不可以用final修饰。

    jdk动态代理的优势:

    最小化依赖关系

    代码实现简单。

    cglib代理优势:

    不需要被代理类实现额外接口。

    只操作我们关心的类,而不必为其他相关类增加工作量。

    高性能。

代理模式的优点

  • 职责清晰

    真实的角色就是实现实际的业务逻辑,不用关心其他非本职责的事务,通过后期的代理完成一件事务,附带的结果就是编程简洁清晰。
  • 高扩展性

    具体主题角色是随时都会发生变化的。只要实现了接口,不管如何变化,都离不开接口的规范,那我们的代理类完全就可以在不做任何修改的情况下使用。

参考文献:设计模式之禅 秦小波 著

浅谈代理模式与java中的动态代理的更多相关文章

  1. 代理模式与java中的动态代理

    前言    代理模式又分为静态代理与动态代理,其中动态代理是Java各大框架中运用的最为广泛的一种模式之一,下面就用简单的例子来说明静态代理与动态代理. 场景    李雷是一个唱片公司的大老板,很忙, ...

  2. 10分钟了解 代理模式与java中的动态代理

    前言    代理模式又分为静态代理与动态代理,其中动态代理是Java各大框架中运用的最为广泛的一种模式之一,下面就用简单的例子来说明静态代理与动态代理. 场景    李雷是一个唱片公司的大老板,很忙, ...

  3. 十分钟理解Java中的动态代理

    十分钟理解 Java 中的动态代理   一.概述 1. 什么是代理 我们大家都知道微商代理,简单地说就是代替厂家卖商品,厂家“委托”代理为其销售商品.关于微商代理,首先我们从他们那里买东西时通常不知道 ...

  4. 深度剖析java中JDK动态代理机制

    https://www.jb51.net/article/110342.htm 本篇文章主要介绍了深度剖析java中JDK动态代理机制 ,动态代理避免了开发人员编写各个繁锁的静态代理类,只需简单地指定 ...

  5. 一文读懂Java中的动态代理

    从代理模式说起 回顾前文: 设计模式系列之代理模式(Proxy Pattern) 要读懂动态代理,应从代理模式说起.而实现代理模式,常见有下面两种实现: (1) 代理类关联目标对象,实现目标对象实现的 ...

  6. 使用Java中的动态代理实现数据库连接池

    2002 年 12 月 05 日 作者通过使用JAVA中的动态代理实现数据库连接池,使使用者可以以普通的jdbc连接的使用习惯来使用连接池. 数据库连接池在编写应用服务是经常需要用到的模块,太过频繁的 ...

  7. java中的动态代理机制

    java中的动态代理机制 在java的动态代理机制中,有两个重要的类或接口,一个是 InvocationHandler(Interface).另一个则是 Proxy(Class),这一个类和接口是实现 ...

  8. Java 中的动态代理

    一.概述 1. 什么是代理 我们大家都知道微商代理,简单地说就是代替厂家卖商品,厂家“委托”代理为其销售商品.关于微商代理,首先我们从他们那里买东西时通常不知道背后的厂家究竟是谁,也就是说,“委托者” ...

  9. Java中的动态代理(jdk和cglib)

    JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类 ...

随机推荐

  1. Redis常用命令(1)——Key

    DEL 格式:DEL key [key ...] 作用:删除一个或多个 key.不存在的 key 会被忽略. 返回值:被删除 key 的数量. 示例: 192.168.1.100:6379> s ...

  2. FastCGI协议分析

    不知道什么时候,就开始有了让HomeServer支持PHP的念头.于是分析起了FastCGI协议.FastCGI用于WebServer与WebApplication之间的通讯,例如Apache与PHP ...

  3. python爬虫中的requests模块

    Requests: 让 HTTP 服务人类 一.简介 虽然Python的标准库中 urllib 模块已经包含了平常我们使用的大多数功能,但是它的 API 使用起来让人感觉不太好,而 Requests ...

  4. 快进来!花几分钟看一下 ReentrantReadWriteLock 的原理!

    前言 在看完 ReentrantLock 之后,在高并发场景下 ReentrantLock 已经足够使用,但是因为 ReentrantLock 是独占锁,同时只有一个线程可以获取该锁,而很多应用场景都 ...

  5. C# 集合类(三)

    C# 集合类自己经常用到: 数组(Array).动态数组(ArrayList).列表(List).哈希表(Hashtable).字典(Dictionary),对于经常使用的这些数据结构,做一个总结,便 ...

  6. 利用命令行将本地代码上传至GitHub仓库

    p.p1 { margin: 0 0 0 60px; font: 16px "PingFang SC"; color: rgba(88, 110, 117, 1) } p.p2 { ...

  7. 准备数据集用于flink学习

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  8. 批处理最小二乘法 python

    参考:系统辨识与自适应控制MATLAB仿真(修订版) 庞中华 崔红 仿真实例2.5 import numpy as np import matplotlib.pyplot as plt from mx ...

  9. 探索 .Net Core 的 SourceLink

    介绍 Source Link是一组软件包和一个规范, 它将一些元数据添加到PDB文件,以将本地文件重新映射到GitHub上的文件,因此Visual Studio可以在这需要时下载文件, 该项目的目的是 ...

  10. Netlink 内核实现分析 1

    Netlink 是一种IPC(Inter Process Commumicate)机制,它是一种用于内核与用户空间通信的机制,在一般情况下,用户态和内核态通信会使用传统的Ioctl.sysfs属性文件 ...