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. HTML5本地存储和本地的数据库

    一个网站如何能在客户的浏览器存储更多的数据呢? 在Html4的时代在浏览器端存储点网站个性化的数据,尤其是用户浏览器的痕迹,用户的相关数据等一般只能存储在Cookie中,但是大多是浏览器对于Cooki ...

  2. drf序列化器的实例

    应用目录结构: views.py from django.shortcuts import render # Create your views here. from django.views imp ...

  3. 【总结】sqlmap-tamper编写小结

    目的:修改sqlmap中的tamper脚本来绕过代码对特定参数的过滤和转义 环境:win10.phpstudy2016.sqli-labs-master平台 工具:sqlmap.burpsuite 地 ...

  4. D15 模块

    模块

  5. oracle 单实例DG(切换篇三)

    一,开篇 此篇操作承接上文,必须完成DG实例搭建完成方可有执行以下内容的实例 二,切换物理备库 one. oracle01库命令 select switchover_status from v$dat ...

  6. (转)跟着老男孩一步步学习Shell高级编程实战

    原文:http://oldboy.blog.51cto.com/2561410/1264627/  跟着老男孩一步步学习Shell高级编程实战 原创作品,允许转载,转载时请务必以超链接形式标明文章 原 ...

  7. TOJ 3635 过山车

    Description RPG girls今天和大家一起去游乐场玩,终于可以坐上梦寐以求的过山车了.可是,过山车的每一排只有两个座位,而且还有条不成文的规矩,就是每个女生必须找 个个男生做partne ...

  8. [linux] uptime 命令中关于平均负载的解释

    1.当前时间 00:13:25 2.系统已运行的时间 9小时19分 3.当前在线用户 2 user 4.平均负载:0.17, 0.12, 0.07 最近1分钟.5分钟.15分钟系统的负载 为了更好地理 ...

  9. 121、Django rest framework入门使用

    框架介绍 为你的django平台通过model生成对应的restfull api,并可以通过对应的http接口来进行 post .get.put.delete等操作.本文是也并非入门级别,不会带你去了 ...

  10. The Definitive C++ Book Guide and List--reference

    http://stackoverflow.com/questions/388242/the-definitive-c-book-guide-and-list Reference Style - All ...