版权声明:本文为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. redis 五大数据类型使用

    redis 五大数据类型使用 字符串str 单个值 127.0.0.1:6379> set name pp # 设置键值[O(1)] OK 127.0.0.1:6379> setex na ...

  2. web自动化-绕过登录

    两个方法: 1.常用的方法: 第一种方法是登录后查看网站的 cookie,请求 url 的时候把 cookie 带上(缺点是:cookie有时间限制.优点:简单,方便) 2.添加cookies的方式: ...

  3. 浅尝 Elastic Stack (四) Logstash + Beats 读取 Spring Boot 日志

    一.Spring Boot 日志配置 采用 Spring Boot 默认的 Logback: <?xml version="1.0" encoding="UTF-8 ...

  4. 定时器:Django-crontab

    定时器是平时编程中比较常用的,今天分享一个Django里非常好用又简单的定时亲:Django-crontab.这个真的是非常的简单好用,比celery+Django执行周期任务简单的多 首先下载dja ...

  5. 【手把手学习flutter】Flutter打Android包的基本配置和包体积优化策略

    [手把手学习flutter]Flutter打Android包的基本配置和包体积优化策略 关注「松宝写代码」,回复"加群" 加入我们一起学习,天天向上 前言 因为最近参加2020FE ...

  6. 最常用的分布式ID解决方案,你知道几个

    一.分布式ID概念 说起ID,特性就是唯一,在人的世界里,ID就是身份证,是每个人的唯一的身份标识.在复杂的分布式系统中,往往也需要对大量的数据和消息进行唯一标识.举个例子,数据库的ID字段在单体的情 ...

  7. 第6.3节 Python动态执行之动态编译的compile函数

    Python支持动态代码主要三个函数,分别是compile.eval和exec.本节介绍compile函数的语法和相关使用.compile函数用来编译一段字符串的源码,将其编译为字节码或者AST(抽像 ...

  8. 第11.27节 Python正则小结:正则静,静则明,明则虚,虚则无为而无不为也

    正则表达式的章节到此就结束了,老猿现在觉得对我们这些身具程序猿基因特色的人来说,正则表达式应该是蛮可口的开胃小菜. 在写标题时,本来想写"正则表达式小结",后来想了想,百度了一下, ...

  9. 团队作业4-Day6

    团队作业4-Day6 项目git地址 1. 站立式会议 2. 项目燃尽图 3. 适当的项目截图 4. 代码/文档签入记录(部分) 5. 每人每日总结 吴梓华:今日修复了图片显示BUG,补充了排位模式出 ...

  10. v-lazyload数据变化图片不切换

    这个问题让我很困惑,明明得到的商品数据已经改变了,但是就图片不变化,随后找到了解决办法,那就是多加一个动态的key <img v-lazy="item.productImage&quo ...