spring中基于JDK和CGLIB代理在项目的应用
一、环境与问题
环境
spring boot的版本是1.2.1.RELEASE、JDK版本是1.7
问题
A服务 PeopleService 调用B服务 HelloService ,其中B服务的方法 say() 是是一个事物方法,并且B服务实现一个接口 IHelloService 。实际过程中发现A服务无法使用 @autowire 把B服务注入,但是去掉接口 `IHelloService 或者去掉 @Transactional 则可注入B服务,亦或者@Autowired注入的类型使用IHelloService 。
二、问题思考
在解决上面问题时,我们必须要知道spring容器在开启事物的过程中使用的是AOP技术,其实底层是通过代理实现的。在spring在选择代理时默认实现了2中代理方式一种是JDK代理、另外一种是CGLIB代理。其中JDK代理是基于接口,通过接口InvocationHandler实现的,而CGLIB代理是基于类的,通过接口MethodInterceptor 来实现。
三、验证思考
为了验证我的解决思路是否正确,去官网查看了一下文档
10.5 Using the ProxyFactoryBean to create AOP proxies
JavaBean properties
In common with most FactoryBean implementations provided with Spring, the ProxyFactoryBean
class is itself a JavaBean. Its properties are used to:
• Specify the target you want to proxy.
• Specify whether to use CGLIB.
上面这段话的大体意思:一般情况下面我们使用FactoryBean来提供bean,当使用AOP时会使用ProxyFactoryBean来提供bean。我们可以指定目标(即需要代理的对象),也可指定是否使用CGLIB代理。换而言之,spring默认使用的是JDK的代理。
再看一下实际的A服务的bean


果然是如我们猜想的那样使用了JDK代理。
四、解决问题
指定使用CGLIB代理
@Scope(proxyMode=ScopedProxyMode.TARGET_CLASS)
再看一下结果


当我们指定用CGLIB代理之后,发现HelloService 可以被正常进行注入,并且HelloService 也由JDK代理对象变成了CGLIB代理对象。
五、问题深入与扩展
随着问题的深入,我们没有解决为什么使用JDK的代理对象不行,而使用CGLIB代理对象却可以?
不知道大家有没有注意我上面讲过一句话“JDK代理是基于接口...,而CGLIB代理是基于类的...”,问题就出现在这边。下面我把基于JDK和CGLIB的对象的类名、父类名及接口名打印出来:
HelloService 基于JDK的代理方式:


基于JDK代理的helloService对象的类名$Proxy44,父类是Proxy,接口是IHelloService
HelloService 基于CGLIB的代理方式:


基于CGLIB代理的helloService对象的类名是HelloService$$EnhancerBySpringCGLIB$$80f67bbd 父类是HelloService。
至此我们终于知道了通过@Autowired 注入HelloService 对象时,使用JDK代理时代理对象实现了IHelloService接口,而使用CGLIB代理时代理对象是继承了HelloService 。
spring中的代理的问题告一段落了,孔夫子讲过"举一隅,不以三隅反,则不复也"。那么我们实际项目中还有那些耳熟能详的框架也使用了代理了呢?
在web开发中不知道大家有没有注意到mybatis中只有接口而没有实现类,其实也是使用了JDK的代理。有兴趣的童鞋可以研究一下。
spring中基于JDK和CGLIB代理在项目的应用的更多相关文章
- Spring AOP中的JDK和CGLib动态代理哪个效率更高?
一.背景 今天有小伙伴面试的时候被问到:Spring AOP中JDK 和 CGLib动态代理哪个效率更高? 二.基本概念 首先,我们知道Spring AOP的底层实现有两种方式:一种是JDK动态代理, ...
- JAVA高级架构师基础功:Spring中AOP的两种代理方式:动态代理和CGLIB详解
在spring框架中使用了两种代理方式: 1.JDK自带的动态代理. 2.Spring框架自己提供的CGLIB的方式. 这两种也是Spring框架核心AOP的基础. 在详细讲解上述提到的动态代理和CG ...
- Spring中的JDK动态代理
Spring中的JDK动态代理 在JDK1.3以后提供了动态代理的技术,允许开发者在运行期创建接口的代理实例.在Sun刚推出动态代理时,还很难想象它有多大的实际用途,现在动态代理是实现AOP的绝好底层 ...
- 基于 JDK 的动态代理机制
『动态代理』其实源于设计模式中的代理模式,而代理模式就是使用代理对象完成用户请求,屏蔽用户对真实对象的访问. 举个最简单的例子,比如我们想要「FQ」访问国外网站,因为我们并没有墙掉所有国外的 IP,所 ...
- Spring中基于xml的AOP
1.Aop 全程是Aspect Oriented Programming 即面向切面编程,通过预编译方式和运行期动态代理实现程序功能的同一维护的一种技术.Aop是oop的延续,是软件开发中的 一个热点 ...
- 基于JDK的动态代理原理分析
基于JDK的动态代理原理分析 这篇文章解决三个问题: What 动态代理是什么 How 动态代理怎么用 Why 动态代理的原理 动态代理是什么? 动态代理是代理模式的一种具体实现,是指在程序运行期间, ...
- 在Spring中使用JDK定时器实现调度任务
在Spring中使用JDK定时器实现调度任务 作者:chszs,转载需注明.博客主页: http://blog.csdn.net/chszs 本文探讨Spring如何集成JDK的Timer定时器,实现 ...
- 【转】在Spring中基于JDBC进行数据访问时怎么控制超时
http://www.myexception.cn/database/1651797.html 在Spring中基于JDBC进行数据访问时如何控制超时 超时分类 超时根据作用域可做如下层级划分: Tr ...
- Spring 框架的概述以及Spring中基于XML的IOC配置
Spring 框架的概述以及Spring中基于XML的IOC配置 一.简介 Spring的两大核心:IOC(DI)与AOP,IOC是反转控制,DI依赖注入 特点:轻量级.依赖注入.面向切面编程.容器. ...
随机推荐
- Redhat更换yum源
redhat 默认自带的 yum 源需要注册,才能更新,所以对于我们来说需要替换掉redhat的yum源.下文更换为网易的. 删除原有的yum rpm -qa|grep yum|xargs rpm - ...
- Python_Runoob
python复合赋值 # Fibonacci series: 斐波纳契数列 # 两个元素的总和确定了下一个数 a, b = 0, 1 while b < 10: print(b) a, b = ...
- Vue小项目二手书商城:(三)前端渲染数据
实现内容: axios取到的数据在前端使用(父子组件各自应该怎么使用) 一.简单使用(在哪取在哪用) 1.在App.vue中script中加上data(data专属于当前组件,父子组件传参通过prop ...
- B/S架构
B/S架构即浏览器和服务器架构模式.它是随着Internet技术的兴起,对C/S架构的一种变化或者改进的架构.在这种架构下,用户工作界面是通过WWW浏览器来实现,极少部分事务逻辑在前端(Browser ...
- Kotlin 对象表达式和对象声明
Kotlin 用对象表达式和对象声明来实现创建一个对某个类做了轻微改动的类的对象,且不需要去声明一个新的子类. 对象表达式 通过对象表达式实现一个匿名内部类的对象用于方法的参数中: window.ad ...
- 第 10 章 容器监控 - 081 - Weave Scope 多主机监控
除了监控容器,Weave Scope 还可以监控 Docker Host 点击顶部 HOSTS 菜单项,地图将显示当前 host. 与容器类似,点击该 host 图标将显示详细信息 host 当前的资 ...
- 20190323——HeadFirestPython学习之异常处理
man=[] other=[] try: data=open('sketch.txt') for each_line in data: try: (role,line_spoken)=each_lin ...
- php 两次encodeURI,解决浏览器跳转请求页乱码报错找不到页面的bug
Not Found The requested URL /index.php/XXX/mid/97329240798095910/bname/3000T/D/sname/水泥粉磨/un ...
- oracle创建删除表空间
create [undo|temporary] tablespace orcp datafile|tempfile 'E:\orcle\oracleBaseDir\oradata\orcp\orcp. ...
- JS 超类和子类
此篇由别的大神的博客与<javascript高级程序设计>一书整理而来 原博客地址:https://hyj1254.iteye.com/blog/628555 看到javascript高级 ...