MyBatis
Posted m0_56426304
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了MyBatis相关的知识,希望对你有一定的参考价值。
目录
JNDI
NDI:Java Naming and Directory Interface。是SUN公司推出的一套规范,属于JavaEE技术之一。目的是模仿windows系统中的注册表。
在服务器中注册数据源:
1.创建Maven的war工程并导入坐标
<dependencies>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.4.5</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.6</version>
</dependency>
<dependency>
<groupId>log4j</groupId>
<artifactId>log4j</artifactId>
<version>1.2.12</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.10</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.0</version>
</dependency>
</dependencies>
2.在webapp文件下创建META-INF目录
3.在META-INF目录中建立一个名为context.xml的配置文件
<?xml version="1.0" encoding="UTF-8"?>
<Context>
<!--
<Resource
name="jdbc/eesy_mybatis" 数据源的名称
type="javax.sql.DataSource" 数据源类型
auth="Container" 数据源提供者
maxActive="20" 最大活动数
maxWait="10000" 最大等待时间
maxIdle="5" 最大空闲数
username="root" 用户名
password="1234" 密码
driverClassName="com.mysql.jdbc.Driver" 驱动类
url="jdbc:mysql://localhost:3306/eesy_mybatis" 连接url字符串
/>
-->
<Resource
name="jdbc/eesy_mybatis"
type="javax.sql.DataSource"
auth="Container"
maxActive="20"
maxWait="10000"
maxIdle="5"
username="root"
password="1234"
driverClassName="com.mysql.jdbc.Driver"
url="jdbc:mysql://localhost:3306/eesy_mybatis"
/>
</Context>
4.修改SqlMapConfig.xml中的配置
<configuration>
<typeAliases>
<package name="com.itheima.domian"></package>
</typeAliases>
<!--配置mybatis的环境-->
<environments default="mysql">
<!--配置mysql的环境>
<environment id="mysql">
<!--配置事务控制的方式-->
<transactionManger type="JDBC"></transactionManager>
<dataSource type="JNDI">
<property name="data_source" value="java:comp/env/jdbc/eesy_mybatis0/">
</dataSource>
</environment>
</environments>
<!--指定mapper配置文件的位置-->
<mapper>
<mapper resource="com/itheima/dao/IUserDao.xml"/>
</mapper>
</configuration>
Mybatis中的延迟加载
什么是延迟加载——在真正使用数据时才发起查询,不用的时候不查询(按需查询(懒查询)) 延迟加载 : 就是在需要用到数据时才进行加载,不需要用到数据时就不加载数据。延迟加载也称懒加载 . 好处:先从单表查询,需要时再从关联表去关联查询,大大提高数据库性能,因为查询单表要比关联查询多张表速度要快 坏处: 因为只有当需要用到数据时,才会进行数据库查询,这样在大批量数据查询时,因为查询工作也要消耗时间,所以可能造成用户等待时间变长,造成用户体验下降 什么是立即加载——不管用不用,只要一调用方法,马上发起查询在对应的四种表关系中:一对多,多对一,一对一,多对多
一对多,多对多:通常情况下我们都是采用延迟加载
多对一,一对一:通常情况下我们都是采用立即加载
使用 assocation 实现延迟加载
需求: 查询账户信息同时查询用户信息。 1 账户的持久层 DAO 接口/**
*
* <p>Title: IAccountDao</p>
* <p>Description: 账户的持久层接口</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public interface IAccountDao
/**
* 查询所有账户,同时获取账户的所属用户名称以及它的地址信息
* @return
*/
List<Account> findAll();
2
账户的持久层映射文件
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.itheima.dao.IAccountDao">
<!-- 建立对应关系 -->
<resultMap type="account" id="accountMap">
<id column="aid" property="id"/>
<result column="uid" property="uid"/>
<result column="money" property="money"/>
<!-- 它是用于指定从表方的引用实体属性的 -->
<association property="user" javaType="user"
select="com.itheima.dao.IUserDao.findById"
column="uid">
</association>
</resultMap>
<select id="findAll" resultMap="accountMap">
select * from account
</select>
</mapper>
select : 填写我们要调用的 select 映射的 id column : 填写我们要传递给 select 映射的参数3 用户的持久层接口和映射文件
/**
*
* <p>Title: IUserDao</p>
* <p>Description: 用户的业务层接口</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public interface IUserDao
/**
* 根据 id 查询
* @param userId
* @return
*/
User findById(Integer userId);
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.itheima.dao.IUserDao">
<!-- 根据 id 查询 -->
<select id="findById" resultType="user" parameterType="int" >
select * from user where id = #uid
</select>
</mapper>
4
开启
Mybatis
的延迟加载策略
进入
Mybaits
的官方文档,找到
settings
的说明信息:
我们需要在
Mybatis
的配置文件
SqlMapConfig.xml
文件中添加延迟加载的配置。
<!-- 开启延迟加载的支持 -->
<settings>
<setting name="lazyLoadingEnabled" value="true"/>
<setting name="aggressiveLazyLoading" value="false"/>
</settings>
5
编写测试只查账户信息不查用户信息。
/**
*
* <p>Title: MybastisCRUDTest</p>
* <p>Description: 一对多账户的操作</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public class AccountTest
private InputStream in ;
private SqlSessionFactory factory;
private SqlSession session;
private IAccountDao accountDao;
@Test
public void testFindAll()
//6.执行操作
List<Account> accounts = accountDao.findAll();
@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 的代理对象
accountDao = session.getMapper(IAccountDao.class);
@After//在测试方法执行完成之后执行
public void destroy() throws Exception
//7.释放资源
session.close();
in.close();
测试结果如下:
我们发现,因为本次只是将
Account
对象查询出来放入
List
集合中,并没有涉及到
User
对象,所以就没有 发出 SQL
语句查询账户所关联的
User
对象的查询。
使用 Collection 实现延迟加载
同样我们也可以在一对多关系配置的 <collection> 结点中配置延迟加载策略。 <collection> 结点中也有 select 属性, column 属性。 需求: 完成加载用户对象时,查询该用户所拥有的账户信息。 1 在 User 实体类中加入 List<Account> 属性/**
*
* <p>Title: User</p>
* <p>Description: 用户的实体类</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public class User implements Serializable
private Integer id;
private String username;
private Date birthday;
private String sex;
private String address;
private List<Account> accounts;
public List<Account> getAccounts()
return accounts;
public void setAccounts(List<Account> accounts)
this.accounts = accounts;
public Integer getId()
return id;
public void setId(Integer id)
this.id = id;
public String getUsername()
return username;
public void setUsername(String username)
this.username = username;
public Date getBirthday()
return birthday;
public void setBirthday(Date birthday)
this.birthday = birthday;
public String getSex()
return sex;
public void setSex(String sex)
this.sex = sex;
public String getAddress()
return address;
public void setAddress(String address)
this.address = address;
@Override
public String toString()
return "User [id=" + id + ", username=" + username + ", birthday=" + birthday
+ ", sex=" + sex + ", address="
+ address + "]";
2
编写用户和账户持久层接口的方法
/**
* 查询所有用户,同时获取出每个用户下的所有账户信息
*/
List<User> findAll();
/**
* 根据用户 id 查询账户信息
* @param uid
* @return
*/
List<Account> findByUid(Integer uid);
3
编写用户持久层映射配置
<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 用于指定集合元素的数据类型
select 是用于指定查询账户的唯一标识(账户的 dao 全限定类名加上方法名称)
column 是用于指定使用哪个字段的值作为条件查询
-->
<collection property="accounts" ofType="account"
select="com.itheima.dao.IAccountDao.findByUid"
column="id">
</collection>
</resultMap>
<!-- 配置查询所有操作 -->
<select id="findAll" resultMap="userMap">
select * from user
</select>
<collection> 标签: 主要用于加载关联的集合对象 select 属性: 用于指定查询 account 列表的 sql 语句,所以填写的是该 sql 映射的 id column 属性: 用于指定 select 属性的 sql 语句的参数来源,上面的参数来自于 user 的 id 列,所以就写成 id 这一 个字段名了4 编写账户持久层映射配置
<!-- 根据用户 id 查询账户信息 -->
<select id="findByUid" resultType="account" parameterType="int">
select * from account where uid = #uid
</select>
5
测试只加载用户信息
/**
*
* <p>Title: MybastisCRUDTest</p>
* <p>Description: 一对多的操作</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public class UserTest
private InputStream in ;
private SqlSessionFactory factory;
private SqlSession session;
private IUserDao userDao;
@Test
public void testFindAll()
//6.执行操作
List<User> users = userDao.findAll();
@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
session.commit();
//7.释放资源
session.close();
in.close();
测试结果如下:
Mybatis中的缓存
——存在于内存中的临时数据
使用缓存可以减少和数据库的交互次数,提高执行效率。
什么样的数据能使用缓存,什么样的数据不能使用
适用于缓存:
经常查询并且不经常改变的。
数据的正确与否对最终结果影响不大的
不适用于缓存:
经常改变的数据
数据的正确与否对最终结果影响很大的
Mybatis中的一级缓存和二级缓存
一级缓存
一级缓存是 SqlSession 级别的缓存,只要 SqlSession 没有 flush 或 close,它就存在。
当我们执行查询之后,查询的结果会同时存入到SqlSession为我们提供一块区域中。
该区域的结构是一个Map。当我们再次查询同样的数据,mybatis会先去sqlsession中查询是否有,有的话直接拿出来用。当SqlSession对象消失时,mybatis的一级缓存也就消失了
/**
*
* <p>Title: IUserDao</p>
* <p>Description: 用户的业务层接口</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public interface IUserDao
/**
* 根据 id 查询
* @param userId
* @return
*/
User findById(Integer userId);
2
编写用户持久层映射文件
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.itheima.dao.IUserDao">
<!-- 根据 id 查询 -->
<select id="findById" resultType="UsEr" parameterType="int" useCache="true">
select * from user where id = #uid
</select>
</mapper>
3
编写测试方法
/**
*
* <p>Title: MybastisCRUDTest</p>
* <p>Description: 一对多的操作</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public class UserTest
private InputStream in ;
private SqlSessionFactory factory;
private SqlSession session;
private IUserDao userDao;
@Test
public void testFindById()
//6.执行操作
User user = userDao.findById(41);
System.out.println("第一次查询的用户:"+user);
User user2 = userDao.findById(41);
System.out.println("第二次查询用户:"+user2);
System.out.println(user == user2);
@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.close();
in.close();
测试结果如下: 我们可以发现,虽然在上面的代码中我们查询了两次,但最后只执行了一次数据库操作,这就是 Mybatis 提供给我们的一级缓存在起作用了。因为一级缓存的存在,导致第二次查询 id 为 41 的记录时,并没有发出 sql 语句从数据库中查询数据,而是从一级缓存中查询2.1一级缓存的分析 一级缓存是 SqlSession 范围的缓存, 当调用 SqlSession 的修改,添加,删除,commit(),close()等方法时,就会清空一级缓存 第一次发起查询用户 id 为 1 的用户信息,先去找缓存中是否有 id 为 1 的用户信息,如果没有,从数据库查询用户信息得到用户信息,将用户信息存储到一级缓存中 如果 sqlSession 去执行 commit 操作(执行插入、更新、删除),清空 SqlSession 中的一级缓存,这样做的目的为了让缓存中存储的是最新的信息,避免脏读 第二次发起查询用户 id 为 1 的用户信息,先去找缓存中是否有 id 为 1 的用户信息,缓存中有,直接从缓存中获取用户信息 2.2 测试一级缓存的清空
/**
* 测试一级缓存
*/
@Test
public void testFirstLevelCache()
User user1 = userDao.findById(41);
System.out.println(user1);
// sqlSession.close();
//再次获取 SqlSession 对象
// sqlSession = factory.openSession();
sqlSession.clearCache();//此方法也可以清空缓存
userDao = sqlSession.getMapper(IUserDao.class);
User user2 = userDao.findById(41);
System.out.println(user2);
System.out.println(user1 == user2);
/**
* 测试缓存的同步
*/
@Test
public void testClearlCache()
//1.根据 id 查询用户
User user1 = userDao.findById(41);
System.out.println(user1);
//2.更新用户信息
user1.setUsername("update user clear cache");
user1.setAddress("北京市海淀区");
userDao.updateUser(user1);
//3.再次查询 id 为 41 的用户
User user2 = userDao.findById(41);
System.out.println(user2);
System.out.println(user1 == user2);
当执行sqlSession.close()后,再次获取sqlSession并查询id=41的User对象时,又重新执行了sql 语句,从数据库进行了查询操作
二级缓存
它指的是Mybatis中SqlSessionFactory对象的缓存。二级缓存是 mapper 映射级别的缓存,多个 SqlSession 去操作同一个 Mapper 映射的 sql 语句,多个SqlSession 可以共用二级缓存,二级缓存是跨 SqlSession 的。
二级缓存的使用步骤:
第一步:让Mybatis框架支持二级缓存(在SqlMapConfig.xml中配置)
第二步:让当前的映射文件支持二级缓存(在IUserDao.xml中配置)
第三步:让当前的操作支持二级缓存(在select标签中配置)
<!-- 开启二级缓存的支持 -->
<setting name="cacheEnabled" value="true"/>
</settings>
因为
cacheEnabled
的取值默认就为
true
,所以这一步可以省略不配置。为
true
代表开启二级缓存;为false 代表不开启二级缓存。
第二步:配置相关的
Mapper
映射文件
<cache>标签表示当前这个 mapper 映射将使用二级缓存,区分的标准就看 mapper 的 namespace 值
。
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.itheima.dao.IUserDao">
<!-- 开启二级缓存的支持 -->
<cache></cache>
</mapper>
第三步:配置
statement
上面的
useCache
属性
<!-- 根据 id 查询 -->
<select id="findById" resultType="user" parameterType="int" useCache="true">
select * from user where id = #uid
</select>
将
UserDao.xml
映射文件中的
<select>
标签中设置
useCache=”true”
代表当前这个
statement
要使用二级缓存,如果不使用二级缓存可以设置为 false
。
注意:针对每次查询都需要最新的数据
sql
,要设置成
useCache=false
,禁用二级缓存。
3
二级缓存测试
/**
* @author 黑马程序员
* @Company http://www.ithiema.com
*/
public class SecondLevelCacheTest
private InputStream in;
private SqlSessionFactory factory;
@Before//用于在测试方法执行之前执行
public void init()throws Exception
//1.读取配置文件,生成字节输入流
in = Resources.getResourceAsStream("SqlMapConfig.xml");
//2.获取 SqlSessionFactory
factory = new SqlSessionFactoryBuilder().build(in);
@After//用于在测试方法执行之后执行
public void destroy()throws Exception
in.close();
/**
* 测试一级缓存
*/
@Test
public void testFirstLevelCache()
SqlSession sqlSession1 = factory.openSession();
IUserDao dao1 = sqlSession1.getMapper(IUserDao.class);
User user1 = dao1.findById(41);
System.out.println(user1);
sqlSession1.close();//一级缓存消失
SqlSession sqlSession2 = factory.openSession();
IUserDao dao2 = sqlSession2.getMapper(IUserDao.class);
User user2 = dao2.findById(41);
System.out.println(user2);
sqlSession2.close();
System.out.println(user1 == user2);
上面的测试执行了两次查询,并且在执行第一次查询后,我们关闭了一级缓存,再去执行第二 次查询时,并没有对数据库发出 sql
语句,所以此时的数据就只能是来自于我们所说的二级缓存。
4
二级缓存注意事项
当我们在使用二级缓存时,所缓存的类一定要实现
java.io.Serializable
接口,这种就可以使用序列化方式来保存对象
/**
*
* <p>Title: User</p>
* <p>Description: 用户的实体类</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public class User implements Serializable
private Integer id;
private String username;
private Date birthday;
private String sex;
private String address;
Mybatis中的注解开发
1 mybatis 的常用注解说明
@Insert: 实现新增 @Update: 实现更新 @Delete: 实现删除 @Select: 实现查询 @Result: 实现结果集封装 @Results: 可以与 @Result 一起使用,封装多个结果集 @ResultMap: 实现引用 @Results 定义的封装 @One: 实现一对一结果集封装 @Many: 实现一对多结果集封装 @SelectProvider: 实现动态 SQL 映射 @CacheNamespace: 实现注解二级缓存的使用2 使用 Mybatis 注解实现基本 CRUD
单表的 CRUD 操作是最基本的操作,前面我们的学习都是基于 Mybaits 的映射文件来实现的。 1 编写实体类/**
*
* <p>Title: User</p>
* <p>Description: 用户的实体类</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public class User implements Serializable
private Integer userId;
private String userName;
private Date userBirthday;
private String userSex;
private String userAddress;
public Integer getUserId()
return userId;
public void setUserId(Integer userId)
this.userId = userId;
public String getUserName()
return userName;
public void setUserName(String userName)
this.userName = userName;
public Date getUserBirthday()
return userBirthday;
public void setUserBirthday(Date userBirthday)
this.userBirthday = userBirthday;
public String getUserSex()
return userSex;
public void setUserSex(String userSex)
this.userSex = userSex;
public String getUserAddress()
return userAddress;
public void setUserAddress(String userAddress)
this.userAddress = userAddress;
@Override
public String toString()
return "User [userId=" + userId + ", userName=" + userName + ", userBirthday="
+ userBirthday + ", userSex="
+ userSex + ", userAddress=" + userAddress + "]";
注意:
此处我们故意和数据库表的列名不一致。
2 使用注解方式开发持久层接口
/**
*
* <p>Title: IUserDao</p>
* <p>Description: 用户的持久层接口</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public interface IUserDao
/**
* 查询所有用户
* @return
*/
@Select("select * from user")
@Results(id="userMap",
value=
@Result(id=true,column="id",property="userId"),
@Result(column="username",property="userName"),
@Result(column="sex",property="userSex"),
@Result(column="address",property="userAddress"),
@Result(column="birthday",property="userBirthday")
)
List<User> findAll();
/**
* 根据 id 查询一个用户
* @param userId
* @return
*/
@Select("select * from user where id = #uid ")
@ResultMap("userMap")
User findById(Integer userId);
/**
* 保存操作
* @param user
* @return
*/
@Insert("insert into
user(username,sex,birthday,address)values(#username,#sex,#birthday,#address
)")
@SelectKey(keyColumn="id",keyProperty="id",resultType=Integer.class,before =
false, statement = "select last_insert_id()" )
int saveUser(User user);
/**
* 更新操作
* @param user
* @return
*/
@Update("update user set
username=#username,address=#address,sex=#sex,birthday=#birthday where id
=#id ")
int updateUser(User user);
/**
* 删除用户
* @param userId
* @return
*/
@Delete("delete from user where id = #uid ")
int deleteUser(Integer userId);
/**
* 查询使用聚合函数
* @return
*/
@Select("select count(*) from user ")
int findTotal();
/**
* 模糊查询
* @param name
* @return
*/
@Select("select * from user where username like #username ")
List<User> findByName(String name);
通过注解方式,我们就不需要再去编写 UserDao.xml 映射文件了
3 编写 SqlMapConfig 配置文件
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<!-- 配置 properties 文件的位置 -->
<properties resource="jdbcConfig.properties"></properties>
<!-- 配置别名的注册 -->
<typeAliases>
<package name="com.itheima.domain"/>
</typeAliases>
<!-- 配置环境 -->
<environments default="mysql">
<!-- 配置 mysql 的环境 -->
<environment id="mysql">
<!-- 配置事务的类型是 JDBC -->
<transactionManager type="JDBC"></transactionManager>
<!-- 配置数据源 -->
<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>
</environment>
</environments>
<!-- 配置映射信息 -->
<mappers>
<!-- 配置 dao 接口的位置,它有两种方式
第一种:使用 mapper 标签配置 class 属性
第二种:使用 package 标签,直接指定 dao 接口所在的包
-->
<package name="com.itheima.dao"/>
</mappers>
</configuration>
4
编写测试方法
/**
*
* <p>Title: MybatisAnnotationCRUDTest</p>
* <p>Description: mybatis 的注解 crud 测试</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public class MybatisAnnotationCRUDTest
/**
* 测试查询所有
*/
@Test
public void testFindAll()
List<User> users = userDao.findAll();
for(User user : users)
System.out.println(user);
/**
* 测试查询一个
*/
@Test
public void testFindById()
User user = userDao.findById(41);
System.out.println(user);
/**
* 测试保存
*/
@Test
public void testSave()
User user = new User();
user.setUserName("mybatis annotation");
user.setUserSex("男");
user.setUserAddress("北京市顺义区");
user.setUserBirthday(new Date());
int res = userDao.saveUser(user);
System.out.println("影响数据库记录的行数:"+res);
System.out.println("插入的主键值:"+user.getUserId());
/**
* 测试更新
*/
@Test
public void testUpdate()
User user = userDao.findById(63);
user.setUserBirthday(new Date());
user.setUserSex("女");
int res = userDao.updateUser(user);
System.out.println(res);
/**
* 测试删除
*/
@Test
public void testDelete()
int res = userDao.deleteUser(63);
System.out.println(res);
/**
* 测试查询使用聚合函数
*/
@Test
public void testFindTotal()
int res = userDao.findTotal();
System.out.println(res);
/**
* 测试模糊查询
*/
@Test
public void testFindByName()
List<User> users = userDao.findByName("%m%");
for(User user : users)
System.out.println(user);
private InputStream in;
private SqlSessionFactory factory;
private SqlSession session;
private IUserDao userDao;
@Before//junit 的注解
public void init()throws Exception
//1.读取配置文件
in = Resources.getResourceAsStream("SqlMapConfig.xml");
//2.创建工厂
SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
factory = builder.build(in);
//3.创建 session
session = factory.openSession();
//4.创建代理对象
userDao = session.getMapper(IUserDao.class);
@After//junit 的注解
public void destroy()throws Exception
//提交事务
session.commit();
//释放资源
session.close();
//关闭流
in.close();
3 使用注解实现复杂关系映射开发
实现复杂关系映射之前我们可以在映射文件中通过配置<resultMap>来实现,在使用注解开发时我们需要借 助@Results 注解, @Result 注解, @One 注解, @Many 注解 1 复杂关系映射的注解说明@Results 注解 代替的是标签 <resultMap> 该注解中可以使用单个 @Result 注解,也可以使用 @Result 集合 @Results ( @Result (), @Result () )或 @Results ( @Result ()) @Resutl 注解 代替了 <id> 标签和 <result> 标签 @Result 中 属性介绍: id 是否是主键字段 column 数据库的列名 property 需要装配的属性名 one 需要使用的 @One 注解( @Result ( one=@One )())) many 需要使用的 @Many 注解( @Result ( many=@many )())) @One 注解(一对一) 代替了 <assocation> 标签,是多表查询的关键,在注解中用来指定子查询返回单一对象。 @One 注解属性介绍: select 指定用来多表查询的 sqlmapper fetchType 会覆盖全局的配置参数 lazyLoadingEnabled 。。 使用格式: @Result (column= " " ,property= "" ,one= @One (select= "" )) @Many 注解(多对一) 代替了 < Collection > 标签 , 是是多表查询的关键,在注解中用来指定子查询返回对象集合。 注意:聚集元素用来处理“一对多”的关系。需要指定映射的 Java 实体类的属性,属性的 javaType (一般为 ArrayList )但是注解中可以不定义; 使用格式: @Result (property= "" ,column= "" ,many= @Many (select= "" ))2 使用注解实现一对一复杂关系映射及延迟加载 需求: 加载账户信息时并且加载该账户的用户信息,根据情况可实现延迟加载。(注解方式实现) 2.1 添加 User 实体类及 Account 实体类
/**
*
* <p>Title: User</p>
* <p>Description: 用户的实体类</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public class User implements Serializable
private Integer userId;
private String userName;
private Date userBirthday;
private String userSex;
private String userAddress;
public Integer getUserId()
return userId;
public void setUserId(Integer userId)
this.userId = userId;
public String getUserName()
return userName;
public void setUserName(String userName)
this.userName = userName;
public Date getUserBirthday()
return userBirthday;
public void setUserBirthday(Date userBirthday)
this.userBirthday = userBirthday;
public String getUserSex()
return userSex;
public void setUserSex(String userSex)
this.userSex = userSex;
public String getUserAddress()
return userAddress;
public void setUserAddress(String userAddress)
this.userAddress = userAddress;
@Override
public String toString()
return "User [userId=" + userId + ", userName=" + userName + ", userBirthday="
+ userBirthday + ", userSex="
+ userSex + ", userAddress=" + userAddress + "]";
/**
*
* <p>Title: Account</p>
* <p>Description: 账户的实体类</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public class Account implements Serializable
private Integer id;
private Integer uid;
private Double money;
//多对一关系映射:从表方应该包含一个主表方的对象引用
private User user;
public User getUser()
return user;
public void setUser(User user)
this.user = user;
public Integer getId()
return id;
public void setId(Integer id)
this.id = id;
public Integer getUid()
return uid;
public void setUid(Integer uid)
this.uid = uid;
public Double getMoney()
return money;
public void setMoney(Double money)
this.money = money;
@Override
public String toString()
return "Account [id=" + id + ", uid=" + uid + ", money=" + money + "]";
2.2
添加账户的持久层接口并使用注解配置
/**
*
* <p>Title: IAccountDao</p>
* <p>Description: 账户的持久层接口</p>
* <p>Company: http://www.itheima.com/ </p>
*/
public interface IAccountDao
/**
* 查询所有账户,采用延迟加载的方式查询账户的所属用户
* @return
*/
@Select("select * from account")
@Results(id="accountMap",
value=
@Result(id=true,column="id",property="id"),
@Result(column="uid",property="uid"),
@Result(column="money",property="money"),
@Result(column="uid",
property="user",
one=以上是关于MyBatis的主要内容,如果未能解决你的问题,请参考以下文章
将 Excel UDF 放入 Worksheet 对象(而不是在模块中)