MyBatis不⽤@Param传递多个参数的操作
背景
假设我们要保存⼀个⼩狗的信息到数据库中
通常的做法
我们在使⽤mybatis 接⼝和xml对应的时候,常常是这样写的:
接⼝
public interface DogDao {
void save(@Param("dogName") String dogName, @Param("age")int age);
}
xml
<insert id="save">
INSERT INTO dog
values (null, #{dogName}, #{age});
</insert>
因为mybatis在执⾏的之后动态⽣成实现类,⽽在java中使⽤反射会将⽅法中的参数名称擦除,所以如果在xml中要接收到规定参数名称的参数,通过注解标识是简单⼜可⾏的⽅法,⼀⽅⾯可以⾃定义⽅法参数的绑定,另外⼀⽅⾯在xml中也可以起有意义的名称与
@Param中的value对应。
如果不加@Param注解,也想⽤xml接收响应的参数
public interface DogDao {
void save(Integer id, String dogName, int age);
}
由于在反射和jdk动态代理会将⽅法名称抹除,所以在解析参数的时候不能像上⾯的xml#{dogName}这
样接收,会将参数的名称序列化成以下⽅式,param1…paramn
所以在xml中我们要⽤以下⽅式来接收:
<insert id="save">
INSERT INTO dogparam name
values (null, #{param2}, #{param2});
</insert>
不想⽤@Param修饰,但是希望在xml中⽤⽅法中的原参数名称接收
我们在jdk不能使⽤反射获取参数名称,但是在jdk1.8之后提供了Parameter这个反射类,可以配置-parameter这个参数到javac编译器上,可以⽤来获取⽅法参数上的名称,但是javac编译器默认是关闭的,所以我们在编译代码的时候应该打开它,我们⽤maven插件的⽅
式开启
在:pom⽂件中加⼊以下插件:
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.0</version>
<configuration>
<compilerArgument>-parameters</compilerArgument>
<encoding>UTF-8</encoding>
</configuration>
</plugin>
这样,javac编译器就能将⽅法的参数在反射编译的时候⼀同传递了,我们也不⽤借助注解就能达到在xml中获取⽅法中的实际参数的名称,除了加⼊编译插件,还要加⼊mybatis的配置开启:
<settings>
<setting name="useActualParamName" value="false"/>
</settings>
官⽅⽂档是这样对这个配置说明的:
允许使⽤⽅法签名中的名称作为语句参数名称。为了使⽤该特性,你的项⽬必须采⽤ Java 8 编译,并且加上 -parameters 选项。(新增于 3.4.1)
java 代码:
public interface DogDao {
void save(Integer id, String dogName, int age);
}
xml代码:
<insert id="save">
INSERT INTO dog
values (null, #{dogName}, #{age});
</insert>
补充:mybatis传多个参数(不使⽤@param注解情况和使⽤@param注解的情况)
⽅法1:顺序传参法
1.不使⽤@param注解传递多个参数的情况
注: 使⽤jdk1.7得到的是: [1, 0, param1, param2]
使⽤1.8得到的则是: [arg1, arg0, param1, param2]
#{}⾥⾯的数字代表你传⼊参数的顺序。
这种⽅法不建议使⽤,sql层表达不直观,且⼀旦顺序调整容易出错。
举个栗⼦:
Dao层
List<User> demo(int userid, String name);
对应的XML编写
jdk1.7
<select id="demo" resultMap="User">
select *
from user where user_id=#{0} and name= #{1}
</select>
jdk1.8之后
第⼀种写法
<select id="demo" resultMap="User">
select *
from user where user_id=#{arg0} and name= #{arg1}
</select>
第⼆种写法
<select id="demo" resultMap="User">
select *
from user where user_id=#{param0} and name= #{param1}
</select>
⽅法2:@Param注解传参法
public User selectUser(@Param("userName") String name, int @Param("deptId") deptId);
<select id="selectUser" resultMap="UserResultMap">
select * from user
where user_name = #{userName} and dept_id = #{deptId}
</select>
#{}⾥⾯的名称对应的是注解@Param括号⾥⾯修饰的名称。
这种⽅法在参数不多的情况还是⽐较直观的,推荐使⽤。
⽅法3:Map传参法
public User selectUser(Map<String, Object> params);
<select id="selectUser" parameterType="java.util.Map" resultMap="UserResultMap">
select * from user
where user_name = #{userName} and dept_id = #{deptId}
</select>
#{}⾥⾯的名称对应的是Map⾥⾯的key名称。
这种⽅法适合传递多个参数,且参数易变能灵活传递的情况。
⽅法4:Java Bean传参法
public User selectUser(User user);
<select id="selectUser" parameterType="st.User" resultMap="UserResultMap">
select * from user
where user_name = #{userName} and dept_id = #{deptId}
</select>
#{}⾥⾯的名称对应的是User类⾥⾯的成员属性。
这种⽅法很直观,但需要建⼀个实体类,扩展不容易,需要加属性,看情况使⽤。
以上为个⼈经验,希望能给⼤家⼀个参考,也希望⼤家多多⽀持。如有错误或未考虑完全的地⽅,望不吝赐教。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论