Mybatis

## 1,简介

#### 1.1,什么是Mybatis

* MyBatis 是一款优秀的**持久层框架**
* 它支持自定义 SQL、存储过程以及高级映射。
* MyBatis 免除了几乎所有的 JDBC 代码以及设置参数和获取结果集的工作。MyBatis 可以通过简单的 XML 或注解来配置和映射原始类型、接口和 Java POJO(Plain Old Java Objects,普通老式 Java 对象)为数据库中的记录。
* MyBatis本是apache的一个[开源项目](https://baike.baidu.com/item/开源项目/3406069)iBatis,
* 2010年这个[项目](https://baike.baidu.com/item/项目/477803)由apache software foundation迁移到了[google code](https://baike.baidu.com/item/google code/2346604),并且改名为MyBatis。
* 2013年11月迁移到[Github](https://baike.baidu.com/item/Github/10145341)。

如何获取Mybatis

* maven仓库
* Github
* 中文文档

#### 1.2,持久化

数据持久化

* 持久化就是将程序的持久状态和瞬时状态转化的过程
* 内存:断电即失
* 数据库(jdbc),io文件持久化

#### 1.3,持久层

Dao层,Service层,Controller层

* 完成持久化工作的代码块
* 层界限十分明显

#### 1.4,为什么需要Mybatis

* 方便
* 传统JDBC代码太复杂,简化,框架
* 帮助程序员将数据存入到数据库中
* 不要Mybatis也可以,更容易上手

## 2,第一个Mybatis程序

思路:搭建环境->导入Mybatis->编写代码->测试

#### 2.1,搭建环境

搭建数据库

新建项目

* 新建一个普通的MAVEN项目
* 删除src目录

#### 2.2,创建一个模块

* 导入jar包

```xml
<dependencies>

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.13</version>
        </dependency>
        <!--mybatis-->
        <dependency>
            <groupId>org.mybatis</groupId>
            <artifactId>mybatis</artifactId>
            <version>3.5.2</version>
        </dependency>
        <!--juit-->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
        </dependency>
        <dependency>
            <groupId>org.apache.logging.log4j</groupId>
            <artifactId>log4j-api</artifactId>
            <version>2.17.2</version>
        </dependency>
        <dependency>
            <groupId>org.apache.logging.log4j</groupId>
            <artifactId>log4j-core</artifactId>
            <version>2.17.2</version>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.10</version>
        </dependency>

    </dependencies>

```

* 编写mybatis的核心配置文件

```java
//建立连接
<?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核心配置文件-->
    <configuration>

    <environments default="development">
        <environment id="development">
            <transactionManager type="JDBC"/>
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.cj.jdbc.Driver"/>//注意 新的版本驱动需要多加一个cj
                <property name="url" value="jdbc:mysql://localhost:3306/mybatis?        useSSL=true&amp;useUnicode=true&amp;characterEncoding=UTF-8&amp;serverTimezone=GMT%2B8"/>
                <property name="username" value="root"/>
                <property name="password" value="123456"/>
            </dataSource>
        </environment>
    </environments>
</configuration>

```

* 编写mybatis工具类

```java
package java.com.chen.util;

import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.apache.ibatis.io.Resources;
import java.io.InputStream;

//sqlSessionFactory --> sqlSession
public class MybatisUtil {
    private static SqlSessionFactory sqlSessionFactory;
    static {
        try {
            //获取SqlSessionFactory对象
            String resource = "mybatis-config.xml";
            InputStream inputStream = Resources.getResourceAsStream(resource);
            sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }//有了Sql工厂,我们就可以执行sql命令所需要的所有方法
    public static SqlSession getSqlSession(){
        SqlSession sqlSession=sqlSessionFactory.openSession();
        return sqlSession;

    }
}


```

#### 2.3,编写代码

* 实体类

```java
package java.com.chen.pojo;
public class user {
    private  int id;
    private  String name;
    private  String pwd;
    public user(int id, String name, String pwd) {
        this.id = id;
        this.name = name;
        this.pwd = pwd;
    }
    public user(){
    }

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getPwd() {
        return pwd;
    }

    public void setPwd(String pwd) {
        this.pwd = pwd;
    }

    @Override
    public String toString() {
        return "user{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", pwd='" + pwd + '\'' +
                '}';
    }
}
```

* Dao接口

```java
//获取实体类,相当于获取数据库里的关键字
package java.com.chen.dao;
import java.com.chen.pojo.user;
import java.util.List;
public interface UserDao {
    //获取全部的对象
    List<user> getUserList();
    //根据id查询用户
    user getUserId(int id)
    //插入对象
    int addUser(User user)
    //修改用户
    int updateUser(User user);
    //删除用户
    int deleteUser(User user);
}
```

* 接口实现类

```java
<?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">
    //namespace=绑定一个对应的Dao/Mapper接口
<mapper namespace="Dao.UserDao">//想当于实现Dao接口
    //查询所有对象<select
    <select id="getUserList" resultType="com.dao.user">//id相当于重写接口中的方法,结果集=集合的泛型
        select * from mybatis.user;//相当于在方法中写sql命令
    </select>
        //根据id查询用户<select
        <select id="getUserId" parameterType="int" resultType="com.dao.user"
        select * from mybatis.user where id=#{id}
        //插入对象标签一定要改 <insert
  <insert id="getUserId" parameterType="int" resultType="com.dao.user"
        insert into mybatis.user(id,name,pwd) values(#{id},#{name},#{pwd})
      //修改用户<update
      <update id="getUserId" parameterType="com.dao.user" resultType="com.dao.user"
        update mybatis.user set  name=#{name},pwd=#{pwd} where id=#{id};
    //删除用户<delete
<delete id="getUserId">
     delete  from mybatis.user where id=#{id};//这里的sql命令不需要加星号
        
</mapper>
```

#### 2.4,测试

```java
package chen;
import org.apache.ibatis.session.SqlSession;
import com.dao.user;
import com.dao.MybatisUtil;
import java.util.List;
//增删改查必须要提交事务
//sqlSession.commit();
public class UserDaoTast {
    public void tast(){
        //获取sqlsession对象
        SqlSession sqlSession= MybatisUtil.getSqlSession();
        //执行sql
        Dao.UserDao userDao = sqlSession.getMapper(Dao.UserDao.class);//获取接口Dao的实例
        List<user> list= userDao.getUserList();//获取接口Dao的方法
        for(user user:list){
            System.out.println(user);
        }
        sqlSession.close();

    }

    public static void main(String[] args) {
        UserDaoTast userDaoTast=new UserDaoTast();
        userDaoTast.tast();
    }
}
```

## 3,CRUD

#### 3.1,namespace

namespace的包名要和接口的包名一致

选择,查询语句

* id:就是对应namespace中的方法名
* resultType:Sql语句执行的返回值
* paramentType:参数类型

#### 3.2,万能MAP

当实体类中的类型太多,就可以调用map对象来代替

```java
public interface UserDao {
    //插入用户
    int InsertUser(Map<String,Object> map);
}
```

```xml
 <insert id="getUserId" parameterType="map" resultType="com.dao.user"
        insert into mybatis.user(id,name,pwd) values(#{userid},#{username},#{userpwd})
         <insert/>
         //用了Map对象,这里的值可以不用跟实体类中的一致
```

```java
        //获取sqlsession对象
        SqlSession sqlSession= MybatisUtil.getSqlSession();
        //执行sql
        Dao.UserDao userDao = sqlSession.getMapper(Dao.UserDao.class);//获取接口Dao的实例
        Map<String,Object> map=new Map();
        map.put("userid",5);
        map.put("username","dasd");
        userDao.InsertUser(map);
```

Map传递参数,直接在sql中取出key即可

对象传递参数,直接在sql中取出属性即可

只有一个基本类型的情况下,可以直接在sql中取到

多个参数用Map,或者注解

## 4,配置解析

#### 4.1,核心配置文件

* mybatis-config-xml
* Mybatis的配置文件包含了会深深影响MyBatis行为的设置和属性信息

```java
configuration(配置)
properties(属性)
settings(设置)
typeAliases(类型别名)
typeHandlers(类型处理器)
objectFactory(对象工厂)
plugins(插件)
environments(环境配置)
environment(环境变量)
transactionManager(事务管理器)
dataSource(数据源)
databaseIdProvider(数据库厂商标识)
mappers(映射器)
```

#### 4.2,环境配置(environments)

MyBatis 可以配置成适应多种环境

**不过要记住:尽管可以配置多个环境,但每个 SqlSessionFactory 实例只能选择一种环境。**

```java
 <environments default="development">
        <environment id="development">//选择的这个
            <transactionManager type="JDBC"/>//事务管理器,默认的JDBC
            <dataSource type="POOLED">//数据源,默认:池(POOLED)
```

**事务管理器(transactionManager)**

在 MyBatis 中有两种类型的事务管理器(也就是 type="[JDBC|MANAGED]"):

- JDBC – 这个配置直接使用了 JDBC 的提交和回滚设施,它依赖从数据源获得的连接来管理事务作用域。
- MANAGED – 这个配置几乎没做什么。

**数据源(dataSource)**

dataSource 元素使用标准的 JDBC 数据源接口来配置 JDBC 连接对象的资源。

- 大多数 MyBatis 应用程序会按示例中的例子来配置数据源。虽然数据源配置是可选的,但如果要启用延迟加载特性,就必须配置数据源。

有三种内建的数据源类型(也就是 type="[UNPOOLED|POOLED|JNDI]");

#### 4.3,属性(properties)

**properties必须在最上面引用,按照上面核心配置文件的顺序来排,properties在第一排所有是在最上面**

我们可以通过properties属性来实现引用配置文件

这些属性可以在外部进行配置,并可以进行动态替换。你既可以在典型的 Java 属性文件中配置这些属性,也可以在 properties 元素的子元素中设置。

dp.properties

```text
dirver=com.mysql.jdbc.Driver
url="jdbc:mysql://localhost:3306/mybatisuseSSL=false&useUnicode=true&characterEncoding=UTF8&serverTimezone=GMT%2B8"/>
name=root
password=123456
```

```xml
    <!--引用外部配置文件-->
<properties
        resource="db.properties">//先读
         <properties name="username" values="root"//后读
    </properties>

<property name="driver" value="${driver}"/>
  <property name="url" value="${url}"/>
  <property name="username" value="${username}"/>
  <property name="password" value="${password}"/>
```

* 可以直接引入外部文件
* 可以在其中增加一些属性配置
* 读取时,先读取外部配置文件中的数据

#### 4.4,类型别名(typeAliases)

* 类型别名可为 Java 类型设置一个缩写名字
* 存在的意义在降低冗余的全限定类名书写

```xml
  <!--给实体类取一个别名-->
 <typeAliases>
      <typeAlias type="com.dao.user" alias="User"/>
  </typeAliases>
//这些所有的类型调用时名字可以直接用User
```

也可以指定一个包名

扫描实体类的包,它的默认别名就为这个类的类名首字母小写

```xml
 <typeAliases>
      <package name="com.dao"/>
  </typeAliases>
```

在实体类比较少的时候使用第一种

如果实体类比较多时就使用第二种

第一种可以DIY别名,第二种不行,如果非要改,需要在实体上增加注解

```java
@Alias("user")
public class user{
    
}
```

#### 4.5,设置(settings)

这是 MyBatis 中极为重要的调整设置,它们会改变 MyBatis 的运行时行为

```java
no_sql  nosql
```

#### 4.6,作用域(Scope)和生命周期

生命周期和作用域类别是至关重要的,因为错误的使用会导致非常严重的并发问题。

SqlSessionFactoryBuild

* 一旦创建了 SqlSessionFactory,就不再需要它了
* 局部变量

SqlSessionFactory

* 数据库连接池
* 一旦被创建就应该在应用的运行期间一直存在,没有任何理由丢弃它或重新创建另一个实例。

SqlSession

* 连接池的一个请求
* 用完必须关闭
* SqlSession 的实例不是线程安全的,因此是不能被共享的,所以它的最佳的作用域是请求或方法作用域

## 5,解决属性名和字段名不一致的问题

#### 5.1ResultMap

```java
id name pwd
id name password
```

```xml
  <!--结果集映射-->
    <resultMap id="Usermap" type="user">
        <!--column数据库中的字段,Property实体类中的属性-->
        <result column="id" property="id"/>
        <result column="name" property="name"/>
        <result column="pwd" property="password"/>
    </resultMap>
    <select id="getUserList" resultMap="Usermap">      
    </select>
```

* `resultMap` 元素是 MyBatis 中最重要最强大的元素
* ResultMap 的设计思想是,对简单的语句做到零配置,对于复杂一点的语句,只需要描述语句之间的关系就行了。

## 6,日志

#### 6.1,日志工厂

如果一个数据库操作出现了异常,我们需要排错,日志就是最好的助手

曾经:Debug,sout

现在:日志工厂

| logImpl | 指定 MyBatis 所用日志的具体实现,未指定时将自动查找。 |
| ------- | ----------------------------------------------------- |
|         |                                                       |

- SLF4J
- Apache Commons Logging
- Log4j 2(掌握)
- Log4j 
- JDK logging
- STDOUT_LOGGING(掌握)

在Mybatis中具体使用哪一个日志实现,在设置中设定

**STDOUT_LOGGING标准日志输出**

```xml
    <settings>
        <setting name="logImpl" value="STDOUT_LOGGING"/>
    </settings>
```

Log4j

* Log4j是[Apache](https://baike.baidu.com/item/Apache/8512995)的一个开源项目,通过使用Log4j,我们可以控制日志信息输送的目的地是[控制台](https://baike.baidu.com/item/控制台/2438626)、文件、[GUI](https://baike.baidu.com/item/GUI)组件,甚至是套接口服务器、[NT](https://baike.baidu.com/item/NT/3443842)的事件记录器、[UNIX](https://baike.baidu.com/item/UNIX) [Syslog](https://baike.baidu.com/item/Syslog)[守护进程](https://baike.baidu.com/item/守护进程/966835)等;
* 我们也可以控制每一条日志的输出格式
* 通过定义每一条日志信息的级别,我们能够更加细致地控制日志的生成过程
* 通过一个[配置文件](https://baike.baidu.com/item/配置文件/286550)来灵活地进行配置,而不需要修改应用的代码。

### 第一步

先导入Log4j的依赖包

```xml
       <dependency>
            <groupId>log4j</groupId>
            <artifactId>log4j</artifactId>
            <version>1.2.17</version>
        </dependency>
```

### 第二步

在resource下建立log4j.properties。内容如下(数字为行号):

```xml
#将等级为DEBUG的日志信息输出到console和file这两个目的地,console和file的定义在下面的代码
log4j.rootLogger=DEBUG,console,file

#控制台输出的相关设置
log4j.appender.console = org.apache.log4j.ConsoleAppender
log4j.appender.console.Target = System.out
log4j.appender.console.Threshold=DEBUG
log4j.appender.console.layout = org.apache.log4j.PatternLayout
log4j.appender.console.layout.ConversionPattern=【%c】-%m%n

#文件输出的相关设置
log4j.appender.file = org.apache.log4j.RollingFileAppender
log4j.appender.file.File=./log/chen.log
log4j.appender.file.MaxFileSize=10mb
log4j.appender.file.Threshold=DEBUG
log4j.appender.file.layout=org.apache.log4j.PatternLayout
log4j.appender.file.layout.ConversionPattern=【%p】【%d{yy-MM-dd}】【%c】%m%n

#日志输出级别
log4j.logger.org.mybatis=DEBUG
log4j.logger.java.sql=DEBUG
log4j.logger.java.sql.Statement=DEBUG
log4j.logger.java.sql.ResultSet=DEBUG
log4j.logger.java.sql.PreparedStatement=DEBUG​

```

### 第三步

配置log4j为日志的实现

```
<settings>
    <setting name="logImpl" value="LOG4J"/>
</settings>
```

### 第四步

使用

1,在使用的Log4j的类中,导入包import.org.apache.log4j.Logger;

2,日志对象,参数为当前的class

```java
static Logger logger = Logger.getLogger(LogDemo.class);
```

3,日志级别

```java
    logger.info();//info级别输出
    logger.dubug();//debug级别输出
    log.error();//error级别输出
```

## 7,日志

* 减少的数据处理量

```java
  select * from mybatis.user  limit #{startIndex},#{pageSize} 起始值(startIndex),总数pageSize
    select * from user where 3;#[0,n]
```

使用Mybatis实现分页,核心SQL

1,接口

```java
List<user> getUserByLimit(Map<String,Integer> map);
```

2,Mapper.xml

```xml
<select id="getUserList" resultType="com.dao.user" resultMap="Usermap">
        select * from mybatis.user where limit #{startIndex},#{pageSize}
```

## 8,使用注解开发

#### 8.1,面向接口编程

\- 大家之前都学过面向对象编程,也学习过接口,但在真正的开发中,很多时候我们会选择面向接口编程
\- **根本原因:解耦,可拓展,提高复用,分层开发中,上层不用管具体的实现,大家都遵守共同的标准,使得开发变得更容易,规范性更好**
\- 在一个面向对象的系统中,系统的各种功能是由许许多多的不同对象协作完成的。在这种情况下,各个对象内部是如何实现自己的,对系统设计人员来讲就不那么重要了;
\- 而各个对象之前的协作关系则成为系统设计的关键,小到不同类之间的通信,大到各模块之间的交互,在系统设计之初都是要着重考虑的,这也是系统设计的主要工作内容,面向接口编程就是指按照这种思想来编程。**关于接口的理解**

\- 接口从更深层次的理解,应是定义(规范,约束)与实现(名实分离)的分离。

\- 接口的本身反映了系统设计人员对系统的抽象理解。

\- 接口应有两类:

 \- 第一类是对一个个体的抽象,它可对应为一个抽象体(abstract class);
 \- 第二类是对一个个体某一方面的抽象,即形成一个抽象面(interface);

\- 一个体有可能有多个抽象面,抽象体与抽象面是有区别的

**三个面向区别**

\- 面向对象是指,我们考虑问题时,以对象为单位,考虑它的属性及方法
\- 面向过程是指,我们考虑问题时,以一个具体的流程(事务过程)为单位,考虑它的实现。
\- 接口设计与非接口设计是针对复用技术而言的,与面向对象(过程)不是一个问题,更多的体现就是对系统整体的架构​

#### 8.2,使用注解开发

1,注解直接在接口上实现

```java
@Select("select * from user")
List<user> getUsers();
```

2,需要在核心配置文件中绑定接口

```xml
<Mappers>
<mapper class="Dao.UserDao.class"
</Mappers>
```

3,测试

本质:反射机制实现

底层:动态代理模式

## 9,Lombok

使用步骤

1,在IDEA中安装Lombok插件

2,在项目中导入Lombok的jar包

3,

```java
@Getter and @Setter  掌握
@FieldNameConstants
@ToString
@EqualsAndHashCode
@AllArgsConstructor, @RequiredArgsConstructor and @NoArgsConstructor
@Log, @Log4j, @Log4j2, @Slf4j, @XSlf4j, @CommonsLog, @JBossLog, @Flogger, @CustomLog
@Data    掌握
@Builder
@SuperBuilder
@Singular
@Delegate
@Value
@Accessors
@Wither
@With
@SneakyThrows
@val
@var
experimental @var
@UtilityClass
Lombok config system
Code inspections
Refactoring actions (lombok and delombok)
```

@Data :无参构造,get,set,toString,hashcode,equals

```java
@Data
@NoArgsConstructor
public class user {
    private  int id;
    private  String name;
    private  String password;
}
```

## 10,多对一处理

* 多个学生,对应一个老师
* 对于学生而言,关联,多个学生,关联一个老师
* 对于老师而言,集合,一个老师,有很多学生

#### 测试环境搭建

1,导入lombok

2,建立实体类teacher,student

3,建立mapper接口

4,建立mapper.xml文件

5,在核心配置文件中,绑定注册的mapper接口或者文件

5,测试

```java
association – 一个复杂类型的关联;许多结果将包装成这种类型
嵌套结果映射 – 关联可以是 resultMap 元素,或是对其它结果映射的引用
collection – 一个复杂类型的集合
嵌套结果映射 – 集合可以是 resultMap 元素,或是对其它结果映射的引用
```

**按照嵌套查询**

```java
 <select id="getStudent" resultMap="StudentTeacher">
        select * from school2.student
    </select>
    <resultMap id="StudentTeacher" type="Student">
        <result property="id" column="id"/>
        <result property="name" column="name"/>
        <!--对象association 集合collect-->
        <association property="teacher" column="tid" javaType="Teacher" select="getTeacher"/>
    </resultMap>
    <select id="getTeacher" resultType="Teacher">
        select * from school2.teacher where id=#{id}
    </select>
```

**按照结果查询**

```java
<!--按结果嵌套-->
<mapper namespace="Dao.studentMapper">
    <select id="getStudent2" resultMap="StudentTeacher" >
        select s.id sid,s.name sname,t.name tname
        from student s,teacher t
        where s.tid=t.id
    </select>
    <resultMap id="StudentTeacher" type="Student">
        <result property="id" column="sid"/>
        <result property="name" column="sname"/>
        <association property="teacher" javaType="Teacher">
            <result property="name" column="tname"/>
        </association>
    </resultMap>
</mapper>
```

## 11,一对多

对于老师而言,集合,一个老师,有很多学生

1,导入lombok

2,建立实体类teacher,student

3,建立mapper接口

4,建立mapper.xml文件

5,在核心配置文件中,绑定注册的mapper接口或者文件

5,测试

```java
association – 一个复杂类型的关联;许多结果将包装成这种类型嵌套结果映射 –
关联可以是 resultMap 元素,或是对其它结果映射的引用
collection – 一个复杂类型的集合嵌套结果映射 –
集合可以是 resultMap 元素,或是对其它结果映射的引用
```

按结果查询

```java
<mapper namespace="Dao.studentMapper">
    <select id="getTeacher" resultMap="StudentTeacher" >
        select s.id sid,s.name sname,t.name tname,t.id tname
        from student s,teacher t
        where s.tid=t.id and tid=#{tid}
    </select>
    <resultMap id="StudentTeacher" type="Teacher">
        <result property="id" column="tid"/>
        <result property="name" column="tname"/>
        <!--association – 一个复杂类型的关联, collection,一个复杂的类型的集合
        javaType=""指定的属性类型
        集合中的泛型信息,我们使用ofType获取
        -->
        <collection property="student" ofType="Student>
            <result property="name" column="sname"/>
            <result property="id column="sid"/>
            <result property="tid" column="tid"/>
            </collection>
    </resultMap>
</mapper>
```

#### 小结

1,关联,association【多对一】

2,集合,collection【一对多】

3.javaType & ofType

* javaType用来指定实体类中属性的类型
* ofType用来指定映射到List或者集合中的Pojo类型,泛型中的约束类型

注意点:

* 保证SQL的可读性
* 主要一对多和多对一中,属性名和字段名的问题
* 如果问题不好排查错误,可以使用日志,建议使用log4j

## 12,动态SQL

动态SQL:动态sql就是根据不同的条件生成不同的SQL语句

利用动态SQL这一特性可以彻底摆脱这种痛苦

```xml
如果你之前用过 JSTL 或任何基于类 XML 语言的文本处理器,你对动态 SQL 元素可能会感觉似曾相识。在 MyBatis 之前的版本中,需要花时间了解大量的元素。借助功能强大的基于 OGNL 的表达式,MyBatis 3 替换了之前的大部分元素,大大精简了元素种类,现在要学习的元素种类比原来的一半还要少。
if
choose (when, otherwise)
trim (where, set)
foreach
```

搭建环境

基础工程

* 导包
* 编写配置文件
* 编写实体类
* 编写mapper接口和mapper.xml文件

#### IF

```xml
<select id="BlogIf" parameterType="Map" resultType="Blog">
        select * from school2.blog
        where 1=1
        <if test="title!=null">
            and title=#{title}
        </if>
            <if test="author">
                and author=#{author}
            </if>
    </select>
```

#### choose、when、otherwise

choose

当其中有一个成立是输出,当都成立时,优先选择第一个成立的输出,通过where标签省略的and

```xml
<select id="BlogChoose" resultType="blog">
        select * from blog
        <where>
            <choose>
                <when test="title!=null"> //当..的时候
                    title=#{title}
                </when>
                <when test="author!=null">//当..的时候
                    and author=#{author}
                </when>
                <otherwise>
                    and views=#{views}
                </otherwise>
            </choose>
        </where>
    </select>
```

#### trim、where、set

where

*where* 元素只会在子元素返回任何内容的情况下才插入 “WHERE” 子句。而且,若子句的开头为 “AND” 或 “OR”,*where* 元素也会将它们去除。

```java
   <select id="BlogIf" parameterType="Map" resultType="Blog">
        select * from school2.blog
        <where>
        <if test="title!=null">
            title=#{title}
        </if>
            <if test="author">
                and author=#{author} 这里的可以用where标签来避免and
            </if>
        </where>
    </select>
```

set

*set* 元素可以用于动态包含需要更新的列,忽略其它不更新的列

*set* 元素会动态地在行首插入 SET 关键字,并会删掉额外的逗号

```xml
update Author
    <set>
      <if test="username != null">username=#{username},</if>
      <if test="password != null">password=#{password},</if>
      <if test="email != null">email=#{email},</if>
      <if test="bio != null">bio=#{bio}</if>
    </set>
  where id=#{id}
```

trim

你也可以通过自定义 trim 元素来定制 *where* 元素的功能。比如,和 *where* 元素等价的自定义 trim 元素为:

```xml
<trim prefix="WHERE" prefixOverrides="AND |OR ">
  ...
</trim>
```

*prefixOverrides* 属性会忽略通过管道符分隔的文本序列

#### SQL片段

有的时候,我们可以将一些公共的部分抽取出来,方便复用

```xml
<sql id="title-author">
        <if test="title!=null">
            title=#{title}
        </if>
        <if test="author">
            and author=#{author}
        </if>
    </sql>
在需要的地方通过include标签引用
 <select id="BlogIf" parameterType="Map" resultType="Blog">
        select * from school2.blog
        <where>
        <include refid="title-author">//通过include标签来引用
        </include>
        </where>
    </select>

```

注意事项

* 最好基于单表来定义SQL片段
* 不要存在where标签

Foreach

```sql
select * from student where id=1 or id=2 or id=3
```

转换为java语句

```xml
select * from student
where id in
<foreach item="id" collection="List"
  open="(" separator="or" close=")">
  #{id}
</foreach>1
例如:
   <select id="getStudent" resultType="Student" parameterType="Map">
        select * from student
        <where>
            <foreach collection="ids" item="id" open="(" separator="or" close=")">
                id=#{id}
            </foreach>
        </where>
    </select>
```

动态SQL就是在拼接SQL语句,我们只要保证SQL的正确性,按照SQL的格式,去排列组合就可以了

建议

* 我们最好在Mysql中写入完整的SQL语句后,在去修改为我们的动态SQL语句

## 13,缓存

#### 13.1,一级缓存

这个简单语句的效果如下:

- 映射语句文件中的所有 select 语句的结果将会被缓存。
- 映射语句文件中的所有 insert、update 和 delete 语句会刷新缓存。
- 缓存会使用最近最少使用算法(LRU, Least Recently Used)算法来清除不需要的缓存。
- 缓存不会定时进行刷新(也就是说,没有刷新间隔)。
- 缓存会保存列表或对象(无论查询方法返回哪种)的 1024 个引用。
- 缓存会被视为读/写缓存,这意味着获取到的对象并不是共享的,可以安全地被调用者修改,而不干扰其他调用者或线程所做的潜在修改。

缓存失效

* 查询不同的东西
* 增删改,可能会改变原来的数据,所以必定会刷新缓存
* 查询不同的Mapper.xml
* 手动清理缓存

```java
sqlSession.clearCache
```

小结:一级缓存默认是开启的,只在一次的sqlSession中有效,也就是拿到连接到关闭连接这个区间段

一级缓存可以看做成一个Map

#### 13.2,二级缓存

* 二级缓存是全局缓存
* 基于namespace级别的缓存,一个命名空间,对应一个二级缓存

步骤

1,开启全局缓存

```xml
<setting name="cacheEnabled" value="true"
```

2,在要使用的二级缓存的Mapper中开启

```xml
<cache
  eviction="FIFO"
  flushInterval="60000"
  size="512"
  readOnly="true"/>
```

3,测试

1,问题:实体类一定要序列化

小结

* 只要开启了二级缓存,在同一个Mapper下就有效
* 所有的数据都会先放在一级缓存中
* 只有当会话提交,或者关闭的时候,才会提交到二级缓存中

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值