一、SM思路分析

1、引入核心依赖及相关依赖:

 spring(略)、mybatis、mysql、mybatis-spring(减少自己实现FactoryBean接口)、druid

<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.22</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.6</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.5.6</version>
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.2.5</version>
</dependency>

2、如何整合?

通过spring框架接管mybatis框架中核心对象的创建

3、mybatis中的核心对象

SqlSessionFactory

4、SM整合实质

通过spring框架接管mybatis中核心的SqlSessionFactory对象的创建。Spring如何管理SqlSessionFactory对象的创建?SqlSessionFactory是一个接口类型的复杂对象。

is=Resource.getResourceAsStrem("mybatis-config.xml);

sqlSessionFactory=new SqlSessionFactoryBuilder().build(is);

1)手动配置:

(0)在类路径下配置mybatis-config.xml的数据源:

<dataSource type="POOLED">
<property name="driver" value="com.mysql.cj.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://127.0.0.1:3306/springboot?characterEncoding=UTF-8&amp;serverTimezone=UTC"/>
<property name="username" value="root"/>
<property name="password" value="xroot"/>
</dataSource>

(1)在spring中可以新建类SqlSessionFactoryBean实现FactoryBean接口

//自定义SqlSessionFactory
public class SqlSessionFactoryBean implements FactoryBean<SqlSessionFactory> {
private String configLocations;//通过参数注入配置文件名 public void setConfigLocations(String configLocations) {
this.configLocations = configLocations;
} public SqlSessionFactory getObject() throws Exception {
//InputStream rs = Resources.getResourceAsStream("mybatis-config.xml");
InputStream rs = Resources.getResourceAsStream(configLocations);
SqlSessionFactory build = new SqlSessionFactoryBuilder().build(rs);
return build;
} public java.lang.Class<?> getObjectType() {
return SqlSessionFactory.class;
} public boolean isSingleton() {
return true;
}
}

(2)工厂管理SqlSessionFactory

<bean id="sqlSessionFactory" class="com.icucoder.SqlSessionFactoryBean">
<property name="configLocations" value="mybatis-config.xml"/>
</bean>

(3)工厂获取

ApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
SqlSessionFactory sqlSessionFactory = (SqlSessionFactory) context.getBean("sqlSessionFactory");
System.out.println(sqlSessionFactory.openSession());

2)spring自动配置(使用mybatis-spring依赖:Mybatis-spring.jar封装了SqlSessionFactory对象的创建)

(0)加入mybatis-spring依赖、druid依赖(简单类型)

(1)工厂管理SqlSessionFactory并配置druid数据源对象dataSource

<!--创建数据源对象 druid c3p0 dbcp-->
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://127.0.0.1:3306/springboot?characterEncoding=UTF-8"/>
<property name="username" value="root"/>
<property name="password" value="xroot"/>
</bean> <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<!--依赖数据源-->
<property name="dataSource" ref="dataSource"/>
<!--依赖mapper文件注册-->
<property name="mapperLocations">
<array>
<value>com/icucoder/mapper/UserDaoMapper.xml</value>
</array>
</property>
</bean>

上述配置每创建一个Mapper文件,需要配置一个array,可以使用mapperLocations配置某个具体类路径下xml的自动扫描。

<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<!-- 依赖数据源 -->
<property name="dataSource" ref="dataSource"/>
<!-- 注入mapper通用方式 -->
<property name="mapperLocations" value="classpath:mapper/*.xml"></property>
<!-- 取别名 -->
<property name="typeAliasesPackage" value="com.icucoder.entity"></property>
</bean>
 

(2)工厂获取

ApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
SqlSessionFactory sqlSessionFactory = (SqlSessionFactory) context.getBean("sqlSessionFactory");
System.out.println(sqlSessionFactory.openSession());

二、DAO层开发

1、建实体类(前提数据库中有表)

public class User {
private Integer id;
private String name;
private Integer age;
private Integer email;
//省略有参和无参构造 setter和getter方法
}

2、DAO接口

public interface UserDao {
List<User>findAll();
}

