Mybatis Notes

Mybatis First

创建Maven项目

配置依赖

<dependencies>
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>RELEASE</version>
    </dependency>
    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
        <version>RELEASE</version>
    </dependency>
    <dependency>
        <groupId>org.mybatis</groupId>
        <artifactId>mybatis</artifactId>
        <version>RELEASE</version>
    </dependency>
    <dependency>
        <groupId>log4j</groupId>
        <artifactId>log4j</artifactId>
        <version>RELEASE</version>
    </dependency>
</dependencies>

log4j配置

src\main\resources\log4j.properties

log4j.rootLogger=DEBUG
log4j.appender.console=org.apache.log4j.ConsoleAppender
log4j.appender.console.Target=System.out
log4j.appender.console.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern = [%-5p] %m%n
log4j.logger.test=debug,console

jdbc配置文件

src\main\resources\jdbc.properties

jdbc.driver=com.mysql.cj.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/test?serverTimezone=GMT%2B8
jdbc.name=root
jdbc.password=123456

主配置文件

src\main\resources\mybatis-config.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>
    <!--导入jdbc配置文件-->
    <properties resource="jdbc.properties" />
    <!--别名-->
    <typeAliases>
        <!--两种方式-->
        <!--<typeAlias type="com.hex.model.Student" alias="xxx" />-->
        <package name="com.hex.model" />
    </typeAliases>
    <!--配置数据库环境-->
    <environments default="mysql_env">
        <environment id="mysql_env">
            <transactionManager type="JDBC"/>
            <dataSource type="POOLED">
                <property name="driver" value="${jdbc.driver}"/>
                <property name="url" value="${jdbc.url}"/>
                <property name="username" value="${jdbc.name}"/>
                <property name="password" value="${jdbc.password}"/>
            </dataSource>
        </environment>
    </environments>
    <!--导入sql映射文件-->
    <mappers>
        <mapper resource="mapper.xml"/>
    </mappers>
</configuration>

mapper映射文件

src\main\resources\mapper.xml

