MyBatis连接池事务、基于XML动态SQL语句、多表查询

一、Mybatis 连接池与事务深入

1、MyBtis连接池分类

配置的位置:主配置文件SqlMapConfig.xml中的dataSource标签,type属性就是表示采用何种连接池方式。
type属性的取值:
UNPOOLED:不使用连接池的数据源,采用传统的获取连接的方式,虽然也实现Javax.sql.DataSource接口,但是并没有使用池的思想
POOLED:使用连接池的数据源, 采用传统的javax.sql.DataSource规范中的连接池,mybatis中有针对规范的实现
JNDI :使用 JNDI 实现的数据源 ,来获取DataSource对象,不同的服务器所能拿到DataSource是不一样

<!-- 配置数据源(连接池)信息 --> 
<dataSource type="POOLED">  
<property name="driver" value="${jdbc.driver}"/>  
<property name="url" value="${jdbc.url}"/>  
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/> 
</dataSource> 

当我们需要创建 SqlSession 对象并需要执行 SQL 语句时,这时候 MyBatis 才会去调用 dataSource 对象来创建java.sql.Connection对象。也就是说,java.sql.Connection对象的创建一直延迟到执行SQL语句的时候。

 @Test 
 public void testSql() throws Exception {  
 InputStream in = Resources.getResourceAsStream("SqlMapConfig.xml");  
 SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(in);  
 SqlSession sqlSession  = factory.openSession();  
 List<User> list = sqlSession.selectList("findUserById",41); 
  System.out.println(list.size()); 
  } 

只有当第 4句sqlSession.selectList(“findUserById”),才会触发MyBatis 在底层执行下面这个方法来创建 java.sql.Connection对象

2、Mybatis 的事务控制

在 JDBC 中我们可以通过手动方式将事务的提交改为手动方式,通过 setAutoCommit()方法就可以调整。Mybatis 框架因为是对 JDBC 的封装,所以 Mybatis 框架的事务控制方式,本身也是用 JDBC的 setAutoCommit()方法来设置事务提交方式的

@Test 
 public void testSaveUser() throws Exception {   
 User user = new User();   
 user.setUsername("mybatis user09"); 
  //6.执行操作   
  int res = userDao.saveUser(user);   
  System.out.println(res);   
  System.out.println(user.getId());  }   
@Before//在测试方法执行之前执行  
public void init()throws Exception { 
  //1.读取配置文件   
  in = Resources.getResourceAsStream("SqlMapConfig.xml");   
  //2.创建构建者对象   
  SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();   
  //3.创建 SqlSession 工厂对象   
  factory = builder.build(in); 
  //4.创建 SqlSession 对象   
  session = factory.openSession(); 
  //5.创建 Dao 的代理对象   
  userDao = session.getMapper(IUserDao.class);  
  }   
 @After//在测试方法执行完成之后执行  
 public void destroy() throws Exception{   
 //7.提交事务 
 session.commit(); 
  //8.释放资源   
  session.close();   
  in.close();  
  }

通过分析能够发现之前的 CUD 操作过程中,我们都要手动进行事务的提交,原因是 setAutoCommit()方法,在执行时它的值被设置为 false 了,所以我们在 CUD 操作中, 必须通过 sqlSession.commit()方法来执行提交操作。
Mybatis 自动提交事务的设置

@Before//在测试方法执行之前执行 
public void init()throws Exception { 
 //1.读取配置文件  
 in = Resources.getResourceAsStream("SqlMapConfig.xml");  
 //2.创建构建者对象  
 SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); 
  //3.创建 SqlSession 工厂对象  
  factory = builder.build(in); 
 //4.创建 SqlSession 对象  
 session = factory.openSession(true);  
 //5.创建 Dao 的代理对象  
 userDao = session.getMapper(IUserDao.class); 
 }   
@After//在测试方法执行完成之后执行 
public void destroy() throws Exception{  
//7.释放资源 
session.close();  
in.close(); 
}

此时事务就设置为自动提交了,同样可以实现CUD操作时记录的保存。虽然这也是一种方式,但就编程而言,设置为自动提交方式为 false再根据情况决定是否进行提交,这种方式更常用。因为我们可以根据业务情况来决定提交是否进行提交

二、Mybatis 的动态 SQL 语句

1、动态SQL之 if 标签

