springIOC动态代理的那些事儿

1.发现问题

今天在使用spring的IOC容器时发现了这样的一个问题:

首先有一个接口定义如下:

public interface BookShopService {
void purchase(String username, Integer isbn) throws Exception;
}

它的实现类如下:

package cn.ccsu.service.impl;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import cn.ccsu.dao.BookMapper;
import cn.ccsu.dao.StockMapper;
import cn.ccsu.dao.UserMapper;
import cn.ccsu.exception.BookStockException;
import cn.ccsu.exception.UserAccountException;
import cn.ccsu.service.BookShopService; @Service("bookShopServiceImpl")
public class BookShopServiceImpl implements BookShopService { @Autowired
private UserMapper userMapper; @Autowired
private BookMapper bookMapper; @Autowired
private StockMapper stockMapper; public BookShopServiceImpl() { } @Transactional
@Override
public void purchase(String userName, Integer id) throws Exception { // 1. 获取书的单价
Integer price = bookMapper.queryPrice(id); // 2. 更新书的库存
if (stockMapper.queryStock(id) == 0) {
throw new BookStockException("库存不足!");
}
System.out.println("更新书的库存:" + stockMapper.updateStock(id)); // 3. 更新用户余额
if (userMapper.queryBalance(userName) < price) {
throw new UserAccountException("余额不足!");
}
System.out.println("\n更新用户余额:" + userMapper.updateBalance(userName, price)); } }

这个类主要是完成对图书的销售工作,这个不是重点。接着我创建spring应用上下文,视图从中获取这个类的实例,这个时候出错了。代码如下:

ClassPathXmlApplicationContext ctxt = new ClassPathXmlApplicationContext("applicationContext.xml");
BookShopService service = (BookShopService) ctxt.getBean("bookShopServiceImpl");

报错信息如下:

com.sun.proxy.$Proxy21 cannot be cast to BookShopServiceImpl

怎么样?是不是懵逼了?且听我细细道来。

2.动态代理

  看到com.sun.proxy.$Proxy21没?这就是突破口:proxy--->代理,这说明spring创建了一个代理对象。为什么是代理对象而不是BookShopServiceImpl类的对象呢?这个后面再说,先看看下面的。

不知道你有没有听说过java的动态代理(不知道的请自行谷歌),java有2种动态代理机制:JDK动态代理和cglib动态代理。前者是基于接口实现的,而后者是基于类实现的。听不懂?行,我简单说下吧!!

比如我刚刚的这个例子,BookShopServiceImpl类实现了BookShopService接口,此时就可以用JDK代理,JDK会创建一个代理对象,暂且叫它$Proxy21吧。$Proxy21和BookShopServiceImpl类没有任何继承关系,但是$Proxy21是BookShopService接口的实现类的对象。也就是说JDK代理创建的是该类的父接口的一个实现对象。

接下来说说cglib代理,cglib代理是基于类的代理。比如有一个基类A,B继承了这个基类A。如果此时创建一个代理对象,该代理对象是可以用B指向的。因为该对象是B的一个实现类的对象。也就是说cglib代理会创建原来的类的一个子类,也就是代理类是原有类的一个子类。

综上所述:JDK代理会创建原有接口的一个实现类,而cglib代理会创建原有类的一个子类。

3.解开谜团

这下明白没?再来看看报错信息:
com.sun.proxy.$Proxy21 cannot be cast to BookShopServiceImpl

这里使用了代理,而且还是JDK代理-->即基于接口的代理,所以不能将该代理对象强转为BookShopServiceImpl类型,因为该代理对象是BookShopService接口的子类型。这就完了吗?还早着呢,继续往下看。

我之前说过:为什么spring IOC容器创建代理对象而不是创建BookShopServiceImpl类的对象呢?仔细看这个类的purchase方法:

	@Transactional
@Override
public void purchase(String userName, Integer id) throws Exception { // 1. 获取书的单价
Integer price = bookMapper.queryPrice(id); // 2. 更新书的库存
if (stockMapper.queryStock(id) == 0) {
throw new BookStockException("库存不足!");
}
System.out.println("更新书的库存:" + stockMapper.updateStock(id)); // 3. 更新用户余额
if (userMapper.queryBalance(userName) < price) {
throw new UserAccountException("余额不足!");
}
System.out.println("\n更新用户余额:" + userMapper.updateBalance(userName, price)); }