3、开发mapper(dao的实现类)

<mapper namespace="com.icucoder.dao.UserDao">
<select id="findAll" resultType="com.icucoder.entity.User">
select id,name,age,email
from user
  </select>
</mapper>

4、spring配置文件中注册mapper配置文件

<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<!--依赖数据源-->
<property name="dataSource" ref="dataSource"/>
<!--依赖mapper文件注册-->
<property name="mapperLocations">
<array>
<value>classpath:mapper/UserDaoMapper.xml</value>
</array>
</property>
</bean>

上述配置每创建一个Mapper文件,需要配置一个array,可以使用mapperLocations配置某个具体类路径下xml的自动扫描(见上)。

5、启动工厂获SqlSessionfactory

1)手动获取DAO:

ApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
SqlSessionFactory sqlSessionFactory = (SqlSessionFactory) context.getBean("sqlSessionFactory");
SqlSession sqlSession = sqlSessionFactory.openSession();
UserDao userDao = sqlSession.getMapper(UserDao.class);
System.out.println(userDao.findAll());

2)使用MapperFactoryBean获取DAO:

(1)在MapperFactoryBean中注入SqlSessionFactory和Dao路径

<!--创建Dao组件类-->
<bean id="userDao" class="org.mybatis.spring.mapper.MapperFactoryBean">
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
<!--注入dao类型-->
<property name="mapperInterface" value="com.icucoder.dao.UserDao"/>
</bean>

上述配置每创建一个Dao组件类,需要配置一个bean,可以使用MapperScannerConfigurer配置某个具体累计下dao的自动扫描

<!--创建Dao组件类-->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>>
<!--扫描DAO接口所在包-->
<property name="basePackage" value="com.icucoder.dao"/>
</bean>

(2)调用bean

ApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
UserDao userDao = (UserDao) context.getBean("userDao");
System.out.println(userDao.findAll());

三、Service层事务控制

在上述dao的基础上:

1、常规做法(使用默认的事务)

1)创建Service及其实现类

//UserService省略
public class UserServiceImpl implements UserService {
private UserDao userDao;
//private DataSource dataSource;//手动注入 dataSoruce无法保证service与dao层事务的连接性
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
} @Override
public List<User> findAll() {
return userDao.findAll();
}
}

2)在spring配置文件中管理service组件

<bean id="userService" class="com.icucoder.service.impl.UserServiceImpl">
<property name="userDao" ref="userDao"/>
</bean>

3)调用Bean

ApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
UserService userService = (UserService) context.getBean("userService");
System.out.println(userService.findAll());

2、mybatis框架中事物控制分析

SqlSession,提交:sqlSession.commit);回滚:sqlSessioin.rollback();

真正负责数据库事物控制的对象:Connection(java.sql.Connection)对象。

注意:数据源DruidDataSource是一个连接池,里面都是Connection对象。

Connection conn=DruidDataSource().getConnection();

可以在业务层手动注入dataSource对象来控制事务,但是无法保证业务层(Sevice)和Dao事务的连接性。

3、数据源事务管理器:DataSourceTransactionManager

在spring和mybatis框架中提供了一个类DataSourceTransactionManager(数据源事务管理器)来全局创建一个事务管理器,用来统一调度业务层当前线程使用连接对象和DAO层实现连接对象一致。它依赖于数据源。

在spring.xml文件中添加DataSourceTransactionManager的bean。

<!---数据源事务管理器-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>

4、给现有业务层加入事务控制(手动配置事务)

1)重写service实现类

public class UserServiceImpl implements UserService {
private PlatformTransactionManager transactionManager; public void setTransactionManager(PlatformTransactionManager transactionManager) {
this.transactionManager = transactionManager;
} private UserDao userDao; public void setUserDao(UserDao userDao) {
this.userDao = userDao;
} @Override
public List<User> findAll() {
return userDao.findAll();
} @Override
public void save(User user) {
TransactionDefinition transactionDefinition = new DefaultTransactionDefinition();
TransactionStatus status = transactionManager.getTransaction(transactionDefinition); //控制事物
try {
//处理业务
//调用业务
userDao.save(user);
int i=1/0;
transactionManager.commit(status);
System.out.println("save success");
}catch (Exception e){
e.printStackTrace();
transactionManager.rollback(status);
}
}
}

