热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

spring学习之代理模式引入

前言在学习springAOP编程之前,必须要掌握的java设计模式之代理模式。代理模式代理(Proxy)是一种设计模式,

前言

在学习spring AOP编程之前,必须要掌握的java设计模式之代理模式。


代理模式

代理(Proxy)是一种设计模式, 提供了对目标对象另外的访问方式;即通过代理访问目标对象。 这样的好处是: 可以在目标对象实现的基础上,增强额外的功能操作。(扩展目标对象的功能)。

如下图所示:
代理模式

代理模式的关键点: 代理对象与目标对象。代理对象能访问目标对象,是对目标对象的功能扩展。而用户通过访问代理对象来实现目标对象的一些方法。

那么问题来了,之前学过装饰者模式,自己再学代理模式的时候发现二者之间很类似。为了弄清楚具体的区别,也是查阅了相关的资料。这里只是点一下二者区别,以免新手误解其思想,日后在设计模式那一章,单独再去介绍。


代理模式和装饰者模式的区别

二者的代码很相似。

二者最主要的区别是:代理模式中,代理类对被代理的对象有控制权,决定其执行或者不执行。而装饰模式中,装饰类对代理对象没有控制权,只能为其增加一层装饰,以加强被装饰对象的功能,仅此而已。
我理解的就是比如:一个运动员,装饰者模式可以对运动员增加一双跑得快的鞋子,增加一个加速器,可以对远动员跑步的功能进行增强。而代理模式只是经纪人可以控制运动员要不要跑步,如果今天下雨可以不跑步,或者在跑步之前和之后给你加油,额外的条件(功能),并不会改变运动员本身跑步的功能。

二者的应用场景
代理模式使用到极致开发就是AOP, 这是各位采用Spring架构开发必然要使用到的技术,它就是使用了代理和反射的技术。代理模式在Java的开发中俯拾皆是, 是大家非常熟悉的模式, 应用非常广泛, 而装饰模式是一个比较拘谨的模式, 在实际应用中接触比较少, 但是也有不少框架项目使用了装饰模式, 例如在JDK的java.io.*包中就大量使用装饰模式, 类似如下的代码:

OutputStream out = new DataOutputStream( new FileOutputStream( "test.txt") )

  这是装饰模式的一个典型应用, 使用DataOutputStream封装了一个FileOutputStream, 以方便进行输出流处理。好吧,区别先说这么多,我们回过头来继续看代理模式。


静态代理

静态代理在使用时,需要定义接口或者父类,被代理对象与代理对象一起实现相同的接口或者是继承相同父类.

举例:
保存用户(模拟)
Dao , 直接保存
DaoProxy, 创建代理对象,给保存方法添加事务处理

代码示例:
接口:

// 接口
public interface IUserDao {void save();
}

目标对象实现接口

public class UserDao implements IUserDao{@Overridepublic void save() {System.out.println("-----已经保存数据!!!------");}}

代理对象实现接口,并扩展目标对象的功能(添加事务处理)

public class UserDaoProxy implements IUserDao{// 接收保存目标对象private IUserDao target;public UserDaoProxy(IUserDao target) {this.target = target;}@Overridepublic void save() {System.out.println("开始事务...");target.save(); // 执行目标对象的方法System.out.println("提交事务...");}

测试,通过调用代理对象的方法,实现对目标对象保存的扩展。

public class App {public static void main(String[] args) {// 目标对象IUserDao target = new UserDao();// 代理IUserDao proxy = new UserDaoProxy(target);proxy.save(); // 执行的是,代理的方法}
}

总结:
静态代理可以做到在不修改目标对象的功能的前提下,对目标对象功能进行扩展。这也符合面向对象设计原则之开闭原则:对扩展开发,对修改关闭。
缺点:
(1)需要定义太多代理类
因代理对象需要和目标对象实现相同的接口。所以会有很多的代理类需要定义。
(2)不方便维护
因为一但用户定义接口需要增加额外的方法,目标对象与代理对象都要修改,代码写死,维护成本过高。

因此解决办法就是代理工厂,使用动态代理的方式。


动态代理

动态代理概念
1)代理对象不需要实现接口,但是目标对象必须实现接口,因为要指定接口类型;
2)代理对象的生成,是利用JDKAPI, 动态的在内存中构建代理对象(需要我们指定创建 代理对象/目标对象 实现的接口的类;);
3) 动态代理也叫做JDK代理, 接口代理;

JDK中生成代理对象的API:

|-- Proxy
static Object newProxyInstance(ClassLoader loader, 指定当前目标对象使用的类加载器Class[] interfaces, 目标对象实现的接口的类型InvocationHandler h 事件处理器
)

代码示例:
接口类IUserDao.java以及接口实现类,目标对象UserDao是一样的,没有做修改.在这个基础上,增加一个代理工厂类(ProxyFactory.java),将代理类写在这个地方,然后在测试类(需要使用到代理的代码)中先建立目标对象和代理对象的联系,然后代用代理对象的中同名方法。

