转载地址:http://iyiguo.net/blog/2012/09/27/mybatis-param-mapping-rules/

规则

非注解参数

当参数未使用@Param注解时,可以通过以下方式访问:

#{参数位置[0..n-1]}

或者

#{param[1..n]}

如果参数类型是自定义对象Bean时,只需加上.对象属性即可

#{参数位置[0..n-1].对象属性}

#{param[1..n].对象属性}

特别的,如果非注解参数只有一个时,可使用

#{任意字符}

注解参数

@Param 注释了参数后相当于给该参数指定了一个别名。注释后的参数只能通过

#{注解别名}

或者

#{param[1..n]}

如果参数类型是自定义对象Bean时,只需加上.对象属性即可访问对象属性

#{注解别名.属性}

#{param[1..n].属性}

示例

为了明确上述规则,我们的示例具体细分了各种情况进行展示。

非注解型

一个参数
  1. User getUserById(int id);
  2. select * from <TABLE> where id = #{id}
  3. // or
  4. select * from <TABLE> where id = #{abdc}
  5. // or
  6. select * from <TABLE> where id = #{param1}
  7. User getUser(User user); // user.getName user.getAge
  8. select * from <TABLE> where name = #{name} and age = #{age}
多个参数
  1. User getUser(String name, int age);
  2. select * from <TABLE> where name = #{0} and age = #{1}
  3. // or
  4. select * from <TABLE> where name = #{param1} and age = #{param2}
  5. User getUser(User usr, int flag);
  6. select * from <TABLE> where name = #{0.name} and age = {0.age} and flag = #{1}
  7. // or
  8. select * from <TABLE> where name = #{param1.name} and age = {param1.age} and flag = #{param2}

注解型

一个参数
  1. User getUserById(@Param(value="keyId") int id);
  2. select * from <TABEL> where id = #{keyId}
  3. // or
  4. select * from <TABLE> where id = #{param1}
  5. User getUser(@Param(value="user") User user); // user.getName user.getAge
  6. select * from <TABLE> where name = #{user.name} and age = #{user.age}
  7. // or
  8. select * from <TABLE> where name = #{param1.name} and age = #{param1.age}
多个参数
  1. User getUser(@Param(value="xm") String name, @Param(value="nl") int age);
  2. select * from <TABLE> where name = #{xm} and age = #{nl}
  3. // or
  4. select * from <TABLE> where name = #{param1} and age = #{param2}
  5. // or
  6. select * from <TABLE> where name = #{xm} and age = #{param2}
  7. User getUser(@Param(value="usr") User user, @Param(value="tag") int flag);
  8. select * from <TABLE> where name = #{usr.name} and age = #{usr.age} and flag = #{tag}
  9. // or
  10. select * from <TABLE> where name = #{param1.name} and age = #{param1.age} and flag = #{param2}
  11. // or
  12. select * from <TABLE> where name = #{usr.name} and age = #{param1.age} and flag = #{param2}

非注解和注解型混合型

当采用部分参数使用@Param注解时,参数注释为将以上两种情况结合起来即可.

  1. User getUser(String name, @Param(value="nl") age, int gendar);
  2. // 对于age的访问不能是 #{1} 只能是 #{param2} | #{nl}
  3. select * from <TABLE> where name = #{0} and age = #{nl} and gendar = #{param3)

框架主要映射处理代码

参数的获取

org.apache.ibatis.binding.MapperMethod

  1. private Object getParam(Object[] args) {
  2. final int paramCount = paramPositions.size();
  3. // 无参数
  4. if (args == null || paramCount == 0) {
  5. return null;
  6. // 无注解并参数个数为1
  7. } else if (!hasNamedParameters && paramCount == 1) {
  8. return args[paramPositions.get(0)];
  9. } else {
  10. Map<String, Object> param = new MapperParamMap<Object>();
  11. for (int i = 0; i < paramCount; i++) {
  12. param.put(paramNames.get(i), args[paramPositions.get(i)]);
  13. }
  14. // issue #71, add param names as param1, param2...but ensure backward compatibility
  15. // 这就是 #{param[1..n]} 的来源
  16. for (int i = 0; i < paramCount; i++) {
  17. String genericParamName = "param" + String.valueOf(i + 1);
  18. if (!param.containsKey(genericParamName)) {
  19. param.put(genericParamName, args[paramPositions.get(i)]);
  20. }
  21. }
  22. return param;
  23. }
  24. }