2)在spring配置文件的service的bean标签中注入数据源事务管理器

<bean id="userService" class="com.icucoder.service.impl.UserServiceImpl">
<property name="userDao" ref="userDao"/>
<property name="transactionManager" ref="transactionManager"/>
</bean>

3)调用Bean

ApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
UserService userService = (UserService) context.getBean("userService");
User user=new User(7,"test",12,"admin@qq.com");
System.out.println(user);
userService.save(user);

四、Service层事务控制优化(自动配置事务)

1、spring中处理事务的两种方式

编程式事务处理:通过在业务层中农注入事务管理器对象,然后通过编码的方式进行事务控制。

声明式事务处理:通过利用aop切面编程进行事务控制,并对事务属性在配置文件中完成细粒度配置。

2、给现有业务层加入事务控制(手动配置事务优化)

开发一个基于事务的通知(环绕通知)TransactionAdvice实现MethodInterceptor。

1)创建环绕通知

2)配置切面

3)使用环绕通知

3、给现有业务层加入事务控制(自动配置事务-使用aop)

spring框架提供了<tx:advice/>标签,该标签可以根据事务管理器创建一个基于事务环绕通知对象,并对业务层方法进行细粒度事务控制。

1)在spring配置文件中加入声明式事务管理

<!--声明式事务管理-->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<!--事务细粒度控制,不控制则事务不生效-->
<tx:attributes>
<tx:method name="save*"/>
</tx:attributes>
</tx:advice>

2)配置切面

<!--配置事务切面-->
<aop:config>
<aop:pointcut id="pc" expression="within(com.icucoder.service.impl.*ServiceImpl)"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="pc"/>
</aop:config>

3)重写service实现类

public class UserServiceImpl implements UserService {

    private UserDao userDao;

    public void setUserDao(UserDao userDao) {
this.userDao = userDao;
} @Override
public List<User> findAll() {
return userDao.findAll();
} @Override
public void save(User user) {
//控制事物
//处理业务
//调用业务
userDao.save(user);
int i = 1 / 0;
System.out.println("save");
}
}

4、给现有业务层加入事务控制(自动配置事务-开启注解式事务驱动管理器)

1)在spring配置文件中加入tx:annotation-driven,这样就不要<tx:advice/>和<aop:config>,可以直接使用注解@Transactional。

<!--开启注解式事务驱动管理器@Transactional-->
<tx:annotation-driven transaction-manager="transactionManager"/>

2)重写service实现类

@Transactional
public class UserServiceImpl implements UserService { private UserDao userDao; public void setUserDao(UserDao userDao) {
this.userDao = userDao;
} @Override
public List<User> findAll() {
return userDao.findAll();
} @Override
public void save(User user) {
//控制事物
//处理业务
//调用业务
userDao.save(user);
int i = 1 / 0;
System.out.println("save"); }
}
 注:
也可以使用easycode直接对数据的表生成entity,controller,service,dao,mapper,无需任何编码。
 