这里用了事务。在spring中如果使用事务或者AOP,都会创建代理对象,让这个代理对象去完成。而spring默认的代理机制是JDK代理,所以这里使用了JDK代理,创建的对象是BookShopService的子类型,和BookShopServiceImpl 没有半点关系,所以不能强转为BookShopServiceImpl。还有一种是cglib代理,之前说了,它是基于类的方式。你可以在配置文件中修改代理方式,如下:

<tx:annotation-driven transaction-manager="transactionManager"  mode="aspectj" />

修改代理方式为cglib代理。(注:aspectj代理方式即cglib代理)

4.验证

接着我写了一个小demo,验证了下,代码如下:
A.java:
package cn.ccsu;

public abstract class A {

	public A() {

	}

}
B.java:
package cn.ccsu;

import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional; @Component
public class B extends A { public B() { } @Transactional
public void testAnno() {
}
}
测试:
ClassPathXmlApplicationContext ctxt = new ClassPathXmlApplicationContext("applicationContext.xml");
B b = (B) ctxt.getBean("b"); b.testAnno();

虽然我在在这里用了事务,但是因为没有牵涉到接口,所以会使用cglib代理,也就是创建B类的一个子类型的对象。即代理类是B类的子类。所以在这里无论使用A指向还是B指向都没问题。

接着又写了一个接口以及它的一个实现类,代码如下:

package cn.ccsu.service;

public interface IUserService {

	void addUser();
}
package cn.ccsu.service.impl;

import org.springframework.stereotype.Repository;
import org.springframework.transaction.annotation.Transactional; import cn.ccsu.service.IUserService; @Repository("iUserServiceImpl")
public class IUserServiceImpl implements IUserService { public IUserServiceImpl() { } @Transactional
@Override
public void addUser() { } }
    测试如下:
ClassPathXmlApplicationContext ctxt = new ClassPathXmlApplicationContext("applicationContext.xml");
IUserServiceImpl service =(IUserServiceImpl) ctxt.getBean("iUserServiceImpl");
service.addUser();

此时会报错:

 java.lang.ClassCastException: com.sun.proxy.$Proxy8 cannot be cast to cn.ccsu.service.impl.IUserServiceImpl

如果我去掉@Transactional注解,程序可以正常后运行。当你使用事务(或者AOP)时,spring会自动创建一个代理对象,让这个代理对象去完成。但如果没有使用事务,spring的IOC容器会正常创建该类的一个对象,所以程序可以正常跑起来。

5.总结

敲黑板ing..划重点啦!!划重点啦!!

1.JDK代理是基于接口的,它会创建被代理类的父接口的一个子类型;cglib代理是基于类的,它会创建被代理类的一个子类型。

2.spring有2中代理机制:JDK代理和cglib代理,默认使用前者。

3.当使用AOP或者事务时会自动创建一个代理对象,让它来完成需要处理的事。

这个问题也让我想起了之前的一个bug,同样的问题,只不过是在使用AOP时遇到的,链接如下:

AOP bug

