一、延迟加载
实际开发过程中很多时候我们并不需要总是在加载用户信息时就一定要加载他的账户信息。 此时就是我们所说的延迟加载
延迟加载:
就是在需要用到数据时才进行加载,不需要用到数据时就不加载数据。
延迟加载也称懒加载
- 好处: 先从单表查询,需要时再从关联表去关联查询,大大提高数据库性能,因为查询单表要比关联查询多张表速度要快
- 坏处:因为只有当需要用到数据时,才会进行数据库查询,这样在大批量数据查询时,因为查询工作也要消耗时间,所以可能造成用户等待时间变长,造成用户体验下降。
主要使用 assocation
和 collection
实现延迟加载,这里略过
二、Mybatis缓存
像大多数的持久化框架一样, Mybatis 也提供了缓存策略,通过缓存策略来减少数据库的查询次数, 从而提高性能
Mybatis 中缓存分为一级缓存,二级缓存
2.1 一级缓存
一级缓存是 SqlSession 级别的缓存,只要 SqlSession 没有 flush 或 close,它就存在
2.1.1 实现
public class UserTest {
private InputStream in;
private SqlSession sqlSession;
private IUserDao dao;
//执行前运行
@Before
public void init() throws IOException {
in = Resources.getResourceAsStream("SqlMapConfig.xml");
SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(in);
sqlSession = factory.openSession();
dao = sqlSession.getMapper(IUserDao.class);
}
//执行后回收
@After
public void destroy() throws IOException {
sqlSession.close();
in.close();
}
//测试一级缓存
@org.junit.Test
public void testFirstLevelCache() {
User user1 = dao.findById(41);
System.out.println(user1);
//如果清除缓存,user1和user2就不是同一个对象了
sqlSession.clearCache();
User user2 = dao.findById(41);
System.out.println(user2);
//如果不清除缓存 user1会等于user2 只发起一次查询
System.out.println(user1 == user2);
}
//测试缓存同步
@org.junit.Test
public void testClearCache() {
User user1 = dao.findById(41);
user1.setUsername("哈哈");
user1.setAddress("纽约");
//更新id=41的客户
dao.updateUser(user1);
sqlSession.commit();
//再次查id=41
User user2 = dao.findById(41);
System.out.println(user1 == user2);
}
我们可以发现,虽然在上面的代码中我们查询了两次,但最后只执行了一次数据库操作.这就是 Mybatis 提供给我们的一级缓存在起作用了。
因为一级缓存的存在,导致第二次查询 id 为 41 的记录时,并没有发出 sql 语句从数据库中查询数据,而是从一级缓存中查询。
2.1.2 一级缓存的分析
一级缓存是 SqlSession 范围的缓存,当调用 SqlSession 的修改,添加,删除, commit(), close()等方法时,就会清空一级缓存。
-
第一次发起查询用户 id 为 1 的用户信息,先去找缓存中是否有 id 为 1 的用户信息如果没有,从数据库查询用户信息。
-
得到用户信息,将用户信息存储到一级缓存中。
-
如果 sqlSession 去执行 commit 操作(执行插入、更新、删除),清空 SqlSession中的一级缓存,这样做的目的为了让缓存中存储的是最新的信息,避免脏读
-
第二次发起查询用户 id 为 1 的用户信息,先去找缓存中是否有 id 为 1 的用户信息,缓存中有,直接从缓存中获取用户信息
2.1.3 缓存的同步
当执行更新操作后后,再次获取sqlSession并查询id=41的User对象时,又重新执行了sql语句,从数据库进行了查询操作。
//测试缓存同步
@org.junit.Test
public void testClearCache() {
User user1 = dao.findById(41);
user1.setUsername("哈哈");
user1.setAddress("纽约");
//更新id=41的客户
dao.updateUser(user1);
sqlSession.commit();
//再次查id=41
User user2 = dao.findById(41);
System.out.println(user1 == user2);
}
2.2 二级缓存
二级缓存是 mapper 映射级别的缓存,多个 SqlSession 去操作同一个 Mapper 映射的sql 语句,多个SqlSession 可以共用二级缓存,二级缓存是跨 SqlSession 的 。
2.2.1 缓存结构
sqlSession1 去查询用户信息,查询到用户信息会将查询数据存储到二级缓存中
如果 SqlSession3 去执行相同 mapper 映射下 sql,执行 commit 提交, 将会清空该mapper 映射下的二级缓存区域的数据
如果 SqlSession3 去执行相同 mapper 映射下 sql,执行 commit 提交, 将会清空该mapper 映射下的二级缓存区域的数据
2.2.2 开启二级缓存
SqlMapConfig.xml 文件配置
<settings>
<!-- 开启二级缓存的支持 -->
<setting name="cacheEnabled" value="true"/>
</settings>
因为 cacheEnabled 的取值默认就为 true,所以这一步可以省略不配置。为 true 代表开启二级缓存;为false 代表不开启二级缓存
Mapper 映射文件配置
<cache>标签表示当前这个 mapper 映射将使用二级缓存,区分的标准就看 mapper 的 namespace 值。
<mapper namespace="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,禁用二级缓存。
2.2.3 注意事项
当我们在使用二级缓存时,所缓存的类一定要实现 java.io.Serializable 接口,这种可以使用序列化方式来保存对象
public class User implements Serializable {
private Integer id;
private String username;
private Date birthday;
private String sex;
private String address;
}
三、注解开发
介绍一下简单的注解开发crud,目录结构如下
3.1 SqlMapConfig.xml
<?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 resource="jdbcConfig.properties"></properties>
<!--配置别名-->
<typeAliases>
<package name="domain"/>
</typeAliases>
<!--配置数据源-->
<environments default="mysql">
<environment id="mysql">
<transactionManager type="JDBC"></transactionManager>
<dataSource type="POOLED">
<property name="driver" value="${driver}"/>
<property name="url" value="${url}"/>
<property name="username" value="${username}"/>
<property name="password" value="${password}"/>
</dataSource>
</environment>
</environments>
<!--指定带有注解的dao接口所在位置-->
<mappers>
<package name="dao"/>
</mappers>
</configuration>
3.2 jdbcConfig.properties
driver=com.mysql.jdbc.Driver
url=jdbc:mysql://localhost:3306/mybatis?useUnicode=true&characterEncoding=utf-8
username=root
password=xxxxx
3.3 log4j.properties
# Set root category priority to INFO and its only appender to CONSOLE.
#log4j.rootCategory=INFO, CONSOLE debug info warn error fatal
log4j.rootCategory=debug, CONSOLE, LOGFILE
# Set the enterprise logger category to FATAL and its only appender to CONSOLE.
log4j.logger.org.apache.axis.enterprise=FATAL, CONSOLE
# CONSOLE is set to be a ConsoleAppender using a PatternLayout.
log4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender
log4j.appender.CONSOLE.layout=org.apache.log4j.PatternLayout
log4j.appender.CONSOLE.layout.ConversionPattern=%d{ISO8601} %-6r [%15.15t] %-5p %30.30c %x - %m\n
# LOGFILE is set to be a File appender using a PatternLayout.
log4j.appender.LOGFILE=org.apache.log4j.FileAppender
log4j.appender.LOGFILE.File=d:\axis.log
log4j.appender.LOGFILE.Append=true
log4j.appender.LOGFILE.layout=org.apache.log4j.PatternLayout
log4j.appender.LOGFILE.layout.ConversionPattern=%d{ISO8601} %-6r [%15.15t] %-5p %30.30c %x - %m\n
3.4 User.java
package domain;
import java.io.Serializable;
import java.sql.Date;
/**
* @author konley
* @date 2020-08-11 11:35
*/
public class User implements Serializable {
private Integer id;
private String username;
private String address;
private String sex;
private Date birthday;
public User() {
}
@Override
public String toString() {
return "User{" +
"id=" + id +
", username='" + username + '\'' +
", address='" + address + '\'' +
", sex='" + sex + '\'' +
", birthday=" + birthday +
'}';
}
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 String getAddress() {
return address;
}
public void setAddress(String address) {
this.address = address;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public Date getBirthday() {
return birthday;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
}
3.5 IUserDao.java
package dao;
import domain.User;
import org.apache.ibatis.annotations.Delete;
import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Select;
import org.apache.ibatis.annotations.Update;
import java.util.List;
/**
* @author konley
* @date 2020-08-11 11:38
*/
public interface IUserDao {
/**
* description: 查询所有用户
* @param
* @return java.util.List<domain.User>
*/
@Select("select * from user")
List<User> findAll();
/**
* description: 保存用户
* @param user
* @return void
*/
@Insert("insert into user(username,address,sex,birthday)values" +
"(#{username},#{address},#{sex},#{birthday})")
void saveUser(User user);
/**
* description: 更新用户
* @param user
* @return void
*/
@Update("update user set username=#{username},sex=#{sex}," +
"address=#{address},birthday=#{birthday} where id=#{id}")
void update(User user);
/**
* description: 删除用户
* @param userId
* @return void
*/
@Delete("delete from user where id=#{id}")
void deleteUser(Integer userId);
/**
* description: 根据id查用户
* @param userId
* @return domain.User
*/
@Select("select * from user where id=#{id}")
User findById(Integer userId);
/**
* description: 根据名称模糊查询
* @param name
* @return java.util.List<domain.User>
*/
@Select("select * from user where username like #{username}")
List<User> findByName(String name);
/**
* description: 查询总记录条数
* @param
* @return int
*/
@Select("select count(id) from user")
int findTotal();
}
3.6 测试类
import dao.IUserDao;
import domain.User;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.After;
import org.junit.Before;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
/**
* @author konley
* @date 2020-08-11 11:48
*/
public class Test {
private static InputStream in;
private static SqlSession sqlSession;
private static IUserDao iUserDao;
@Before
public void before() throws IOException {
in = Resources.getResourceAsStream("SqlMapConfig.xml");
sqlSession = new SqlSessionFactoryBuilder().build(in).openSession();
iUserDao = sqlSession.getMapper(IUserDao.class);
}
@After
public void after() throws IOException {
sqlSession.commit();
sqlSession.close();
in.close();
}
/*
* 测试注解的查询所有方法
* */
@org.junit.Test
public void test1() throws IOException {
List<User> users = iUserDao.findAll();
for (User user : users) {
System.out.println(user);
}
}
/*
* 测试注解的添加用户方法
* */
@org.junit.Test
public void test2() {
User user = new User();
user.setUsername("lla");
user.setAddress("北京");
user.setSex("男");
iUserDao.saveUser(user);
}
/*
* 测试注解的更新用户方法
* */
@org.junit.Test
public void test3() {
User user = new User();
user.setId(49);
user.setUsername("略略略");
user.setAddress("天津");
user.setSex("女");
iUserDao.update(user);
}
/*
* 测试注解的删除用户方法
* */
@org.junit.Test
public void test4() {
iUserDao.deleteUser(50);
}
/*
* 测试注解的精确查询用户方法
* */
@org.junit.Test
public void test5(){
User user = iUserDao.findById(49);
System.out.println(user);
}
/*
* 测试注解的模糊查询用户方法
* */
@org.junit.Test
public void test6(){
List<User> users = iUserDao.findByName("%小%");
for (User user : users) {
System.out.println(user);
}
}
/*
* 测试注解的统计记录数方法
* */
@org.junit.Test
public void test7(){
int total = iUserDao.findTotal();
System.out.println(total);
}
}