mybatis resultmap 为对象赋值的调用顺序

写了一个mybatis的mapper映射文件,其中java bean定义如下:

public class GroupCourseResult extends GroupResult {

    private String cid;
    private String cname;

    public GroupCourseResult(int stuschool, int nian, String stuclass, String cid, String cname) {
        super(stuschool, nian, stuclass);
        this.cid = cid;
        this.cname = cname;
    }

    public String getCid() {
        return cid;
    }

    public void setCid(String cid) {
        this.cid = cid;
    }

    public String getCname() {
        return cname;
    }

    public void setCname(String cname) {
        this.cname = cname;
    }
}

部分mybatis映射文件如下:

<select id="selectFailedCourseRationByGroupIdAndCourseName" resultType="GroupCourseResult">
      ...
  </select>

实体类中的属性名和查询的列名完全匹配,但是没有查询stuclass,则封装后的实体类中的stuclass属性应该为空。
然而程序运行后,stuclass属性不仅不为空,还与cname完全相同,百思不得其解,故翻了翻mybatis的源码。
在mybatis中的DefaultResultSetHandler类中,createResultObject方法的代码如下:

  private Object createResultObject(ResultSetWrapper rsw, ResultMap resultMap, List<Class<?>> constructorArgTypes, List<Object> constructorArgs, String columnPrefix)
      throws SQLException {
    final Class<?> resultType = resultMap.getType();
    final MetaClass metaType = MetaClass.forClass(resultType, reflectorFactory);
    final List<ResultMapping> constructorMappings = resultMap.getConstructorResultMappings();
    if (hasTypeHandlerForResultObject(rsw, resultType)) {
      return createPrimitiveResultObject(rsw, resultMap, columnPrefix);
    } else if (!constructorMappings.isEmpty()) {
      return createParameterizedResultObject(rsw, resultType, constructorMappings, constructorArgTypes, constructorArgs, columnPrefix);
    } else if (resultType.isInterface() || metaType.hasDefaultConstructor()) {
      return objectFactory.create(resultType);
    } else if (shouldApplyAutomaticMappings(resultMap, false)) {
      return createByConstructorSignature(rsw, resultType, constructorArgTypes, constructorArgs);
    }
    throw new ExecutorException("Do not know how to create an instance of " + resultType);
  }

经过调试发现,mybatis会先查找该javabean有无默认构造方法,如果有则采用设值注入,若没有,则根据javabean的有参构造方法进行设值,而在8以前的jdk版本中,我们利用反射只能获取到参数类型,不能获取到参数名称,这其中设值可能出现了匹配失误,将cname的值同时赋给了cname和stuclass。想要解决这个问题,只须在javabean中添加默认构造方法即可。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 使用MyBatis调用Oracle存储过程可以按照以下步骤进行: 1. 在mapper XML 文件中定义存储过程的调用语句,可以使用 select、update、insert 等标签,如: ``` <select id="callProcedure" statementType="CALLABLE"> {call PROCEDURE_NAME(#{param1,mode=IN,jdbcType=VARCHAR},#{param2,mode=OUT,jdbcType=CURSOR,resultMap=RESULT_MAP})} </select> ``` 其中,PROCEDURE_NAME 为存储过程的名称,param1 和 param2 分别为输入参数和输出参数的名称,RESULT_MAP 是输出结果集的映射关系。 2. 在代码中调用 mapper 方法,如: ``` MyMapper mapper = sqlSession.getMapper(MyMapper.class); Map<String, Object> params = new HashMap<String, Object>(); params.put("param1", "input_value"); mapper.callProcedure(params); List<ResultType> results = (List<ResultType>)params.get("param2"); ``` 其中,MyMapper 为 mapper 的接口类,sqlSession 是 MyBatis 的会话对象,params 是参数 Map 对象,ResultType 是输出结果集的类型。调用 mapper 方法后,会将参数传入存储过程,执行存储过程,并将输出结果集放入 params 对象的 param2 属性中,最后通过 params.get("param2") 可以获取输出结果集。 ### 回答2: 在MyBatis调用Oracle存储过程需要以下步骤: 1. 在Oracle数据库中创建存储过程。示例如下: CREATE OR REPLACE PROCEDURE my_proc (param1 IN VARCHAR2, param2 OUT VARCHAR2) AS BEGIN -- 存储过程的逻辑代码 -- 可以使用IN参数param1进行操作 -- 将OUT参数param2赋值 END; 2. 在MyBatis的Mapper XML文件中编写调用存储过程的代码。示例如下: <insert id="callMyProc" statementType="CALLABLE"> {call my_proc(#{param1}, #{param2, mode=OUT, jdbcType=VARCHAR})} </insert> 3. 在Java代码中调用Mapper中的方法。示例如下: String param1 = "value1"; String param2; MyMapper mapper = sqlSession.getMapper(MyMapper.class); mapper.callMyProc(param1, param2); System.out.println("param2: " + param2); 在上述代码中,param1为传入存储过程的参数值,param2为输出参数。通过调用Mapper中的方法,将param1传入存储过程,并将结果赋值给param2。 其中,CALLABLE为MyBatis的一个特殊语句类型,用于调用存储过程或函数。#{param1}和#{param2}用于替换存储过程中的参数。 需要注意的是,通过Mapper调用存储过程后,param2的值会被存储过程赋值。在上述示例中,可以通过System.out.println打印param2的值。 以上就是在MyBatis调用Oracle存储过程的简单示例代码。根据实际业务需求,可以调整参数和存储过程的逻辑。 ### 回答3: 使用MyBatis调用Oracle存储过程需要按照下面的步骤进行操作: 1. 创建存储过程:首先在Oracle数据库中创建存储过程,定义好输入参数、输出参数以及存储过程的逻辑。 2. 在MyBatis的Mapper文件中编写调用存储过程的代码: - 在Mapper文件的命名空间中引入Oracle存储过程的标识符: ``` <mapper namespace="com.example.mapper.MyMapper"> <package name="oracle"/> </mapper> ``` - 编写调用存储过程的代码: ``` <select id="callProcedure" statementType="CALLABLE"> {call PROCEDURE_NAME(#{param1, mode=IN}, #{param2, mode=OUT, jdbcType=VARCHAR})} </select> ``` 其中,`PROCEDURE_NAME`是存储过程的名称,`param1`和`param2`分别是存储过程的输入参数和输出参数。设置`mode=IN`表示输入参数,`mode=OUT`表示输出参数,使用`jdbcType=VARCHAR`指定输出参数的类型。 3. 在Java代码中调用Mapper接口方法: ``` @Autowired private MyMapper myMapper; public void callProcedure() { String inputParam = "input"; Map<String, Object> outputParam = new HashMap<>(); outputParam.put("param2", null); myMapper.callProcedure(inputParam, outputParam); String outputValue = (String) outputParam.get("param2"); System.out.println("Output Parameter: " + outputValue); } ``` 在调用存储过程的方法中,传入输入参数的值和一个Map用于接收输出参数的值。在方法执行后,可以从Map中取得输出参数的值。 注意:在进行存储过程调用时,需要确保MyBatis的配置中启用了对Oracle存储过程的支持。 这是使用MyBatis调用Oracle存储过程的基本步骤,根据具体需求和存储过程的不同,代码可能会有所变化。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值