版权声明:本文为Fighter168原创文章,未经允许不得转载。

 

目录(?)[+]

 

说到代理,脑袋中浮现一大堆代理相关的名词,代理模式,静态代理,jdk代理,cglib代理等等。

记忆特别深刻的是,一次面试中,一位面试官问我,spring的AOP核心采用的设计模式是什么什么模式,阅读过24种设计模式,以及阅读过spring源代码的我竟然答错了,回想起来,真是日了狗了,学过那么多遍的东西都忘记了,结果是装逼失败,自己要狠下心来,把代理都搞懂!

代理模式简述

代理模式是常用的Java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方法,来提供特定的服务。

静态代理

由程序员创建或特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。下面通过例子展示。

定义接口

  1. /**
  2. * 定义一个账户接口
  3. *
  4. * @author Fighter
  5. * @date 2016-04-20
  6. *
  7. */
  8. public interface Count {
  9. // 查看账户方法
  10. public void queryCount();
  11. // 修改账户方法
  12. public void updateCount();
  13. }

实现接口

  1. /**
  2. * 委托类(包含业务逻辑)
  3. *
  4. * @author Fighter
  5. * @date 2016-04-20
  6. *
  7. */
  8. public class CountImpl implements Count {
  9. @Override
  10. public void queryCount() {
  11. System.out.println("查看账户方法...");
  12. }
  13. @Override
  14. public void updateCount() {
  15. System.out.println("修改账户方法...");
  16. }
  17. }

添加代理

  1. /**
  2. * 这是一个代理类(增强CountImpl实现类)
  3. *
  4. * @author Fighter
  5. * @date 2016-04-20
  6. *
  7. */
  8. public class CountProxy implements Count {
  9. private CountImpl countImpl;
  10. /**
  11. * 覆盖默认构造器
  12. *
  13. * @param countImpl
  14. */
  15. public CountProxy(CountImpl countImpl) {
  16. this.countImpl = countImpl;
  17. }
  18. @Override
  19. public void queryCount() {
  20. System.out.println("事务处理之前");
  21. // 调用委托类的方法;
  22. countImpl.queryCount();
  23. System.out.println("事务处理之后");
  24. }
  25. @Override
  26. public void updateCount() {
  27. System.out.println("事务处理之前");
  28. // 调用委托类的方法;
  29. countImpl.updateCount();
  30. System.out.println("事务处理之后");
  31. }
  32. }

测试

  1. /**
  2. *测试Count类
  3. *
  4. * @author Fighter
  5. * @date 2016-04-20
  6. *
  7. */
  8. public class TestCount {
  9. public static void main(String[] args) {
  10. CountImpl countImpl = new CountImpl();
  11. CountProxy countProxy = new CountProxy(countImpl);
  12. countProxy.updateCount();
  13. countProxy.queryCount();
  14. }
  15. }

JDK动态代理

特点:只能对实现了接口的类生产代理,不能针对类

定义接口

  1. /**
  2. * 创建业务接口,包含业务可以提供对外的接口
  3. *
  4. * @author Fighter
  5. * @date 2016-04-19
  6. *
  7. */
  8. public interface UserService{
  9. /**
  10. * 目标方法
  11. */
  12. public void add();
  13. }

定义实现类

  1. /**
  2. * 创建业务接口实现类
  3. *
  4. * @author Fighter
  5. * @date 2016-04-19
  6. *
  7. */
  8. public class UserServiceImpl implements UserService{
  9. @Override
  10. public void add() {
  11. System.out.println("----------add----------");
  12. }
  13. }

定义代理

  1. /**
  2. * 自定义简单的Invocation,对接口提供的方法进行增强
  3. *
  4. * @author Fighter
  5. * @date 2016-04-19
  6. */
  7. public class MyInvocationHandler implements InvocationHandler {
  8. //目标对象
  9. private Object target;
  10. /**
  11. * 构造方法
  12. * @param target 目标对象
  13. */
  14. public MyInvocationHandler(Object target) {
  15. super();
  16. this.target=target;
  17. }
  18. <span style="white-space:pre">    </span>/**
  19. * 执行目标对象的方法
  20. */
  21. public Object invoke(Object proxy, Method method, Object[] args)
  22. throws Throwable {
  23. //在目标方法执行前简单打印一下
  24. System.out.println("----------before----------");
  25. //执行目标方法对象
  26. Object result=method.invoke(target, args);
  27. //在目标方法执行之后简单打印一下
  28. System.out.println("----------after----------");
  29. return result;
  30. }
  31. /**
  32. * 获取目标对象的代理对象
  33. * @return 代理对象
  34. */
  35. public Object getProxy(){
  36. return Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),
  37. this.target.getClass().getInterfaces(),this);
  38. }
  39. }

