MyBatist庖丁解牛(四)
什么是MyBatis-Spring?
MyBatis-Spring
就是帮助你将MyBatis
代码无缝的整合到Spring
中。Spring
将会加载必要的sqlSessionFactory
类和session
类。
配置数据源
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="locations">
<list>
<value>classpath:app.properties</value>
</list>
</property>
</bean> <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close">
<property name="driverClass" value="${jdbc.driver}"></property>
<property name="jdbcUrl" value="${jdbc.url}"></property>
<property name="user" value="${jdbc.user}"></property>
<property name="password" value="${jdbc.password}"></property> <property name="maxPoolSize" value="20" />
<property name="minPoolSize" value="5" />
<property name="acquireIncrement" value="3" />
<property name="initialPoolSize" value="5"></property>
</bean>
配置SqlSessionFactory
我们需要在Spring应用上下文定义一个SqlSessionFactory
和数据源dataSource
。在MyBatis-Spring
中,SqlSessionFacotoryBean
调用其getObject()
方法去创建SqlSessionFactory
实例。
创建SqlSessionFactory
实例中用到了享元设计模式,是一个种结构型模式。
- 享元模式(Flyweight Pattern):主要用于减少创建对象的数量,以减少内存占用和提高性能。这种类型的设计模式属于结构型模式,它提供了极少对象数量从而改善应用所需的对象结构的方式。
- 优点:大大减少对象的创建,降低系统的内存,提供效率。
- 缺点: 提高了系统的复杂度,需要分离出外部状态和内部状态,而且外部状态具有固有化的性质,不应该随着内部的状态而改变,否则会造成系统的混乱。
言归正传,首先在这个SqlSessionFactoryBean
类中,调用getObject()
时,会判断sqlSessionFacotory
是否为空,如果不为空直接返回sqlSessionFacotory
。如果为空,则调用afterPropertiesSet()
方法。
public SqlSessionFactory getObject() throws Exception {
if (this.sqlSessionFactory == null) {
this.afterPropertiesSet();
} return this.sqlSessionFactory;
}
afterPropertiesSet()
会检查dataSource
和sqlSessionFactoryBuilder
是否为空。接着会调用buildSqlSessionFactory()
方法。
public void afterPropertiesSet() throws Exception {
Assert.notNull(this.dataSource, "Property 'dataSource' is required");
Assert.notNull(this.sqlSessionFactoryBuilder, "Property 'sqlSessionFactoryBuilder' is required");
Assert.state(this.configuration == null && this.configLocation == null || this.configuration == null || this.configLocation == null, "Property 'configuration' and 'configLocation' can not specified with together");
this.sqlSessionFactory = this.buildSqlSessionFactory();
}
在buildSqlSessionFactory()
方法会读取xml
配置,然后创建sqlSessionFactory
实例。
protected SqlSessionFactory buildSqlSessionFactory() throws IOException {
//...省略几百行代码
return this.sqlSessionFactoryBuilder.build(configuration);
}
言归正传,配置如下:
mapperLocations
: 我们存放MyBatis
的mapper.xml
文件路径。typeAliasesPackage
:一般对应着我们实体类所在的包。多个package
之间可以用逗号或者分号等来进行分隔。
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource" />
<property name="typeAliasesPackage" value="com.helping.wechat.model"></property>
<property name="mapperLocations" value="classpath*:com/helping/wechat/model/**/mysql/*Mapper.xml" />
</bean>
事务
一个使用MyBatis-Spring
的主要原因是它允许MyBatis
参与到Spring
的事务管理中,而不是给MyBatis
创建一个新的事务管理器。MyBatis-Spring
利用了存在Spring
中的org.springframework.jdbc.datasource.DataSourceTransactionManager
。
我们可以通过配置<tx:annotation-driven transaction-manager="transactionManager"/
>来使用@Transactional
注解面对切面编程。同时我们也可以通常的AOP
来配置Service层的事务。
在事务处理期间,一个单独的SqlSession
对象将会被创建和使用,当事务完成时,这个Session
就会以合适的方式进行提交和回滚。
具体配置如下:
<!-- 声明式事务 -->
<bean name="txmanager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<!-- 事务开启必须使用session -->
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 当Service层的方法不是以delete,insert,update,save方法开头时,发生异常时无法进行事务回滚
但是可以通过@Transactional注解在类或者方法上,可以启动事务。
-->
<tx:annotation-driven transaction-manager="transactionManager"/>
<!-- 任何RuntimeException将触发事务回滚,但是任何Checked Exception不会触发事务回滚 -->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="delete*" propagation="REQUIRED" read-only="false" isolation="DEFAULT"/>
<tx:method name="insert*" propagation="REQUIRED" read-only="false" isolation="DEFAULT"/>
<tx:method name="update*" propagation="REQUIRED" read-only="false" isolation="DEFAULT"/>
<tx:method name="save*" propagation="REQUIRED" read-only="false" isolation="DEFAULT"/>
<tx:method name="*" propagation="SUPPORTS" read-only="true"/>
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut expression="execution(* com.helping.wechat.service..*.*(..))" id="pc"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="pc" order="1"/>
</aop:config>
使用SqlSession
在
MyBatis
中,我们可以使用SqlSessionFactory
来创建SqlSession
。一旦获得了一个session
之后,我们可以使用它来执行映射语句,提交或者回滚。当我们不需要它的时候,我们可以关闭session
。而在
MyBatis-Spring
之后,你不再需要SqlSessionFactory
了,我们的Dao
层可以注入一个线程安全的SqlSession
,然后进行提交或者回滚,最后关闭session
。
SqlSession Template
SqlSession Template
是MyBatis-Spring
的核心,这个类负责MyBatis
的SqlSession
。
this.sqlSession = new SqlSessionTemplate(sqlSessionFactory);
我们可以知道SqlSessionTemplate
类的构造器第一个参数是SqlSessionFactory
的实例。SqlSessionTemplate
实现了SqlSession
接口。
public class SqlSessionTemplate implements SqlSession {
//...省略n行
}
之前的我是这样配置的:
<bean id="sqlSessionTemplate" class="org.mybatis.spring.SqlSessionTemplate">
<constructor-arg index="0" ref="sqlSessionFactory" />
</bean>
<bean id="baseDao" abstract="true">
<property name="sqlSessionTemplate" ref="sqlSessionTemplate" />
</bean>
<bean id="adminUserDao" class="com.helping.wechat.dao.admin.mybatis.AdminUserDao" parent="baseDao" />
其实没有必要去Spring.xml
去配置Dao
层,Service
层,Controller
类。可以用注解代替,后面会讲到。对了,SqlSessionTemplate
其实也不需要配,SqlSession
我们可以通过new SqlSessionTemplate()
创建出来。
SqlSessionDaoSupport
SqlSessionDaoSupport
是一个抽象的支持类,为你提供SqlSession
。我们自定义的Dao
层的实现类继承它就行了。调用getSession()
做你想要的。SqlSessionDaoSupport
内部实现:
public abstract class SqlSessionDaoSupport extends DaoSupport {
private SqlSession sqlSession;
private boolean externalSqlSession;
public void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) {
if (!this.externalSqlSession) {
this.sqlSession = new SqlSessionTemplate(sqlSessionFactory);
}
}
public void setSqlSessionTemplate(SqlSessionTemplate sqlSessionTemplate) {
this.sqlSession = sqlSessionTemplate;
this.externalSqlSession = true;
}
/**
* Users should use this method to get a SqlSession to call its statement methods
* This is SqlSession is managed by spring. Users should not commit/rollback/close it
* because it will be automatically done.
*
* @return Spring managed thread safe SqlSession
* 返回线程安全的sqlSession
*/
public SqlSession getSqlSession() {
return this.sqlSession;
}
/**
* {@inheritDoc}
*/
protected void checkDaoConfig() {
notNull(this.sqlSession, "Property 'sqlSessionFactory' or 'sqlSessionTemplate' are required");
}
}
看到getSession()
方法的注释写着,用户不应该去提交,回滚,或者关闭session
的操作,因为session
会自动关闭它。这个方法返回的是一个线程安全的sqlSession
。
使用注解代替XML的配置
如果不使用注解,你的Spring.xml
配置肯定是多的爆炸,比如这样:
<bean id="sqlSessionTemplate" class="org.mybatis.spring.SqlSessionTemplate">
<constructor-arg index="0" ref="sqlSessionFactory" />
</bean>
<bean id="baseDao" abstract="true">
<property name="sqlSessionTemplate" ref="sqlSessionTemplate" />
</bean>
<bean id="adminUserDao" class="com.helping.wechat.dao.admin.mybatis.AdminUserDao" parent="baseDao" />
<bean id="adminUserService" class="com.helping.wechat.service.admin.impl.AdminUserService" >
<property name="adminUserDao" ref="adminUserDao" />
</bean>
<bean id="adminUserController" class="com.helping.wechat.controller.admin.AdminUserController" >
<property name="adminUserService" ref="adminUserService" />
</bean>
<bean id="ordinaryUserDao" class="com.helping.wechat.dao.admin.mybatis.OrdinaryUserDao" parent="baseDao" />
<bean id="ordinaryUserService" class="com.helping.wechat.service.admin.impl.OrdinaryUserService" >
<property name="ordinaryUserDao" ref="ordinaryUserDao" />
</bean>
<bean id="ordinaryUserController" class="com.helping.wechat.controller.admin.OrdinaryUserController" >
<property name="ordinaryUserService" ref="ordinaryUserService" />
</bean>
如果使用了@Service
,@Component
,@Repository
,@Controller
,@Resource
,@Autowired
这些注解,这些配置统统可以不要。首先要在SpringMVC.xml
配置自动扫描注解的tag
。
<context:component-scan base-package="com.helping.wechat.controller" />
<context:component-scan base-package="com.helping.wechat.handler" />
<context:component-scan base-package="com.helping.wechat.dao" />
<context:component-scan base-package="com.helping.wechat.service" />
@Service
用于标注业务Service
层的类上,有value
这个属性。
@Service(value = "adminUserService")
public class AdminUserService implements IAdminUserService {
@Resource(name = "adminUserDao")
private IAdminUserDao adminUserDao;
}
@Controller
用于标注控制Controller
层的类上。
@Controller
@RequestMapping("/api/admin")
public class AdminUserController extends BaseController {
@Resource(name = "adminUserService")
private IAdminUserService adminUserService;
}
@Repository
用于标注在Dao
层的类上,有value
这个属性。
@Repository(value = "adminUserDao")
public class AdminUserDao extends SqlSessionDaoSupport implements IAdminUserDao {
}
@Component
就是把类注册到Spring.xml
文件中。
@Component
public abstract class BaseDao {
protected SqlSession sqlSession;
protected SqlSessionFactory sqlSessionFactory;
@Resource(name = "sqlSessionFactory")
public void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) {
this.sqlSessionFactory = sqlSessionFactory;
this.sqlSession = new SqlSessionTemplate(sqlSessionFactory);
}
public void setSqlSession(SqlSession sqlSession) {
this.sqlSession = sqlSession;
}
public SqlSession getSqlSession() {
return sqlSession;
}
}
Dao
层的类继承它就行了,就可以调用getSqlSession()
进行想要的操作了。
@Resource
默认按名称进行装配字段或者setter
方法。名称可以通过name
属性进行指定,如果没有指定name
属性。当注解写在字段上,默认按字段名进行安装名称查找,如果注解写在setter
方法上,默认按属性名进行装配,当找不到与名称相匹配的bean
时候,才会按照类型进行装配。如果name
属性指定了,那么就只能按名称进行装配了。
@Service(value = "adminUserService")
public class AdminUserService implements IAdminUserService {
@Resource(name = "adminUserDao")
private IAdminUserDao adminUserDao;
}
@Autowired
:默认按类型装配,默认情况下必须要求依赖对象必须存在,如果要运行null
值,可以设置它的required
属性为false
。如果我们想要使用名称进行装配,可以结合@Qualifier
注解使用。
@Autowired
public void setAdminUserService(IAdminUserService adminUserService) {
this.adminUserService = adminUserService;
}
尾言
明天去看看通用Mapper
和pageHelper
怎么配置。晚安,地球人。
作者:cmazxiaoma
链接:https://www.jianshu.com/p/67c99ebdd5d8
来源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。
MyBatist庖丁解牛(四)的更多相关文章
- MyBatist庖丁解牛(一)
站在巨人的肩膀上,感谢! https://www.jianshu.com/p/ec40a82cae28?utm_campaign=maleskine&utm_content=note& ...
- MyBatist庖丁解牛(五)
很多时候我们在自己的每个service中没有中注入SqlSessionTemplate; 但是我们直接调用mapper接口方法就直接能够操作数据库 这个是为什么??下面开始解惑: Mybatis Sq ...
- MyBatist庖丁解牛(三)
从MyBatis代码实现的角度来看,MyBatis的主要的核心部件有以下几个: SqlSession:作为MyBatis工作的主要顶层API,表示和数据库交互的会话,完成必要数据库增删改查功能: Ex ...
- MyBatist庖丁解牛(二)
站在巨人的肩膀上 https://blog.csdn.net/xiaokang123456kao/article/details/76228684 一.概述 我们知道,Mybatis实现增删改查需要进 ...
- TLD网络资源汇总--学习理解之(四)
原文:http://blog.csdn.net/mysniper11/article/details/8726649 引文地址:http://www.cnblogs.com/lxy2017/p/392 ...
- 干货 | 45张图庖丁解牛18种Queue,你知道几种?
在讲<21张图讲解集合的线程不安全>那一篇,我留了一个彩蛋,就是Queue(队列)还没有讲,这次我们重点来看看Java中的Queue家族,总共涉及到18种Queue.这篇恐怕是市面上最全最 ...
- 构建一个基本的前端自动化开发环境 —— 基于 Gulp 的前端集成解决方案(四)
通过前面几节的准备工作,对于 npm / node / gulp 应该已经有了基本的认识,本节主要介绍如何构建一个基本的前端自动化开发环境. 下面将逐步构建一个可以自动编译 sass 文件.压缩 ja ...
- 《Django By Example》第四章 中文 翻译 (个人学习,渣翻)
书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:祝大家新年快乐,这次带来<D ...
- 如何一步一步用DDD设计一个电商网站(四)—— 把商品卖给用户
阅读目录 前言 怎么卖 领域服务的使用 回到现实 结语 一.前言 上篇中我们讲述了“把商品卖给用户”中的商品和用户的初步设计.现在把剩余的“卖”这个动作给做了.这里提醒一下,正常情况下,我们的每一步业 ...
随机推荐
- Java中实现函数的阻塞
使用Object.wait()即可实现阻塞,使用Object.notify()解除阻塞,代码示例如下 MainFrame.java import javax.swing.JFrame; import ...
- MongoDB 学习一
这一章,我们先介绍几个MongoDB的概念: 1.document: 它是MongoDB的基础数据单元,它大概等价于关系型数据库中的行. 2.collection: 可以想象成动态的表. 3.一个简单 ...
- JSOI2004 平衡点 / 吊打XXX
题目描述 有n个重物,每个重物系在一条足够长的绳子上.每条绳子自上而下穿过桌面上的洞,然后系在一起.图中X处就是公共的绳结.假设绳子是完全弹性的(不会造成能量损失),桌子足够高(因而重物不会垂到地上) ...
- SpringBoot-(3)-RestController接口参数
一,无参接口: //无参接口 @RequestMapping("/appSecret") public String secret() { return "EK125EK ...
- 有关 java 不定参数
不定参数实际为数组参数的一种写法而已,本质上与数组参数完全相同 //1.数组参数函数 public static int sum(int[] values) { } //2.不定参数函数 不定参数只能 ...
- govendor
cd 到工程目录. govendor init : 初始化 govendor fetch : 拉取包 go 1.6以后编译go代码会优先从vendor目录先寻找依赖包: controllers\ar ...
- dokcer3
安装好的文件位置: /usr/sbin/nginx:主程序 /etc/nginx:存放配置文件 /usr/share/nginx:存放静态文件 /var/log/nginx:存放日志 其实从上面的根目 ...
- ubuntu安装ros indigo
版本是14.04.1 一.先配置 1.点击新立得软件包管理器,输入密码exbot123, 2,点击最上面一栏的设置,选择软件源,前四个打勾,后一个不打,把sevice america改成mainsev ...
- 数据结构之 图论---连通分量的个数(dfs搜索)
数据结构实验:连通分量个数 Time Limit: 1000MS Memory limit: 65536K 题目描述 在无向图中,如果从顶点vi到顶点vj有路径,则称vi和vj连通.如果图中任意两个 ...
- Codeforces Round #374 (Div. 2) A. One-dimensional Japanese Crossword —— 基础题
题目链接:http://codeforces.com/contest/721/problem/A A. One-dimensional Japanese Crossword time limit pe ...