SQL预编译参数设置

org.apache.ibatis.executor.parameter.DefaultParameterHandler

  1. public void setParameters(PreparedStatement ps) throws SQLException {
  2. ErrorContext.instance().activity("setting parameters").object(mappedStatement.getParameterMap().getId());
  3. List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();
  4. if (parameterMappings != null) {
  5. MetaObject metaObject = parameterObject == null ? null : configuration.newMetaObject(parameterObject);
  6. for (int i = 0; i < parameterMappings.size(); i++) {
  7. ParameterMapping parameterMapping = parameterMappings.get(i);
  8. if (parameterMapping.getMode() != ParameterMode.OUT) {
  9. Object value;
  10. String propertyName = parameterMapping.getProperty();
  11. PropertyTokenizer prop = new PropertyTokenizer(propertyName);
  12. if (parameterObject == null) {
  13. value = null;
  14. } else if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {
  15. value = parameterObject;
  16. } else if (boundSql.hasAdditionalParameter(propertyName)) {
  17. value = boundSql.getAdditionalParameter(propertyName);
  18. } else if (propertyName.startsWith(ForEachSqlNode.ITEM_PREFIX)
  19. && boundSql.hasAdditionalParameter(prop.getName())) {
  20. value = boundSql.getAdditionalParameter(prop.getName());
  21. if (value != null) {
  22. value = configuration.newMetaObject(value).getValue(propertyName.substring(prop.getName().length()));
  23. }
  24. } else {
  25. value = metaObject == null ? null : metaObject.getValue(propertyName);
  26. }
  27. TypeHandler typeHandler = parameterMapping.getTypeHandler();
  28. if (typeHandler == null) {
  29. throw new ExecutorException("There was no TypeHandler found for parameter " + propertyName + " of statement " + mappedStatement.getId());
  30. }
  31. JdbcType jdbcType = parameterMapping.getJdbcType();
  32. if (value == null && jdbcType == null) jdbcType = configuration.getJdbcTypeForNull();
  33. typeHandler.setParameter(ps, i + 1, value, jdbcType);
  34. }
  35. }
  36. }
  37. }