<select id="findByUser" resultType="user" parameterType="user">  
select * from user where 1=1 
 <if test="username!=null and username != '' ">   
 and username like #{username}  
 </if> 
 <if test="address != null">   
 and address like #{address}  
 </if>    
 </select> 

注意:if标签的 test 属性中写的是对象的属性名,如果是包装类的对象要使用 OGNL 表达式的写法。另外要注意 where 1=1 的作用

2、动态 SQL 之where标签

为了简化上面 where 1=1 的条件拼装,我们可以采用标签来简化开发

<!-- 根据用户信息查询 -->  
<select id="findByUser" resultType="user" parameterType="user">   
<include refid="defaultSql"></include>   
<where> 
   <if test="username!=null and username != '' ">     
   and username like #{username}    
   </if> 
   <if test="address != null">     
   and address like #{address}    
   </if>   
   </where>  
   </select>

3、 动态标签之foreach标签

<!-- 查询所有用户在 id 的集合之中 -->  
<select id="findInIds" resultType="user" parameterType="queryvo">  
<!--  select * from user where id in (1,2,3,4,5); --> 
  <include refid="defaultSql"></include>   
  <where> 
   <if test="ids != null and ids.size() > 0">     
   <foreach collection="ids" open="id in ( " close=")" item="uid"  separator=","> 
        #{uid}     
    </foreach>    
    </if>   
    </where>  
    </select> 

SQL 语句:select 字段 from user where id in (?)
<oreach标签用于遍历集合,它的属性:
collection:代表要遍历的集合元素,注意编写时不要写#{}
open:代表语句的开始部分
close:代表结束部分
item:代表遍历集合的每个元素,生成的变量名
sperator:代表分隔符

三、 Mybatis 多表查询

1、 Mybatis 多表查询之一对一

<mapper namespace="com.itheima.dao.IAccountDao">  
<!-- 配置查询所有操作-->  
<select id="findAll" resultType="accountuser">   
select a.*,u.username,u.address from account a,user u where a.uid =u.id;  
</select> 
 </mapper> 
 

注意:因为上面查询的结果中包含了账户信息同时还包含了用户信息,所以我们的返回值类型 returnType 的值设置为 AccountUser 类型,这样就可以接收账户信息和用户信息了。

2、 Mybatis 多表查询之一对多

<mapper namespace="com.itheima.dao.IUserDao">   
 <resultMap type="user" id="userMap">   
 	<id column="id" property="id"></id>   
	 <result column="username" property="username"/>   
	 <result column="address" property="address"/>   
	 <result column="sex" property="sex"/>   
	 <result column="birthday" property="birthday"/>   
	 <!-- collection 是用于建立一对多中集合属性的对应关系    ofType 用于指定集合元素的数据类型    -->
<collection property="accounts" ofType="account">    
	<id column="aid" property="id"/>    
	<result column="uid" property="uid"/>    
	<result column="money" property="money"/> 
</collection>  
</resultMap> 
 
 <!-- 配置查询所有操作 -->  
 <select id="findAll" resultMap="userMap">   
 select u.*,a.id as aid ,a.uid,a.money from user u left outer join account a on u.id =a.uid
   </select> 
   </mapper>

collection 部分定义了用户关联的账户信息。表示关联查询结果集
property=“accList” :关联查询的结果集存储在 User 对象的上哪个属性。
ofType=“account” :指定关联查询的结果集中的对象类型即List中的对象类型。此处可以使用别名,也可以使用全限定名

3、 Mybatis 多表查询之多对多

实现查询所有对象并且加载它所分配的用户信息

<mapper namespace="com.itheima.dao.IRoleDao"> 
 <!--定义 role 表的 ResultMap-->     
 <resultMap id="roleMap" type="role">         
 	<id property="roleId" column="rid"></id>         
 	<result property="roleName" column="role_name"></result>         
 	<result property="roleDesc" column="role_desc"></result>         
 	<collection property="users" ofType="user">             
 		<id column="id" property="id"></id>             
 		<result column="username" property="username"></result>             
 		<result column="address" property="address"></result>             
 		<result column="sex" property="sex"></result>             
 		<result column="birthday" property="birthday"></result>        
 	 </collection>     
 </resultMap> 
 
    <!--查询所有-->     
    <select id="findAll" resultMap="roleMap">        
    select u.*,r.id as rid,r.role_name,r.role_desc from role r  
    left outer join user_role ur  on r.id = ur.rid         
    left outer join user u on u.id = ur.uid    
     </select> 
</mapper> 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值