Mybatis–动态SQL
以前:我们的SQL语句都是直接写死在配置文件中,一但写好了就固定了。
概念:在代码的执行过程中,根据代码的执行情况动态拼接成不同的SQL语句
where:条件标签
如果有动态条件,则使用该标签代替where 关键字。
根据拼接SQL字符串的情况,自动去掉多余的and, or , where关键字
if:条件判断标签
resultType : 返回值类型,定义了别名
parameterType : 输入类型 ,也是定义了别名,注意,在使用这个对象时,对象的属性一定要和表中字段名相对应,符合ORM映射关系。符合之后,我们才能在if标签的test属性中直接 使用 contact对象的属性名进行判断。
username 和 sex都是contact对象的属性。
if标签写在where标签内部
- if:判断用户名称不为空,且不为空字符串,则作为查询条件
- if:判断用户性别不为空,且不为空字符串,则作为查询条件
代码示例
<select id="findAllContact" resultType="contact" parameterType="contact">
select * from contact
<where>
<if test="username!=null and username !=''">
username like "%"#{username}"%"
</if>
<if test="sex!=null and sex!=''">
and sex = #{sex}
</if>
</where>
</select>
set标签
用于update标签
- 相当于update中set关键字
- 同时修改多个字段,字段之间使用逗号分隔。可能拼接的时候会出现多余的逗号,set标签可以去掉多余的逗号
应用场景:
更新用户信息的时候,有些实体类的属性为空,为空就不更新,不为空才更新这个字段。
<update id="updateUser">
update user
<set>
<if test="username!=null and username!=''">
username=#{username},
</if>
<if test="sex!=null and sex!=''">
sex=#{sex}
</if>
</set>
where id=#{id}
</update>
foreach标签
作用:遍历集合或数组,将集合或数组中每个元素都做为SQL语句的一部分进行拼接,拼接多次
一条insert语句插入多条记录的MySQL语句如何编写?
所以需要进行遍历
参数类型时集合
foreach标签的属性:
collection 指定是集合还是数组,如果是集合使用list,如果是数组使用array
item 表示集合中每个元素的变量名,这里是一个实体类对象
separator 每次拼接完以后,添加1个符号分隔
<!--使用foreach标签遍历一个集合,生成SQL语句-->
<insert id="addUsers">
insert into user (username, birthday, sex, address) values
<foreach collection="list" item="user" separator=",">
(#{user.username},#{user.birthday},#{user.sex},#{user.address})
</foreach>
</insert>
参数类型是数组
collection: 表示要遍历的集合类型,数组使用array
item: 表示其中每个元素的变量名
open:表示循环开始前添加一个符号,只会执行1次
separator 每次拼接完以后,添加1个符号分隔,执行多次
close:表示循环结果以后添加一个符号,只会执行1次
<!--批量删除用户 -->
<delete id="deleteUsers">
delete from user where id in
<foreach collection="array" open="(" item="id" separator="," close=")">
#{id}
</foreach>
</delete>
sql和include标签
作用:
- sql标签:定义一个SQL语句代码段,并且使用id设置唯一的标识
- include标签:包含上面定义的语句代码段
这2个标签是要配合使用的,对定义的SQL代码段进行重用
<!--
1. sql标签定义查询条件代码块
Map = {minDate='1999-9-9', maxDate='2000-1-1'}
在XML中<或>符号是有特殊含义的,通常不能直接写在XML中
有两种解决方案:
1) 使用转义:< >
2) 使用CDATA包裹起来的内容表示纯文本内容,XML解析器不对它进行解析
-->
<sql id="sqlCondition">
<where>
<if test="minDate!=null and minDate!=''">
birthday >= #{minDate}
</if>
<if test="maxDate!=null and maxDate!=''">
<![CDATA[
and birthday <= #{maxDate}
]]>
</if>
</where>
</sql>
<!--
include标签的作用:引用使用sql标签定义的代码段
属性:refid对应上面sql的id值
-->
<select id="findUsersByMap" resultType="user">
select * from user <include refid="sqlCondition"/>
</select>
<select id="findCountByMap" resultType="int">
select count(*) from user <include refid="sqlCondition"/>
</select>
Mybatis–分页插件
分页插件介绍
在企业级开发中,分页也是一种常见的技术。而目前使用的MyBatis 是不带分页功能的,如果想实现分页的 功能,需要我们手动编写LIMIT 语句。但是不同的数据库实现分页的SQL 语句也是不同的,所以手写分页 成本较高。这个时候就可以借助分页插件来帮助我们实现分页功能。
PageHelper:第三方分页助手。将复杂的分页操作进行封装,从而让分页功能变得非常简单。
分页插件实现步骤
-
导入 jar 包。
-
在核心配置文件中集成分页助手插件。
<!-- 注意标签书写的先后顺序 写在environments标签之前--> <plugins> <plugin interceptor="com.github.pagehelper.PageInterceptor"></plugin> </plugins>
-
在测试类中使用分页助手相关API 实现分页功能。
public class Test01 { @Test public void selectPaging() throws Exception{ //1.加载核心配置文件 InputStream is = Resources.getResourceAsStream("MyBatisConfig.xml"); //2.获取SqlSession工厂对象 SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(is); //3.通过工厂对象获取SqlSession对象 SqlSession sqlSession = sqlSessionFactory.openSession(true); //4.获取StudentMapper接口的实现类对象 StudentMapper mapper = sqlSession.getMapper(StudentMapper.class); //通过分页助手来实现分页功能, 参数1:当前页数 参数2:每页显示条数 //在查询语句前执行即可,这是插件提供的静态方法。 // 第一页:显示3条数据 //PageHelper.startPage(1,3); // 第二页:显示3条数据 //PageHelper.startPage(2,3); // 第三页:显示3条数据 PageHelper.startPage(3,3); //------------------------------------------------------ //获取总的数据,因为上面执行了插件的方法,所以获得的数据会受插件的影响 // 还解决了不同数据库 分页语句不同的问题 //5.调用实现类的方法,接收结果 List<Student> list = mapper.selectAll(); //6.处理结果 for (Student student : list) { System.out.println(student); } //获取分页相关参数 PageInfo<Student> info = new PageInfo<>(list); System.out.println("总条数:" + info.getTotal()); System.out.println("总页数:" + info.getPages()); System.out.println("当前页:" + info.getPageNum()); System.out.println("每页显示条数:" + info.getPageSize()); System.out.println("上一页:" + info.getPrePage()); System.out.println("下一页:" + info.getNextPage()); System.out.println("是否是第一页:" + info.isIsFirstPage()); System.out.println("是否是最后一页:" + info.isIsLastPage()); //7.释放资源 sqlSession.close(); is.close(); } }
分页插件相关参数