Spring入门二:整合mybatis的更多相关文章

  1. Spring Boot:整合MyBatis框架

    综合概述 MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可以使用简单 ...

  2. MyBatis原理,Spring、SpringBoot整合MyBatis

    1. MyBatis概述 MyBatis 是支持定制化 SQL.存储过程以及高级映射的优秀的持久层框架.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可 ...

  3. spring boot 2整合mybatis

    mybatis-spring-boot-starter主要有两种解决方案,一种是使用注解,一种是使用XML. 参考这篇文章动手跑了一个例子,稍微不同之处,原文是spring boot,这里改成了spr ...

  4. springboot(二)整合mybatis,多数据源和事务管理

     -- 1.整合mybatis -- 2.整合多数据源 -- 3. 整合事务 代码地址:https://github.com/showkawa/springBoot_2017/tree/master/ ...

  5. Spring入门(二):自动化装配bean

    Spring从两个角度来实现自动化装配: 组件扫描(component scanning):Spring会自动发现应用上下文中需要创建的bean. 自动装配(autowiring):Spring会自动 ...

  6. Spring Boot 应用系列 3 -- Spring Boot 2 整合MyBatis和Druid,多数据源

    本文演示多数据源(MySQL+SQL Server)的配置,并且我引入了分页插件pagehelper. 1. 项目结构 (1)db.properties存储数据源和连接池配置. (2)两个数据源的ma ...

  7. Spring Boot2 系列教程(二十二)整合 MyBatis 多数据源

    关于多数据源的配置,前面和大伙介绍过 JdbcTemplate 多数据源配置,那个比较简单,本文来和大伙说说 MyBatis 多数据源的配置. 其实关于多数据源,我的态度还是和之前一样,复杂的就直接上 ...

  8. 10、Spring教程之整合MyBatis

    1.步骤 1.导入相关jar包 junit <dependency> <groupId>junit</groupId> <artifactId>juni ...

  9. SpringBoot入门篇--整合mybatis+generator自动生成代码+druid连接池+PageHelper分页插件

    原文链接 我们这一篇博客讲的是如何整合Springboot和Mybatis框架,然后使用generator自动生成mapper,pojo等文件.然后再使用阿里巴巴提供的开源连接池druid,这个连接池 ...

随机推荐

  1. Jvm内存回收

    一.什么内存会被回收 可达性分析算法 通过一系列的GC ROOT的对象作为超始点,从这些节点开始向下搜索,搜索所走的路径称为"引用链",当一个对象到GC ROOT之间没有任何引用链 ...

  2. QA(测试) 工作准则建议

    身为一个专业的 QA 当然需要有自己的测试原则,这些测试原则不仅可以帮助我们提高产品质量,对外还能体现出我们的专业性,从而让合作方后续还有意愿和我们合作. 1 测试前 1.1 需求评审 必须参与,有问 ...

  3. Java进制的转换

    进制:进制是一种记数方式 ,可以用有限的数字符号代表所有的数值.由特定的数值组成. 整型的表现形式 十进制: 都是以0-9这九个数字组成,不能以0开头. 二进制: 由0和1两个数字组成. 八进制: 由 ...

  4. iconv(gb2312<->utf-8)

    转载请注明来源:https://www.cnblogs.com/hookjc/ unix下安装PHP的module,需要重新编译PHP,Windows下安装模板,只需将php.ini里的配置打开相应的 ...

  5. 使用Java开发桌面即时通讯程序遇到的问题

    项目:https://www.lking.top/?p=87 1. JPanel面板绘制背景图片问题. 参考大佬:https://www.jb51.net/article/101516.htm 本项目 ...

  6. node.js 使用domain模块捕获异步回调中的异常

    和其他服务器端语言相比,貌似node.js 对于异常捕捉确实非常困难. 首先你会想到try/catch ,但是在使用过程中我们会发现并没有真正将错误控制在try/catch 语句中. 为什么? 答案是 ...

  7. node.js中的fs.appendFile方法使用说明

    方法说明: 该方法以异步的方式将 data 插入到文件里,如果文件不存在会自动创建.data可以是任意字符串或者缓存. 语法: 代码如下: fs.appendFile(filename, data, ...

  8. python基础1-类属性和实例属性

    类属性就是类对象所拥有的属性,它被所有类对象的实例对象所共有,在内存中只存在一个副本,这个和C++中类的静态成员变量有点类似.对于公有的类属性,在类外可以通过类对象和实例对象访问 类属性 class ...

  9. iOS中属性 (nonatomic, copy, strong, weak)的使用 By hL

    以下内容来自Stackflow的详解 1.Nonatomicnonatomic is used for multi threading purposes. If we have set the non ...

  10. python 使用pip安装包的总结

    multiprocessing.logging模块安装 如果使用在cmd中使用 pip install multiprocessing 会报错, 将命令改为 pip3 install multipro ...