springIOC的那些事的更多相关文章

  1. 数据交换格式与SpringIOC底层实现

    1.数据交换格式 1.1 有哪些数据交换格式 客户端与服务器常用数据交换格式xml.json.html 1.2 数据交换格式应用场景 1.2.1 移动端(安卓.iOS)通讯方式采用http协议+JSO ...

  2. 【腾讯Bugly干货分享】H5 视频直播那些事

    本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/57a42ee6503dfcb22007ede8 Dev Club 是一个交流移动 ...

  3. CSharpGL(31)[译]OpenGL渲染管道那些事

    CSharpGL(31)[译]OpenGL渲染管道那些事 +BIT祝威+悄悄在此留下版了个权的信息说: 开始 自认为对OpenGL的掌握到了一个小瓶颈,现在回头细细地捋一遍OpenGL渲染管道应当是一 ...

  4. TODO:字节的那点事Go篇

    TODO:字节的那点事Go篇 (本文go version go1.7.3 darwin/amd64) 在Golang中string底层是由byte数组组成的. fmt.Println(len(&quo ...

  5. Microsoft Visual Studio 2013 — Project搭载IIS配置的那些事

    前段时间在改Bug打开一个project时,发生了一件奇怪的事,好好的一直不能加载solution底下的这个project,错误如下图所示:大致的意思就是这个project的web server被配置 ...

  6. OpenNLP:驾驭文本,分词那些事

    OpenNLP:驾驭文本,分词那些事 作者 白宁超 2016年3月27日19:55:03 摘要:字符串.字符数组以及其他文本表示的处理库构成大部分文本处理程序的基础.大部分语言都包括基本的处理库,这也 ...

  7. 深入理解Spring--动手实现一个简单的SpringIOC容器

    接触Spring快半年了,前段时间刚用Spring4+S2H4做完了自己的毕设,但是很明显感觉对Spring尤其是IOC容器的实现原理理解的不到位,说白了,就是仅仅停留在会用的阶段,有一颗想读源码的心 ...

  8. HTTPS那些事(一)HTTPS原理

    转载来自:http://www.guokr.com/post/114121/ 谣言粉碎机前些日子发布的<用公共WiFi上网会危害银行账户安全吗?>,文中介绍了在使用HTTPS进行网络加密传 ...

  9. 做一个 App 前需要考虑的几件事

    做一个 App 前需要考虑的几件事  来源:limboy的博客   随着工具链的完善,语言的升级以及各种优质教程的涌现,做一个 App 的成本也越来越低了.尽管如此,有些事情最好前期就做起来,避免当 ...

随机推荐

  1. Oracle笔记-Multitable INSERT 的用法

    [转自]  http://blog.chinaunix.net/uid-8504518-id-3310531.html 为避免日趋衰退的记忆力,参考官方E文文档<Introduction to ...

  2. jacob自己动生成word文档目录

    任务目的 1自动生成word文档目录. 用例测试操作步骤 在一个word文档的第二页填写占位符: {目录}保存.调用程序读取目标文档,自动根据标题生成目录到{目录}位置. 效果 关键代码 insert ...

  3. JavaScript中有var和没var的区别

    Js中的变量声明的作用域是以函数为单位,所以我们经常见到避免全局变量污染的方法是 (function(){ // ... })(); 在函数内部,有var和没var声明的变量是不一样的.有var声明的 ...

  4. 数据挖掘:基于Spark+HanLP实现影视评论关键词抽取(1)

    1. 背景 近日项目要求基于爬取的影视评论信息,抽取影视的关键字信息.考虑到影视评论数据量较大,因此采用Spark处理框架.关键词提取的处理主要包含分词+算法抽取两部分.目前分词工具包较为主流的,包括 ...

  5. git提交代码报错 trailing whitespace的解决方法

    1. git提交代码报错 trailing whitespace 禁止执行pre-commit脚本 进入到项目目录中 chmod a-x .git/hooks/pre-commit 2.git提交代码 ...

  6. IE6,IE7,IE8 css bug搜集及浏览器兼容性问题解决方法汇总

    断断续续的在开发过程中收集了好多的bug以及其解决的办法,都在这个文章里面记录下来了!希望以后解决类似问题的时候能够快速解决,也希望大家能在留言里面跟进自己发现的ie6 7 8bug和解决办法! 1: ...

  7. nopCommerce 3.9 版本发行

    NopCommerce中文信息地址:http://www.nopcn.com/nopcommerce39-blog-release-notes.html NopCommerce英文地址:http:// ...

  8. html和css(一)

    简单点来说html和css就是一起连在使用,有了html和css会使网页更加有活力,看起来更加的好看. html是做关于网页标签这一块相当于骨架,更深的还需要另一个来完成,那就是css,相当于向里面加 ...

  9. 运算符重载关键字operator

    operator关键字用来重载内置运算符,使用方法如下: public class OperatorController : Controller { // // GET: /Operator/ pu ...

  10. 04.Continue,和三元表达式的学习

    立即结束本次循环,判断循环条件,如果成立,则进入下一次循环,否则退出循环. 举例:运动员跑步喝水的例子 比如:我编写代码的时候,上个厕所,回来继续写代码 练习1: namespace _09.练习02 ...