mybatis参数映射的更多相关文章

  1. MyBatis参数绑定规则及原理分析

    MyBatis参数的传递有几种不同的方法,本文通过测试用例出发,对其中的方式进行总结和说明,并对其部分源码进行分析. 一.测试用例(环境参考之前博客SSM接口编程一文 http://www.cnblo ...

  2. Mybatis sql映射文件浅析 Mybatis简介(三)

    简介 除了配置相关之外,另一个核心就是SQL映射,MyBatis 的真正强大也在于它的映射语句. Mybatis创建了一套规则以XML为载体映射SQL 之前提到过,各项配置信息将Mybatis应用的整 ...

  3. Mybatis sql映射文件浅析 Mybatis简介(三) 简介

    Mybatis sql映射文件浅析 Mybatis简介(三)   简介 除了配置相关之外,另一个核心就是SQL映射,MyBatis 的真正强大也在于它的映射语句. Mybatis创建了一套规则以XML ...

  4. Mybatis输入映射和输出映射

    本节内容: 输入参数映射 输出映射 resultMap Mapper.xml映射文件中定义了操作数据库的sql,每个sql是一个statement,映射文件是mybatis的核心. 一.环境准备 复制 ...

  5. Mybatis(二):Mybatis的映射文件sqlmapper详解

    MyBatis 真正的力量是在映射语句中.这里是奇迹发生的地方.对于所有的力量,SQL 映射的 XML 文件是相当的简单.当然如果你将它们和对等功能的 JDBC 代码来比较,你会发现映射文件节省了大约 ...

  6. (十一)mybatis之映射器(select)

    映射器 映射器的主要元素有八种: 元素名称 描述 select 查询语句,可自定义参数 insert 插入语句,执行后返回插入的条数 update 更新语句,执行后返回更新的条数 delete 删除语 ...

  7. Mybatis输入输出映射_动态sql_关联关系(一对一、一对多、多对多)

    Mybatis输入输出映射_动态sql_关联关系(一对一.一对多.多对多)输入输出映射parameterType完成输入映射parameterType可以传入的参数有,基本数据类型(根据id查询用户的 ...

  8. mybatis中映射文件和实体类的关联性

    mybatis的映射文件写法多种多样,不同的写法和用法,在实际开发过程中所消耗的开发时间.维护时间有很大差别,今天我就把我认为比较简单的一种映射文件写法记录下来,供大家修改建议,争取找到一个最优写法~ ...

  9. mybatis 高级映射和spring整合之查询缓存(5)

    mybatis 高级映射和spring整合之查询缓存(5) 2.0 查询缓存 2.0.1 什么是查询缓存 mybatis提供缓存,用于减轻数据压力,提高数据库性能. mybatis提供一级缓存和二级缓 ...

随机推荐

  1. 第04项目:淘淘商城(SpringMVC+Spring+Mybatis) 的学习实践总结【第一天】

    本人做过一年的MATLAB编程和简单维护过VB和C++的项目.是跟着网上获得的黑马的Java双元视频课来自学入门Java知识和常用框架的使用. 淘淘商城(SpringMVC+Spring+Mybati ...

  2. Java之同步代码块处理实现Runnable的线程安全问题

    /** * 例子:创建三个窗口卖票,总票数为100张.使用实现Runnable接口的方式 * * 1.问题:卖票过程中,出现了重票.错票 -->出现了线程的安全问题 * 2.问题出现的原因:当某 ...

  3. Deep-Learning-with-Python] 文本序列中的深度学习

    https://blog.csdn.net/LSG_Down/article/details/81327072 将文本数据处理成有用的数据表示 循环神经网络 使用1D卷积处理序列数据 深度学习模型可以 ...

  4. 嵌入式Linux环境变量如何参与程序运行

    1.环境变量一共有两份,一份在Flash中,另一份在DDR中.uboot开机时一次性从Flash中读取全部环境变量到DDR中作为环境变量的初始化值,然后使用过程中都是用DDR这一份,用户可以用save ...

  5. JKFZ%你赛炸裂祭

    Md爆40了身败名裂 上来就刚T1是什么习惯?居然不看T2导致明明能刚出正解却止步40 , T3找到原题看懂题解后却不敢交+难码 , 最近怕不是做毒瘤%你赛多了总以为T1能刚到点分 md最近怕不是炸了 ...

  6. POJ 3585 Accumulation Degree【换根DP】

    传送门:http://poj.org/problem?id=3585 题意:给定一张无根图,给定每条边的容量,随便取一点使得从这个点出发作为源点,发出的流量最大,并且输出这个最大的流量. 思路:最近开 ...

  7. CSS 定位体系概述

    三种定位体系简介 框( box )布局影响因素之一,便是定位体系.定位体系也是其最为重要的影响因素. CSS2.1 中,一个框可以根据三种定位体系布局.CSS2.1 中的定位体系帮助作者使他们的文档更 ...

  8. hdu2222 (AC自动机模板)

    题:http://acm.hdu.edu.cn/showproblem.php?pid=2222 学习出:https://bestsort.cn/2019/04/28/402/ 主要是fail的建立. ...

  9. Linux常见指令x-mind

  10. 史上最强maven配置详情

    史上最强maven配置详情 优点 对第三方依赖库进行了统一的版本管理 统一了构建过程 统一了项目的目录结构 构建 清理 : mvn clear 编译 : mvn compile 测试 : mvn te ...