时间:2021-07-01 10:21:17 帮助过:8人阅读
执行测试方法,查看控制台日志:
DEBUG [main] - ==> Preparing: SELECT id, username, birthday, sex, address FROM `user` WHERE sex = ? AND username LIKE ‘%张%‘ DEBUG [main] - ==> Parameters: 1(String) DEBUG [main] - <== Total: 4 User [id=10, username=张三, sex=1, birthday=Thu Jul 10 00:00:00 CST 2014, address=北京市] User [id=16, username=张小明, sex=1, birthday=null, address=河南郑州] User [id=24, username=张三丰, sex=1, birthday=null, address=河南郑州] User [id=27, username=张飞, sex=1, birthday=Mon Dec 25 00:00:00 CST 2017, address=蜀国]
假设现在不需要性别这个条件了,注释掉 user.setSex("1"),测试结果如下:
DEBUG [main] - ==> Preparing: SELECT id, username, birthday, sex, address FROM `user` WHERE sex = ? AND username LIKE ‘%张%‘ DEBUG [main] - ==> Parameters: null DEBUG [main] - <== Total: 0
这显然不合理。
按照之前所学的,要解决这个问题,需要编写多个sql,查询条件越多,需要编写的sql就更多了,显然这样是不靠谱的。
解决方案,使用动态sql的if标签。
(4)使用if标签
<!-- 根据条件查询用户 -->
<select id="queryUserByWhere" parameterType="user" resultType="user">
SELECT id, username, birthday, sex, address FROM `user`
WHERE 1=1
<if test="sex != null and sex !=‘‘">
AND sex = #{sex}
</if>
<if test="username != null and username != ‘‘">
AND username LIKE ‘%${username}%‘
</if>
</select>
注意字符串类型的数据需要要做不等于空字符串校验。另外,为什么where后加上 1=1,因为如果不加这个,当代码中注释掉 user.setSex("1") 时,打印出来的sql语句就是 “where and username like...”,显然语法错误。
还是注释掉 user.setSex("1"),再次执行测试方法,查看控制台日志:
DEBUG [main] - ==> Preparing: SELECT id, username, birthday, sex, address FROM `user` WHERE 1=1 AND username LIKE ‘%张%‘ DEBUG [main] - ==> Parameters: DEBUG [main] - <== Total: 4 User [id=10, username=张三, sex=1, birthday=Thu Jul 10 00:00:00 CST 2014, address=北京市] User [id=16, username=张小明, sex=1, birthday=null, address=河南郑州] User [id=24, username=张三丰, sex=1, birthday=null, address=河南郑州] User [id=27, username=张飞, sex=1, birthday=Mon Dec 25 00:00:00 CST 2017, address=蜀国]
2. where标签
上面的sql还有where 1=1 这样的语句,很麻烦。可以使用where标签进行改造。
<!-- 根据条件查询用户 -->
<select id="queryUserByWhere" parameterType="user" resultType="user">
SELECT id, username, birthday, sex, address FROM `user`
<where>
<if test="sex != null and sex !=‘‘">
AND sex = #{sex}
</if>
<if test="username != null and username != ‘‘">
AND username LIKE ‘%${username}%‘
</if>
</where>
</select>
3. sql片段
Sql中可将重复的sql提取出来,使用时用include引用即可,最终达到sql重用的目的。
把上面例子中的id, username, birthday, sex, address提取出来,作为sql片段,如下:
<!-- sql片段 -->
<select id="queryUserBySql" parameterType="user" resultType="user">
<!-- SELECT id, username, birthday, sex, address FROM `user` -->
<!-- 使用include标签加载sql片段;refid是sql片段id -->
SELECT <include refid="userFields" /> FROM `user`
<!-- where标签可以自动添加where关键字,同时处理sql语句中第一个and关键字 -->
<where>
<if test="sex != null">
AND sex = #{sex}
</if>
<if test="username != null and username != ‘‘">
AND username LIKE
‘%${username}%‘
</if>
</where>
</select>
<!-- 声明sql片段 -->
<sql id="userFields">
id, username, birthday, sex, address
</sql>
4. foreach标签
向sql传递数组或List,mybatis使用foreach解析,如下:
根据多个id查询用户信息,查询sql:
SELECT * FROM user WHERE id IN (1,10,24)
(1)UserMapper.java添加方法(存在三种形式)
//根据多个id查询用户信息
public List<User> queryUserByIds(Integer[] ids);
public List<User> queryUserByIds(List<Integer> ids);
public List<User> queryUserByIds(QueryVo vo);
先来个 public List<User> queryUserByIds(QueryVo vo); 改造QueryVo.java
如下图在pojo中定义list属性ids存储多个用户id,并添加getter/setter方法。
public class QueryVo implements Serializable { //序列化,对象转成二进制进行传输。
private static final long serialVersionUID = 1L;
private User user;
List<Integer> idsList;
Integer[] ids;
public List<Integer> getIdsList() {
return idsList;
}
public void setIdsList(List<Integer> idsList) {
this.idsList = idsList;
}
public Integer[] getIds() {
return ids;
}
public void setIds(Integer[] ids) {
this.ids = ids;
}
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
}
(2)Mapper.xml文件
UserMapper.xml添加sql,如下:
<!--根据多个id查询用户信息 (1,2,3)-->
<select id="queryUserByIds" parameterType="QueryVo" resultType="user">
SELECT * FROM `user`
<where>
id IN
<!-- foreach标签,进行遍历 -->
<!-- collection:遍历的集合,这里是QueryVo的ids属性 -->
<!-- item:遍历的项目,可以随便写,,但是和后面的#{}里面要一致 -->
<!-- open:在前面添加的sql片段 -->
<!-- close:在结尾处添加的sql片段 -->
<!-- separator:指定遍历的元素之间使用的分隔符 -->
<foreach collection="idsList" item="item" separator="," open="(" close=")">
#{item}
</foreach>
</where>
</select>
或者将 id IN 放进 open 中,如下:
<foreach collection="idslist" item="item" separator="," open="id IN (" close=")">
(4)测试方法
//根据多个ID查询用户
@Test
public void testID() throws Exception {
//加载核心配置文件
String resource = "sqlMapConfig.xml";
InputStream in = Resources.getResourceAsStream(resource);
//创建SqlSessionFactory
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(in);
// 获取sqlSession,和spring整合后由spring管理
SqlSession sqlSession = sqlSessionFactory.openSession();
//SqlSEssion会帮我生成一个实现类 (需要我们给接口,它帮我们生成实现类,返回给我们的还是接口)
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
QueryVo vo = new QueryVo();
List<Integer> ids = new ArrayList<>();
ids.add(10);
ids.add(16);
ids.add(22);
vo.setIdsList(ids);
List<User> users = userMapper.queryUserByIds(vo);
for (User u: users) {
System.out.println(u);
}
// 和spring整合后由spring管理
sqlSession.close();
}
执行测试方法,查看控制台日志:
DEBUG [main] - ==> Preparing: SELECT * FROM `user` WHERE id IN ( ? , ? , ? ) DEBUG [main] - ==> Parameters: 10(Integer), 16(Integer), 22(Integer) DEBUG [main] - <== Total: 3 User [id=10, username=张三, sex=1, birthday=Thu Jul 10 00:00:00 CST 2014, address=北京市] User [id=16, username=张小明, sex=1, birthday=null, address=河南郑州] User [id=22, username=陈小明, sex=1, birthday=null, address=河南郑州]
在实现一个 public List<User> queryUserByIds(Integer[] ids);
修改UserMapper.xml,如下:
<!--根据多个id查询用户信息 (1,2,3)-->
<select id="queryUserByIds" parameterType="QueryVo" resultType="user">
SELECT * FROM `user`
<where>
id IN
<foreach collection="array" item="item" separator="," open="(" close=")">
#{item}
</foreach>
</where>
</select>
【注意】:当传递的参数是数组时,collection的值一定要是array,否则会报错。同样地,当传递的参数是list的时候,collection的值一定要是list。而当传入的参数是个对象(比如QueryVo)时,直接使用对象里面的属性名(比如上例中是idsList)。
测试方法:
//根据多个ID查询用户
@Test
public void testID2() throws Exception {
//加载核心配置文件
String resource = "sqlMapConfig.xml";
InputStream in = Resources.getResourceAsStream(resource);
//创建SqlSessionFactory
SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(in);
// 获取sqlSession,和spring整合后由spring管理
SqlSession sqlSession = sqlSessionFactory.openSession();
//SqlSEssion会帮我生成一个实现类 (需要我们给接口,它帮我们生成实现类,返回给我们的还是接口)
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
Integer[] ids = new Integer[3];
ids[0] = 16;
ids[1] = 22;
ids[2] = 10;
List<User> users = userMapper.queryUserByIds(ids);
for (User u: users) {
System.out.println(u);
}
// 和spring整合后由spring管理
sqlSession.close();
}
执行测试方法,查看日志:
DEBUG [main] - ==> Preparing: SELECT * FROM `user` WHERE id IN ( ? , ? , ? ) DEBUG [main] - ==> Parameters: 16(Integer), 22(Integer), 10(Integer) DEBUG [main] - <== Total: 3 User [id=10, username=张三, sex=1, birthday=Thu Jul 10 00:00:00 CST 2014, address=北京市] User [id=16, username=张小明, sex=1, birthday=null, address=河南郑州] User [id=22, username=陈小明, sex=1, birthday=null, address=河南郑州]
【注意】:拿到需求,可以先定义接口,然后在写sql。这样比较有头绪。
Mybatis动态sql
标签:app 管理 his user tor ids htm 方法 int