jdk动态代理测试

  1. public class ProxyTest{
  2. @Test
  3. public void testProxy() throws Throwable{
  4. //实例化目标对象
  5. UserService userService=new UserServiceImpl();
  6. //实例化Invocation
  7. MyInvocationHandler invocationHandler=new MyInvocationHandler(userService);
  8. //根据目标生成代理对象
  9. UserService proxy=(UserService)invocationHandler.getProxy();
  10. //调用代理对象方法
  11. proxy.add();
  12. }
  13. }

CGLIB动态代理示例


JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。 

CGLIB是一个强大的高性能的代码生成包。被广泛的许多AOP框架使用,如Spring的AOP和dynaop,为他们提供方法的interceptor(拦截),最流行的是OR Mapping工具hibernate也是使用CGLIB来代理单端的single-ended(多对一和一对一)关联(对集合的延迟抓取是采用其他机制实现)。EsayMock和jMock是通过模仿(moke)对象来测试java代码的包。他们都是通过使用CGLIB来为那些没有接口的类创建模仿(moke)对象。

我们先通过demo来快速了解CGLIB的使用示例。

定义实现类

  1. /**
  2. * 这个是没有实现接口的实现类
  3. *
  4. * @author student
  5. *
  6. */
  7. public class BookFacadeImpl {
  8. public void addBook() {
  9. System.out.println("增加图书的普通方法...");
  10. }
  11. }

定义代理

  1. /**
  2. * 使用cglib动态代理
  3. *
  4. * @author student
  5. *
  6. */
  7. public class BookFacadeCglib implements MethodInterceptor {
  8. private Object target;
  9. /**
  10. * 创建代理对象
  11. *
  12. * @param target
  13. * @return
  14. */
  15. public Object getInstance(Object target) {
  16. this.target = target;
  17. Enhancer enhancer = new Enhancer();
  18. enhancer.setSuperclass(this.target.getClass());
  19. // 回调方法
  20. enhancer.setCallback(this);
  21. // 创建代理对象
  22. return enhancer.create();
  23. }
  24. @Override
  25. // 回调方法
  26. public Object intercept(Object obj, Method method, Object[] args,
  27. MethodProxy proxy) throws Throwable {
  28. System.out.println("事物开始");
  29. proxy.invokeSuper(obj, args);
  30. System.out.println("事物结束");
  31. return null;
  32. }
  33. }

编写Cglib测试

  1. public class TestCglib {
  2. public static void main(String[] args) {
  3. BookFacadeCglib cglib=new BookFacadeCglib();
  4. BookFacadeImpl bookCglib=(BookFacadeImpl)cglib.getInstance(new BookFacadeImpl());
  5. bookCglib.addBook();
  6. }
  7. }

总结

当阅读到spring的AOP章节的时候发现其中使用了代理的一些方法,在此复习一下代理的一些实现以及操作。代理-Spring
AOP的核心设计模式。

