mybatis的深入理解

Mybatis的三大对象的生命周期
SqlSessionFactoryBuilder

SqlSessionFactoryBuilder一旦创建了SqlSessionFactory之后就被销毁了,最佳作用域就是方法作用域就是局部方法变量

SqlSessionFactory

SqlSessionFactory一旦被创建就应该在应用运行的时候一直存在,没有理由丢弃它,除非要连接另一个数据库才会重新创建一个新的实例,在运行期间不要重复创建多次,建议放到静态代码块中

SqlSession

每个线程都应该有自己的SqlSession实例,SqlSession的实例不是线程安全的,因此是不能被共享的,所以他的它的最佳作用域是请求作用域

Mybatis的动态生成实现类

mybatis中实际上采用了代理模式,在内存中生成Dao接口的代理类,然后创建代理类的实例

Mybatis面向接口编程的两个一致

  • 映射文件的namespace和mapper接口的全类名一致
  • 映射文件中sql语句的id和mapper接口的方法名一致
Mybatis的小技巧
#{}和${}的区别

#{}:底层是使用PreparedStatement。特点:先进行SQL语句的编译,然后给SQL语句的占位符?传值

${}:底层是使用Statement。特点是先进行SQL语句的拼接,再进行SQL语句的编译

Statement存在SQL注入的现象

如果需要SQL语句的关键字放到SQL语句当中就要使用${}

#{}是以值的形式放到SQL语句当中的

批量删除

批量删除有两种方式

  • 用or的方式

    delete from t_car
    where id=1 or id=2 or id=3

  • 用in的方式

    delete from t_car
    where id in(${ids})

模糊查询
  1. 模糊查询传参数的时候要用${}接收

  2. 模糊查询还可以用concat来拼接字符串

    select *
    from t_car
    where brand like concat('%',#{brand},'%')

Mybatis的起别名

在核心配置文件当中可以添加typeAliases标签,要写在properties后面,别名不区分大小写

namespace不能用别名机制,resultType可以用别名机制

alias可以省略,省略alias的别名就是类的简名

复制代码
<typeAliases>
<typeAliase type="指定给哪个类型起别名",alias="指定别名"></typeAliase>
</typeAliases>
指定包名的方式

将这个包下的所有类都起别名,都是类的简名

复制代码
<typeAliases>
<package name=""/>
</typeAliases>
Mapper的配置

mapper的标签属性有三个

  • resource:这种方式是从类的根路径下查找文件

  • url:是从盘符下开始查找文件

  • class:提供的是mapper接口的全限定接口名,必须带有包名,如果采用这种方式,必须保证SqlMapper.xml文件和mapper接口在同一个包下

  • 在开发中一般使用package,前提是接口和xm文件必须在同一个包下,并且名字一致

    <mappers> <package name=""/> </mappers>
插入数据时获取自动生成的主键

加上 useGeneratedKeys="true"表示可以使用获取的自增的id值

keyProperty="id"表示将获取的主键赋到对象的哪个属性上

复制代码
 <insert id="insertCarUseGenerateKeys" useGeneratedKeys="true" keyProperty="id">
        insert into t_Car values (null,#{carNum},#{brand},#{guidePrice},#{produceTime},#{carType})
    </insert>

public void insertCatUseGenerateKeysTest() throws IOException {
                SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream("org/example/mapper/mybatis-config.xml"));
                SqlSession sqlSession = sqlSessionFactory.openSession();
                //获取mapper对象
                CarMapper mapper = sqlSession.getMapper(CarMapper.class);
                Car car = new Car(null,"11111","奥迪a8",200.00,"2023-12-12","新能源");
                mapper.insertCarUseGenerateKeys(car);
                sqlSession.commit();
                sqlSession.close();
                System.out.println(car);
            }

注意:mapper的方法应该和映射文件当中的sql的id对应

Mybatis的参数处理
单个简单类型参数

parameterType属性的作用是:告诉方法的参数类型是什么类型

mybatis框架自身带有类型自动推断机制,所以大部分情况下,parameterType属性是省略不写的

Map类型单个参数

当传入的是Map参数的时候,映射文件中sql语句中的#{}写的是map中的key

复制代码
<insert id="insertStudentByMap" parameterType="Map">
        insert into t_student (id,name,age,height,birth,sex) values (null,#{姓名},#{年龄},#{身高},#{出生日期},#{性别});
    </insert>

public void testInsertStudentByMap(){
        SqlSession sqlSession = SqlSessionUtils.openSession();
        StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
        Map<String,Object> map = new HashMap<>();
        map.put("姓名","赵六");
        map.put("年龄",18);
        map.put("身高",1.82);
        map.put("出生日期",new Date());
        map.put("性别",'男');
        mapper.insertStudentByMap(map);
        sqlSession.commit();
        sqlSession.close();
    }
实体类参数

当传入的参数是POJO实体类的时候,映射文件中的sql语句的#{}是实体类的属性

复制代码
 <insert id="insertByStudent">
        insert into t_student (id,name,age,height,birth,sex) values (null,#{name},#{age},#{height},#{birth},#{sex});
    </insert>
多个参数

mybatis框架会自动创建map集合,并且map集合是以map的方式存储参数的

map.put("arg0",name)

map.put("arg1",age)

当传入的参数是多个参数是,映射文件中的sql语句的#{}是param1,param2....,param n

或者arg0,arg1,...,argn

复制代码
<select id="selectByNameSex" resultType="Student">
        select *
        from t_student
        where name=#{param1} and sex=#{param2}
    </select>
相关推荐
七夜zippoe24 分钟前
Rust `std::iter` 深度解析:`Iterator` Trait、适配器与性能
开发语言·算法·rust
靠沿30 分钟前
JavaSE知识分享——继承(下)
java·开发语言
Catfood_Eason42 分钟前
CMPP3020作业2
java·开发语言
CryptoRzz44 分钟前
印度实时股票数据源接口对接文档-IPO新股、k线数据
java·开发语言·数据库·区块链
宸津-代码粉碎机1 小时前
Java内部类内存泄露深度解析:原理、场景与根治方案(附GC引用链分析)
java·开发语言·jvm·人工智能·python
NEU-UUN1 小时前
C语言 . 第三章第二节 .递归函数
c语言·开发语言
weixin_307779131 小时前
Python编码规范之字符串规范修复程序详解
开发语言·python·代码规范
郝学胜-神的一滴1 小时前
深入理解 Python 的 __init_subclass__ 方法:自定义类行为的新方式 (Effective Python 第48条)
开发语言·python·程序人生·个人开发
東雪木1 小时前
Java基础语言进阶学习——1,JVM内存模型(堆、栈、方法区)
java·jvm·学习
毕设源码-郭学长1 小时前
【开题答辩全过程】以 常二社区线上养老院管理系统为例,包含答辩的问题和答案
java·eclipse