代理工厂类:ProxyFactory.java

public class ProxyFactory {// 维护一个目标对象private Object target;public ProxyFactory(Object target){this.target = target;}// 给目标对象,生成代理对象 public Object getProxyInstance() {return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(),new InvocationHandler() {@Overridepublic Object invoke(Object proxy, Method method, Object[] args)throws Throwable {System.out.println("开启事务");// 执行目标对象方法Object returnValue = method.invoke(target, args);System.out.println("提交事务");return returnValue;}});}
}

测试App.java

public class App {public static void main(String[] args) {// 目标对象IUserDao target = new UserDao();// 【原始的类型 class com.nwpu.geeker.UserDao】System.out.println(target.getClass());// 给目标对象,创建代理对象IUserDao proxy = (IUserDao) new ProxyFactory(target).getProxyInstance();// 类型:class $Proxy0 内存中动态生成的代理对象System.out.println(proxy.getClass());// 执行方法 【代理对象】proxy.save();}
}

动态代理总结:
代理对象不需要实现接口,但是目标对象一定要实现接口;否则不能用动态代理!

思考:
有一个目标对象,想要功能进行扩展,但目标对象没有实现接口,怎样功能扩展?

Class UserDao{}// 通过子类的方式Class subclass extends UserDao{}

使用cglib代理。通过运行时期创建子类扩展目标对象的方法。


cglib代理

Cglib代理,也叫做子类代理。在内存中构建一个子类对象从而实现对目标对象功能的扩展。

CGLIB是一个强大的高性能的代码生成包,它可以在运行期扩展Java类与实现Java接口。它广泛的被许多AOP的框架使用,例如Spring AOP和dynaop,为他们提供方法的interception(拦截)。

CGLIB包的底层是通过使用一个小而快的字节码处理框架ASM,来转换字节码并生成新的类。不鼓励直接使用ASM,因为它要求你必须对JVM内部结构包括class文件的格式和指令集都很熟悉。

代码示例:

public class UserDao {public void save() {System.out.println("-----已经保存数据!!!------");}}

cglib代理实现

import java.lang.reflect.Method;import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;/*** Cglib子类代理工厂* (对UserDao 在内存中动态构建一个子类对象)*/
public class ProxyFactory implements MethodInterceptor{// 维护目标对象private Object target;public ProxyFactory(Object target){this.target = target;}// 给目标对象创建代理对象public Object getProxyInstance(){//1. 工具类Enhancer en = new Enhancer();//2. 设置父类en.setSuperclass(target.getClass());//3. 设置回调函数en.setCallback(this);//4. 创建子类(代理对象)return en.create();}@Overridepublic Object intercept(Object obj, Method method, Object[] args,MethodProxy proxy) throws Throwable {System.out.println("开启事务.....");// 执行目标对象的方法Object returnValue = method.invoke(target, args);System.out.println("提交事务.....");return returnValue;}}

Cglib子类代理:
1) 需要引入cglib – jar文件, 但是spring的核心包中已经包括了cglib功能,所以直接引入spring-core-3.2.5.jar即可。
2)引入功能包后,就可以在内存中动态构建子类
3)代理的类不能为final, 否则报错。
4) 目标对象的方法如果为final/static, 那么就不会被拦截,即不会执行目标对象额外的业务方法。

总结:
在Spring的AOP编程中,
如果加入容器的目标对象有实现接口,用JDK代理;
如果目标对象没有实现接口,用Cglib代理;


