在MyBatis中可能会有一些特殊的SQL需要去执行,一般就是模糊查询、批量删除、动态设置表名、添加功能获取自增的主键这几种,现在分别来进行说明。
为了方便演示 ,定义了访问的接口
public interface SQLMapper {
/**
* 根据用户名模糊查询用户信息
*/
List<User> getUserByLike(@Param("username") String username);
/**
* 批量删除
*/
int deleteMore(String ids);
/**
* 查询指定表中的数据
*/
List<User> getUserByTableName(String tableName);
/**
* 添加用户
*/
void insetUser(User user);
}
模糊查询非常的有用,对于一些访问量不是很大的搜索都是直接使用模糊查询的方式来做的。
SQLMapper类:
public interface SQLMapper {
/**
* 根据用户名模糊查询用户信息
*/
List<User> getUserByLike(@Param("username") String username);
}
对于SQLXml的编写;
<select id="getUserByLike" resultType="User">
select * from t_user where username like "%"#{username}"%"
select>
需要注意的是Mybatis
对JDBC
进行了进一步封装,使得我们可以更加便捷的使用Java操作数据库。Mybatis获取参数值有两种方式:#{}
和${}
在大部分情况下,#{}
和${}
都能相互替代,使用两者之一即可,更加推荐使用#{}
,因为可以防止SQL
注入问题,但是由于#{}
和${}
本质上的不同,部分SQL
语句使用#{}
和${}
需要格外注意
#{}
和${}
本质区别
#{}
本质上是占位符赋值,为字符串类型或日期类型的字段进行赋值时,可以自动添加单引号${}
本质上是字符串拼接,为字符串类型或日期类型的字段进行赋值时,需要手动加单引号这个场景下,使用#{}
和${}
都能达到目的,但是用法稍有不同
如果是直接使用
<select id="selectLike" resultType="pojo.User">
select * from user where user_name like '%#{username}%'
select>
<select id="selectLike" resultType="pojo.User">
select * from user where user_name like '%${username}%'
select>
成功执行
如果非要使用#{},也不是没有解决办法
使用""
拼接
<select id="selectLike" resultType="pojo.User">
select * from user where user_name like "%"#{username}"%"
select>
<select id="selectLike" resultType="pojo.User">
select * from user where user_name like concat('%',#{likeString},'%')
select>
在某些场景下,我们需要来回操作各种表,但SQL
语句功能一致,这时我们可以使用动态表名,即传参为表名类型,这时就要从#{}
和${}
中进行选择了
Mapper接口
List<User> selectAllFromTable(@Param("tableName") String tableName);
如果是直接使用#{}
的方式
<select id="selectAllFromTable" resultType="pojo.User">
select * from #{tableName}
select>
结果报错,原因在于#{}
为占位符赋值,传参为String
的话就会自动补上单引号''
,而表名不允许添加单引号,所以导致出错。
直接使用${}
的方式
<select id="selectAllFromTable" resultType="pojo.User">
select * from ${tableName}
select>
需要在xml中配置 useGeneratedKeys
, keyProperty
两个属性的值.
<insert id="insetUser" useGeneratedKeys="true" keyProperty="id">
insert into t_user values(null, #{username}, #{password},#{age},#{gender},#{email})
insert>
有些场景,需要我们根据id数组批量删除记录,这个时候也有一些坑
由于id
数组的长度是不确定的,所以我们不能确定参数的个数,但是我们可以使用in
关键字,这个时候我们将id数组转为字符串进行传参就好了。
[1,2,3] => 1,2,3
Mapper接口
Integer deleteByIds(String Ids);
使用#{}
<delete id="deleteByIds">
delete from user where id in (#{ids})
delete>
结果报错,原因在于in后面的小括号里面的'1,2,3'
为字符串类型且为一个整体,与整数类型不符,因此不能使用#{}
使用${}
<delete id="deleteByIds">
delete from user where id in (${ids})
delete>