在实际开发中,查询条件千变万化,有时是各个条件单个进行查询,也有可能是其中两个、三个乃至更多条件组合起来进行查询,怎样用简洁的代码实现这个功能呢,刚开始想到的是在一种条件写一个方法,对应一条sql,这种方法固然能实现功能,但特别繁琐,造成代码冗余。后来想到mybatis的mapper.xml文件中用if test的条件判断,当即决定,使用动态sql进行查询。具体如下:
mapper.xml:
<!-- 根据灵活条件查看用户绑定设备详情 -->
<select id="queryUserDevices" parameterType="com.*.bean.Device" resultMap="deviceResultMap" flushCache="true">
select
<include refid="Base_Column_List" />
from pc_device
<where>
<if test="deviceType!=null and deviceType!=' ' ">
device_type like concat ('%',#{deviceType},'%')
</if>
<if test="patId!=null and patId!=0">
and pat_id= #{patId}
</if>
<if test="machineId!=null and machineId!=0">
and machine_id=#{machineId}
</if>
</where>
</select>
注意:1、条件要用where标签拼接,不要用choose when otherwise或者在外面写 where 1=1,下面用<if test=" ">拼接,不然如果直接写where后面如果第一个没有,那么后面的就有and 语句不对。用where标签可以解决,过滤掉and,要记住:第一个条件不要and,后面不管有多少个条件,都用and拼接起来。
2、if test里面的是类的属性,不是字段名,if test外面才是字段名,如果在if test里放字段名,将会导致 属性no getter的异常,如果字段的类型是Integer,就是这样判断:patId!=null and patId!=0,如果是string,则这样判断deviceType!=null and deviceType!=' ',具体看各个类型没有赋值时的默认值来定。如果是模糊查询,应用like #{ } 至于多个条件,我是把它们封装成一个对象进行查询。这样,如果if条件成立,将会把对应条件拼接到where子句后面,如果if条件不成立,where 子句将会忽略这个条件。我们可以在mybatis-config.xml的配置文件中的
<settings> 节点增加这个设置:<setting name="logImpl" value="STDOUT_LOGGING" />,这样在执行到相应操作时,后台就会把sql语句打印出来,方便我们查询sql语句写得是否正确了。