推荐阅读
  • 使用 ListView 浏览安卓系统中的回收站文件 ... [详细]
  • 本文探讨了如何利用Java代码获取当前本地操作系统中正在运行的进程列表及其详细信息。通过引入必要的包和类,开发者可以轻松地实现这一功能,为系统监控和管理提供有力支持。示例代码展示了具体实现方法,适用于需要了解系统进程状态的开发人员。 ... [详细]
  • 使用Maven JAR插件将单个或多个文件及其依赖项合并为一个可引用的JAR包
    本文介绍了如何利用Maven中的maven-assembly-plugin插件将单个或多个Java文件及其依赖项打包成一个可引用的JAR文件。首先,需要创建一个新的Maven项目,并将待打包的Java文件复制到该项目中。通过配置maven-assembly-plugin,可以实现将所有文件及其依赖项合并为一个独立的JAR包,方便在其他项目中引用和使用。此外,该方法还支持自定义装配描述符,以满足不同场景下的需求。 ... [详细]
  • 深入剖析Java中SimpleDateFormat在多线程环境下的潜在风险与解决方案
    深入剖析Java中SimpleDateFormat在多线程环境下的潜在风险与解决方案 ... [详细]
  • 本指南介绍了如何在ASP.NET Web应用程序中利用C#和JavaScript实现基于指纹识别的登录系统。通过集成指纹识别技术,用户无需输入传统的登录ID即可完成身份验证,从而提升用户体验和安全性。我们将详细探讨如何配置和部署这一功能,确保系统的稳定性和可靠性。 ... [详细]
  • 优化后的标题:深入探讨网关安全:将微服务升级为OAuth2资源服务器的最佳实践
    本文深入探讨了如何将微服务升级为OAuth2资源服务器,以订单服务为例,详细介绍了在POM文件中添加 `spring-cloud-starter-oauth2` 依赖,并配置Spring Security以实现对微服务的保护。通过这一过程,不仅增强了系统的安全性,还提高了资源访问的可控性和灵活性。文章还讨论了最佳实践,包括如何配置OAuth2客户端和资源服务器,以及如何处理常见的安全问题和错误。 ... [详细]
  • C++ 异步编程中获取线程执行结果的方法与技巧及其在前端开发中的应用探讨
    本文探讨了C++异步编程中获取线程执行结果的方法与技巧,并深入分析了这些技术在前端开发中的应用。通过对比不同的异步编程模型,本文详细介绍了如何高效地处理多线程任务,确保程序的稳定性和性能。同时,文章还结合实际案例,展示了这些方法在前端异步编程中的具体实现和优化策略。 ... [详细]
  • 在使用 Qt 进行 YUV420 图像渲染时,由于 Qt 本身不支持直接绘制 YUV 数据,因此需要借助 QOpenGLWidget 和 OpenGL 技术来实现。通过继承 QOpenGLWidget 类并重写其绘图方法,可以利用 GPU 的高效渲染能力,实现高质量的 YUV420 图像显示。此外,这种方法还能显著提高图像处理的性能和流畅性。 ... [详细]
  • 在Java Web服务开发中,Apache CXF 和 Axis2 是两个广泛使用的框架。CXF 由于其与 Spring 框架的无缝集成能力,以及更简便的部署方式,成为了许多开发者的首选。本文将详细介绍如何使用 CXF 框架进行 Web 服务的开发,包括环境搭建、服务发布和客户端调用等关键步骤,为开发者提供一个全面的实践指南。 ... [详细]
  • 在多年使用Java 8进行新应用开发和现有应用迁移的过程中,我总结了一些非常实用的技术技巧。虽然我不赞同“最佳实践”这一术语,因为它可能暗示了通用的解决方案,但这些技巧在实际项目中确实能够显著提升开发效率和代码质量。本文将深入解析并探讨这四大高级技巧的具体应用,帮助开发者更好地利用Java 8的强大功能。 ... [详细]
  • 分享一款基于Java开发的经典贪吃蛇游戏实现
    本文介绍了一款使用Java语言开发的经典贪吃蛇游戏的实现。游戏主要由两个核心类组成:`GameFrame` 和 `GamePanel`。`GameFrame` 类负责设置游戏窗口的标题、关闭按钮以及是否允许调整窗口大小,并初始化数据模型以支持绘制操作。`GamePanel` 类则负责管理游戏中的蛇和苹果的逻辑与渲染,确保游戏的流畅运行和良好的用户体验。 ... [详细]
  • QT框架中事件循环机制及事件分发类详解
    在QT框架中,QCoreApplication类作为事件循环的核心组件,为应用程序提供了基础的事件处理机制。该类继承自QObject,负责管理和调度各种事件,确保程序能够响应用户操作和其他系统事件。通过事件循环,QCoreApplication实现了高效的事件分发和处理,使得应用程序能够保持流畅的运行状态。此外,QCoreApplication还提供了多种方法和信号槽机制,方便开发者进行事件的定制和扩展。 ... [详细]
  • Java Socket 关键参数详解与优化建议
    Java Socket 的 API 虽然被广泛使用,但其关键参数的用途却鲜为人知。本文详细解析了 Java Socket 中的重要参数,如 backlog 参数,它用于控制服务器等待连接请求的队列长度。此外,还探讨了其他参数如 SO_TIMEOUT、SO_REUSEADDR 等的配置方法及其对性能的影响,并提供了优化建议,帮助开发者提升网络通信的稳定性和效率。 ... [详细]
  • 本文详细介绍了 Java 中遍历 Map 对象的几种常见方法及其应用场景。首先,通过 `entrySet` 方法结合增强型 for 循环进行遍历是最常用的方式,适用于需要同时访问键和值的场景。此外,还探讨了使用 `keySet` 和 `values` 方法分别遍历键和值的技巧,以及使用迭代器(Iterator)进行更灵活的遍历操作。每种方法都附有示例代码和具体的应用实例,帮助开发者更好地理解和选择合适的遍历策略。 ... [详细]
  • 在Java基础中,私有静态内部类是一种常见的设计模式,主要用于防止外部类的直接调用或实例化。这种内部类仅服务于其所属的外部类,确保了代码的封装性和安全性。通过分析JDK源码,我们可以发现许多常用类中都包含了私有静态内部类,这些内部类虽然功能强大,但其复杂性往往让人感到困惑。本文将深入探讨私有静态内部类的作用、实现方式及其在实际开发中的应用,帮助读者更好地理解和使用这一重要的编程技巧。 ... [详细]
author-avatar
高正_飞翔之殇_826
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有