<?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.hex.model.Student">
    <!--values里面填的是属性(setget方法名去掉setget首字母小写)-->
    <insert id="insertStudent" parameterType="Student">
        insert into student(name,age,score) values (#{name},#{age},#{score})
    </insert>
    <delete id="deleteStudentById" parameterType="int">
        DELETE FROM student WHERE id=#{id}
    </delete>
    <delete id="deleteStudentByName" parameterType="string">
        DELETE FROM student WHERE NAME=#{name}
    </delete>
    <select id="searchStudentById" parameterType="int" resultType="Student">
        SELECT * FROM student WHERE id=#{id}
    </select>
    <select id="searchStudentByName" parameterType="string" resultType="Student">
        SELECT * FROM student WHERE NAME=#{name}
    </select>
    <select id="modifyStudent" parameterType="Student">
        UPDATE student SET NAME=#{name},age=#{age},score=#{score} where id=#{id}
    </select>
    <select id="fuzzyQueryByName" parameterType="string" resultType="Student">
        SELECT * FROM student WHERE NAME LIKE '%${value}%'
    </select>
</mapper>

实体类

com.hex.model.Student

public class Student {
    private Integer id;
    private String name;
    private int age;
    private double score;
    public Student() {}
    public Student(String name, int age, double score) {
        this.name = name;
        this.age = age;
        this.score = score;
    }
    /* 省略set、get方法 */
}

dao层

com.hex.dao.StudentDao

package com.hex.dao;
import com.hex.model.Student;
import com.hex.utils.MybatisUtils;
import org.apache.ibatis.session.SqlSession;
import org.junit.Test;
import javax.sound.midi.SoundbankResource;
import java.util.ArrayList;
import java.util.List;
public class StudentDao {
    private SqlSession sqlSession;
    public void insertStudent(Student student){
        try{
            sqlSession = MybatisUtils.sqlSessionFactory();
            sqlSession.insert("insertStudent",student);
            sqlSession.commit();
        }finally {
            sqlSession.close();
        }
    }
    public void deleteStudentById(int id){
        try{
            sqlSession = MybatisUtils.sqlSessionFactory();
            sqlSession.delete("deleteStudentById",id);
            sqlSession.commit();
        }finally {
            sqlSession.close();
        }
    }
    public void deleteStudentByName(String name){
        try{
            sqlSession = MybatisUtils.sqlSessionFactory();
            sqlSession.delete("deleteStudentByName",name);
            sqlSession.commit();
        }finally {
            sqlSession.close();
        }
    }
    public List<Student> searchStudentById(int id){
        List<Student> studentList  = new ArrayList<Student>();
        try{
            sqlSession = MybatisUtils.sqlSessionFactory();
            studentList = sqlSession.selectList("searchStudentById",id);
        }finally {
            sqlSession.close();
            return studentList;
        }
    }
    public List<Student> searchStudentByName(String name){
        List<Student> studentList  = new ArrayList<Student>();
        try{
            sqlSession = MybatisUtils.sqlSessionFactory();
            studentList = sqlSession.selectList("searchStudentByName",name);
        }finally {
            sqlSession.close();
            return studentList;
        }
    }
    public void modifyStudent(Student student){
        try{
            sqlSession = MybatisUtils.sqlSessionFactory();
            sqlSession.update("modifyStudent",student);
            sqlSession.commit();
        }finally {
            sqlSession.close();
        }
    }
    public List<Student> fuzzyQueryByName(String word){
        List<Student> studentList  = new ArrayList<Student>();
        try{
            sqlSession = MybatisUtils.sqlSessionFactory();
            studentList = sqlSession.selectList("fuzzyQueryByName",word);
        }finally {
            sqlSession.close();
            return studentList;
        }
    }
}

工具类

com.hex.utils.MybatisUtils

import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import java.io.IOException;
import java.io.InputStream;
public class MybatisUtils {
    private static SqlSessionFactory sqlSessionFactory;
    /**
     * 读取主配置文件,创建SqlSession,如果SqlSession已经存在则直接返回
     * @return
     */
    public static SqlSession sqlSessionFactory() {
        try {
            InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
            if (sqlSessionFactory == null) {
                sqlSessionFactory = new SqlSessionFactoryBuilder().build(is);
            }
            return sqlSessionFactory.openSession();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return null;
    }
}

测试类

com.hex.test.MyTest

import com.hex.dao.StudentDao;
import com.hex.model.Student;
import org.junit.Test;
import java.io.IOException;
public class MyTest {
    private StudentDao dao = new StudentDao();
    @Test
    public void test() throws IOException {
        Student student = new Student("hex",23,98.5);
        dao.insertStudent(student);
    }
    /*省略其他测试方法*/
}

单表的CURD操作

sql语句拼接

<select id="fuzzyQueryByName" parameterType="string" resultType="Student">
    SELECT * FROM student WHERE NAME like '%' #{word} '%'
</select>
<select id="fuzzyQueryByName" parameterType="string" resultType="Student">
    SELECT * FROM student WHERE NAME like concat('%',#{word},'%')
</select>
<!--第3中方法难以避免sql注入问题-->
<select id="fuzzyQueryByName" parameterType="string" resultType="Student">
    SELECT * FROM student WHERE NAME LIKE '%${value}%'
</select>

resultMap

<resultMap id="studentMapper" type="Student">
    <id column="tid" property="id" />
    <result column="tname" property="name" />
</resultMap>
<select id="searchStudentById" resultMap="studentMapper">
    SELECT tid,tname,tage,score FROM student WHERE id=#{id}
</select>

数据库字段名和实体类属性名不一致:将数据库的tid、tname、tage、score封装到Student(id,name,age,score)对象中

mapper动态代理

一般流程
dao对象调用dao层接口的实现类 -> sqlSession.insert(映射器id,映射器需要的参数) -> mapper中对应id的sql语句
mapper的动态代理
new sqlSession -> dao = sqlSession.getMapper(dao层接口.class) -> dao调用接口

通常dao层接口的实现类里面定义了sqlSession去指定mapper.xml映射文件中的哪一个映射方法。然后mapper的动态代理要设置mapper的命名空间,sqlSession.getMapper(dao层接口.class)就相当于指定要找该指定命名空间的映射器。可是映射文件里面定义了那么多select、insert等查询语句,怎么调用我要的呢?这个时候就可以直接用dao对象调用接口,因为接口名和sql语句名是相同的,就可以定位到相应的sql语句。看上去好像是dao直接执行sql语句一样。

两个关键:mapper的命名空间是接口类的全限定类名。接口的方法名和mapper中定义的查询语句的id要保持一致。

@Test
public void mapperDynamicProxy(){
        SqlSession sqlSession = MybatisUtils.sqlSessionFactory();
        StudentDao dao = sqlSession.getMapper(StudentDao.class);
        Student student = dao.selectStudentById(1);
        System.out.println(student.getName());
}

多条件查询

  • 把查询条件封装成实体类的对象
  • 把查询条件封装成Map
  • 索引
#{}中可以放什么内容:
1.参数对象的属性
2.随意内容,此时的#{}是个占位符
3.参数为Map时的key
4.参数为Map时key所对应的value为对象,则可将该对象的属性放入
5.参数的索引号

动态SQL

原符号 < <= > >= & ' "
替换符号 &lt; &lt;= &gt; &gt;= &amp; &apos; &quot;
  • if

    <select id="selectStudentByCondition" resultType="Student">
      select * form student
      <where>
          <if test="name != null and name != ''">
              and name like '%' #{name} '%'
          </if>
          <if test="age &gt; 0">
              and age > #{age}
          </if>
      </where>
    </select>
  • choose, when, otherwise

    <select id="selectStudentByCondition" resultType="Student">
        select * form student
        <where>
            <choose>
                <when test="name != null and name != ''">
                    and name like '%' #{name} '%'
                </when>
                <when test="age &gt; 0">
                    and age > #{age}
                </when>
                <otherwise>
                    1 = 2
                </otherwise>
            </choose>
        </where>
    </select>
  • trim, where, set

  • foreach

    <!--遍历数组-->
    <select id="selectStudentByCondition" resultType="Student">
        select * form student
        <if test="array.length > 0">
            where id in
            <foreach collection="array" item="myid" open="(" close=")" separator=",">
                #{myid}
            </foreach>
        </if>
    </select>
    <!--遍历list-->
    <select id="selectStudentByCondition" resultType="Student">
        select * form student
        <if test="list.size > 0">
            where id in
            <foreach collection="list" item="myid" open="(" close=")" separator=",">
                #{myid}
            </foreach>
        </if>
    </select>
  • bind

sql片段

<sql id="selectColumns">id,name,age,score</sql>
<select id="selectStudentByCondition" resultType="Student">
    select <include refid="selectColumns" /> form student
</select>

关联关系查询

一对多

创建数据表和实体类

country minister
结构 cid、cname mid、mname、countryid
关系

mapper映射文件

<mapper namespace="com.one2many.dao.CountryDao">
    <resultMap id="countryMapper" type="Country">
        <id column="cid" property="cid" />
        <result column="cname" property="cname"/>
        <collection property="ministers" ofType="Minister">
            <id column="mid" property="mid"/>
            <result column="mname" property="mname" />
        </collection>
    </resultMap>
    <select id="selectCountryById" resultMap="countryMapper" parameterType="int">
        SELECT cid,cname,mid,mname FROM country,minister WHERE cid = countryid and cid = #{id}
    </select>
</mapper>

测试类

@Test
public void test01(){
    SqlSession sqlSession = MybatisUtils.sqlSessionFactory();
    CountryDao dao = sqlSession.getMapper(CountryDao.class);
    Country country = dao.selectCountryById(1);
    System.out.println(country.toString());
}

mapper还可以这样写,一个集合的值是从另外一个select语句的结果得到的

<mapper namespace="com.one2many.dao.CountryDao">
    <select id="selectMinisterByCountry" resultType="Minister">
        select mid,mname from minister where countryid = #{xxx}
    </select>
    <resultMap id="countryMapper" type="Country">
        <id column="cid" property="cid" />
        <result column="cname" property="cname"/>
        <collection property="ministers"
                    ofType="Minister"
                    select="selectMinisterByCountry"
                    column="cid"/>
    </resultMap>
    <select id="selectCountryById" resultMap="countryMapper" parameterType="int">
        SELECT cid,cname,mid,mname FROM country,minister WHERE cid = countryid and cid = #{id}
    </select>
</mapper>

多对一

创建数据表和实体类

country minister
结构 cid、cname mid、mname、country
关系

mapper映射文件

<mapper namespace="com.many2one.dao.MinisterDao">
    <resultMap id="ministerMapper" type="Minister">
        <id column="mid" property="mid" />
        <result column="mname" property="mname" />
        <association property="country" javaType="Country">
            <!--property映射成javaType-->
            <id column="cid" property="cid" />
            <result column="cname" property="cname" />
        </association>
    </resultMap>
    <select id="selectMinisterById" resultMap="ministerMapper">
        SELECT MID,mname,cid,cname FROM minister,country
        WHERE countryid = cid AND MID = #{xxx}
    </select>
</mapper>

测试类

@Test
public void test01(){
    SqlSession sqlSession = MybatisUtils.sqlSessionFactory();
    MinisterDao dao = sqlSession.getMapper(MinisterDao.class);
    Minister minister = dao.selectMinisterById(2);
    System.out.println(minister.toString());
}

mapper还可以这样写,一个关联的值是从另外一个select语句的结果得到的

<mapper namespace="com.many2one.dao.MinisterDao">
    <select id="selectCountryById" resultType="Country">
        select cid,cname FROM country where cid=#{xxx}
    </select>
    <resultMap id="ministerMapper" type="Minister">
        <id column="mid" property="mid" />
        <result column="mname" property="mname" />
        <association property="country"
                     javaType="Country"
                     select="selectCountryById" column="countryid">
        </association>
    </resultMap>
    <select id="selectMinisterById" resultMap="ministerMapper">
        SELECT MID,mname,countryid FROM minister WHERE MID = #{xxx}
    </select>
</mapper>

自关联

NewsLabel模型

public class NewsLabel {
    private int id;
    private String name;
    private Set<NewsLabel> children;
    /*省略set、get、toString方法*/
}
<mapper namespace="com.hex.dao.NewsLabelDao">
    <resultMap id="newslabelMapper" type="NewsLabel">
        <id column="id" property="id" />
        <result column="name" property="name"/>
        <collection property="children"
                    ofType="NewsLabel"
                    select="selectChildrenByParent"
                    column="id" />
    </resultMap>
    <select id="selectChildrenByParent" resultMap="newslabelMapper">
        select id,name FROM newslabel where pid=#{xxx}
    </select>
</mapper>
@Test
public void test01(){
    SqlSession sqlSession = MybatisUtils.sqlSessionFactory();
    NewsLabelDao dao = sqlSession.getMapper(NewsLabelDao.class);
    List<NewsLabel> list = dao.selectChildrenByParent(2);
    for (NewsLabel li:list) {
        System.out.println(li.toString());
    }
}

多对多

延迟加载

<settings>
    <setting name="lazyLoadingEnable" value="false" />
    <setting name="aggressiveLazyLoading" value="false" />
</settings>

查询缓存

一级缓存、二级缓存

ehcache

  • 导包

    <dependency>
        <groupId>net.sf.ehcache</groupId>
        <artifactId>ehcache</artifactId>
        <version>2.8.3</version>
    </dependency>
    <dependency>
        <groupId>org.mybatis</groupId>
        <artifactId>mybatis-ehcache</artifactId>
        <version>1.0.0</version>
    </dependency>
  • 引入ehcache类
    xml <mapper> <cache type="org.mybatis.caches.ehcache.EhcacheCache" /> </mapper>

  • 导入配置文件

    把ehcache-core里面的配置文件导入项目

Mybatis注解式开发(dao层接口)

  • @Insert(value="")
  • @Delete(value="")
  • @Update(value="")
  • @Select(value="")
<mapper>
    <!--映射文件所在的包-->
    <package name="com.hex.dao" />
</mapper>

SSM学习(一)Mybatis的更多相关文章

  1. SSM(Spring +SpringMVC + Mybatis)框架搭建

    SSM(Spring +SpringMVC + Mybatis)框架的搭建 最近通过学习别人博客发表的SSM搭建Demo,尝试去搭建一个简单的SSMDemo---实现的功能是对用户增删改查的操作 参考 ...

  2. SSM(Spring+SpringMVC+Mybatis)框架环境搭建(整合步骤)(一)

    1. 前言 最近在写毕设过程中,重新梳理了一遍SSM框架,特此记录一下. 附上源码:https://gitee.com/niceyoo/jeenotes-ssm 2. 概述 在写代码之前我们先了解一下 ...

  3. SSM Spring+SpringMVC+mybatis+maven+mysql环境搭建

    SSM Spring+SpringMVC+mybatis+maven环境搭建 1.首先右键点击项目区空白处,选择new->other..在弹出框中输入maven,选择Maven Project. ...

  4. SSM(Spring + Springmvc + Mybatis)框架面试题

    JAVA SSM框架基础面试题https://blog.csdn.net/qq_39031310/article/details/83050192 SSM(Spring + Springmvc + M ...

  5. SSM框架-初学Mybatis框架

    SSM(Spring+SpringMVC+Mybatis)是目前项目开发比较流行的一套组合框架,而Mybatis是负责数据库操作的那部分框架,具体 我也说不上来 传统的JDBC操作比较冗长而繁琐,而用 ...

  6. MyBatis学习总结-MyBatis快速入门的系列教程

    MyBatis学习总结-MyBatis快速入门的系列教程 [MyBatis]MyBatis 使用教程 [MyBatis]MyBatis XML配置 [MyBatis]MyBatis XML映射文件 [ ...

  7. SSM Spring +SpringMVC+Mybatis 整合配置 及pom.xml

    SSM Spring +SpringMVC+Mybatis 配置 及pom.xml SSM框架(spring+springMVC+Mybatis) pom.xml文件 maven下的ssm整合配置步骤

  8. (转)MyBatis框架的学习(七)——MyBatis逆向工程自动生成代码

    http://blog.csdn.net/yerenyuan_pku/article/details/71909325 什么是逆向工程 MyBatis的一个主要的特点就是需要程序员自己编写sql,那么 ...

  9. (转)MyBatis框架的学习(六)——MyBatis整合Spring

    http://blog.csdn.net/yerenyuan_pku/article/details/71904315 本文将手把手教你如何使用MyBatis整合Spring,这儿,我本人使用的MyB ...

  10. (转)MyBatis框架的学习(二)——MyBatis架构与入门

    http://blog.csdn.net/yerenyuan_pku/article/details/71699515 MyBatis框架的架构 MyBatis框架的架构如下图: 下面作简要概述: S ...

随机推荐

  1. Holt-Winters

    https://blog.csdn.net/u010665216/article/details/78051192 mark

  2. halcon 图片加载和设置XY轴滑动块的先后顺序

    //必须先加载图片,然后执行 hWndControl.setGUICompRangeX( new int[]{ XTrackBar.Minimum, XTrackBar.Maximum}, XTrac ...

  3. COMP9334 Project

    COMP9334 Project, Term 1, 2019:Fog/cloud ComputingVersion 1.0Due Date: 11:00pm Friday 26 April 2019. ...

  4. python 包下载地址

    https://www.lfd.uci.edu/~gohlke/pythonlibs/

  5. python基础之 装饰器,内置函数

    1.闭包回顾 在学习装饰器之前,可以先复习一下什么是闭包? 在嵌套函数内部的函数可以使用外部变量(非全局变量)叫做闭包! def wrapper(): money =10 def inner(num) ...

  6. seq2seq笔记

    max_encoder_seq_length = max([len(txt) for txt in input_texts]) encoder_input_data = np.zeros(     ( ...

  7. A-the Beatles

    传送门: 题意:题目给出n,k分别代表在这个环中饭店的个数和两个饭店相离的距离.然后再给出一组a,b分别代表在某一点s里最近饭店的距离和在这个s点走一步之后到达的点离最近饭店的距离. 然后问这个人再次 ...

  8. zookeeper的使用demo(c#/java)

    Zookeeper 作为一个分布式的服务框架,主要用来解决分布式集群中应用系统的一致性问题,它能提供基于类似于文件系统的目录节点树方式的数据存储,但是 Zookeeper 并不是用来专门存储数据的,它 ...

  9. 感觉不错的随笔 关于C、C++的

    [effective C++的网页版] http://www.kuqin.com/effectivec2e/ 内存四区模型 https://www.cnblogs.com/crazyzhang/p/5 ...

  10. Rsync数据同步工具

                                        Rsync数据同步工具 什么是Rsync? Rsync是一款开源的.快速的.多功能的,可以实现全量及增量的本地或原程数据同步备份 ...