java代理(静态代理和jdk动态代理以及cglib代理)的更多相关文章

  1. java静态代理和JDK动态代理

    静态代理 编译阶段就生产了对应的代理类 public interface IBussiness { void execute(); } public class BussinessImpl imple ...

  2. 静态代理和jdk动态代理

    要说动态代理,必须先聊聊静态代理. 静态代理 假设现在项目经理有一个需求:在项目现有所有类的方法前后打印日志. 你如何在不修改已有代码的前提下,完成这个需求? 我首先想到的是静态代理.具体做法是: 1 ...

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

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

  4. 设计模式——代理模式(静态代理和JDK、CGLib动态代理)

    简介 什么是代理模式? 代理模式就是多一个代理类出来,代替原对象进行一些操作.比如说租房的中介.打官司的律师.旅行社,他们可以代替我们做一些事情,这就是代理. 代理模式的应用场景: 如果已有的方法在使 ...

  5. AOP的底层实现-CGLIB动态代理和JDK动态代理

    AOP是目前Spring框架中的核心之一,在应用中具有非常重要的作用,也是Spring其他组件的基础.它是一种面向切面编程的思想.关于AOP的基础知识,相信多数童鞋都已经了如指掌,我们就略过这部分,来 ...

  6. Java动态代理(二)——jdk动态代理

    一.什么是动态代理?代理类在程序运行时创建的代理方式被成为动态代理.动态代理的代理类并不是在Java代码中定义的,而是在运行时根据我们在Java代码中的“指示”动态生成的.相比于静态代理, 动态代理的 ...

  7. Java基础之反射生成JDK动态代理

    在Java的java.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口.通过这个类和接口可以生成JDK动态代理类或动态代理对象. JDK动态代理例子: / ...

  8. java学习笔记(中级篇)—JDK动态代理

    一.什么是代理模式 相信大家都知道代理商这个概念,在商业中,代理商无处不在.假设你要去买东西,你不可能去找真正的厂家去买,也不可能直接跟厂家提出需求,代理商就是这中间的一桥梁,连接买家和厂商.你要买或 ...

  9. Spring学习之设计模式,动态代理和gclib动态代理

    传统的代理模式是静态代理,也就是在方法区域中写入方法. 而动态代理的作用是,不修改实现类的代码,能够在代码的前后或者抛出异常的前后执行某个方法. 动态代理类的实现 //Interface public ...

随机推荐

  1. vue获取浏览器地址栏参数(?及/)路由+非路由实现方式

    1.? 参数 浏览器参数形式:http://javam4.com/m4detail?id=1322914793170014208 1.1.路由取参方式 this.$route.query.id 前端跳 ...

  2. xargs--冬天里的一丝暖意

    本文为博客园作者所写: 一寸HUI,个人博客地址:https://www.cnblogs.com/zsql/ 你有批量kill作业吗?有因为删除文件夹的内容太多而报错吗?-bash: /bin/rm: ...

  3. 我在苦苦坚持的时候,WebStorm已经悄悄的“真香”起来

    前言 最近接了一个活儿,是用WebStorm开发一个基于VUE的网站,但是我真的是几乎没接触过VUE相关的项目实践,更别说用WebStorm在实际中的应用,之前只是听朋友说多好用,但是,因为现有工具不 ...

  4. PyQt(Python+Qt)学习随笔:Qt Designer中toolBar的toolButtonStyle属性

    tooButtonStyle属性保存主工具栏按钮的样式设置,用来表示工具栏按钮的文字和图标怎么显示. 该属性的可设置值类型为枚举类型Qt.ToolButtonStyle,它包含如下值: 该属性的缺省值 ...

  5. Hbase 2.2.2 简单API操作

    前言 小案例中有创建表.创建命名空间.插入数据.获取数据. 环境准备 maven依赖可根据自己的版本进行调整 <!-- hbase依赖--> <dependency> < ...

  6. PHP代码审计分段讲解(4)

    08 SESSION验证绕过 源代码为: <?php ​ $flag = "flag"; ​ session_start(); if (isset ($_GET['passw ...

  7. Hello!OA!Hello!工作流!寻找OA和工作流的旅途记录

    最近新到了一家公司,这家公司做的人力资源管理,需要一个OA系统,所以就让我做一个选型,经过我2周时间的筛选,试用,沟通,测试,最终确定了几款,这个艰辛的路程,在这里记录一下~ 寻找OA的路程----- ...

  8. 冰点文库下载器 v3.2.12(0314) 去广告单文件

    冰点文库,免积分免登陆文档下载神器!付费文档免费下载工具.百度文库免费下载工具.        冰点文库下载器,免费下载文档工具,无需积分也无需登陆就能自由下载百度文库.豆丁网.丁香网.电器网.MBA ...

  9. 6、Sping Cloud Feign

    1.Spring Cloud Feign简介 (1).Fegin简介 官方文档:http://projects.spring.io/spring-cloud/spring-cloud.html#spr ...

  10. 连接数从异常到 300 到 5(RDS MySQL 的一个大坑•后记)

    在 <记 RDS MySQL 的一个大坑> 中,我提到遇到 User juxxxxxxxxxx already has more than 'max_user_connections' a ...