【问题标题】:Mapping a nested Oracle CURSOR (ResultSet) with myBatis用 myBatis 映射一个嵌套的 Oracle CURSOR (ResultSet)
【发布时间】:2016-02-07 00:31:27
【问题描述】:

是否可以在 myBatis resultMap 中映射嵌套的 java.sql.ResultSet

例如。假设我有一个这样定义的过程映射:

 <select id="selectBlog" statementType="CALLABLE">
        {call getCarsByYear(
           #{year,jdbcType=INTEGER,mode=IN},
           #{results, jdbcType=CURSOR, mode=OUT, javaType=java.sql.ResultSet, jdbcType=CURSOR, resultMap=cars}
        )}
    </select>

还有我的映射器。它返回汽车对象列表,还返回经销商列表(嵌套的 CURSOR):

<resultMap id="cars" type="some.package.Car">
   <result property="name" column="car_name">
   <!-- here is my problem -->
   <collection property="dealerships" column="dealerships_rf" ofType="some.package.Dealership">
       <result property="model" column="model" />
       <result property="year" column="year" />
</resultMap>

<!-- dealership resultMap of type some.package.Dealership -->

这里的问题是,当我检查生成的 java 对象时,dealerships 是一个空列表。

我编写了一些普通的 java.sql JDBC 代码,它运行良好。任何人都可以让我走上正确的道路吗?我完全迷失了这个。

提前致谢。

这是预期的 SQL 输出:

Car
|name  |dealerships|
|nissan|ref_cursor|

Dealership
|location     |established|....
|....         |1974       |...

车型:

public class Car {

 private String name;
 private List<Dealership> dealerships;

 // getters & setters ...

}

public class Dealership {

  private String model;
  private Integer year;

  // getters & setters ...
}

【问题讨论】:

  • 更新了描述以提供有关该问题的更多信息。 @diziaq
  • 请添加过程'getCarsByYear'的PL/SQL代码。
  • 我认为您只是没有在 XML 文件中关闭 标记。可能是这样吗?

标签: java xml oracle jdbc mybatis


【解决方案1】:

我做了一个例子来展示它是如何工作的。

包模型包含两个类:

public class Result {
    public int start_from;    
    public List<Model> models;
}

public class Model {
    public int a;
    public String b;
}

存储过程

CREATE OR REPLACE PROCEDURE get_data( p_start IN NUMBER
                                    , p_cur   OUT SYS_REFCURSOR)
IS
BEGIN

  OPEN p_cur FOR
    SELECT p_start a,'abc' b FROM dual
          UNION ALL
    SELECT p_start + 1,'cde' FROM dual
          UNION ALL
    SELECT p_start + 2,'xyz' FROM dual;

END;

mybatis-config.xml(必须提供数据库的 URL)

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD Config 3.0//EN" "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
  <environments default="development">
    <environment id="development">
      <transactionManager type="JDBC"/>
      <dataSource type="POOLED">
        <property name="driver" value="oracle.jdbc.OracleDriver"/>
        <property name="url" value="${set_the_url}"/>
      </dataSource>
    </environment>
  </environments>

  <mappers>
    <mapper resource="mybatis-mapper.xml"/>
  </mappers>
</configuration>

mybatis-mapper.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">

  <resultMap id="map_res_4" type="models.Model">
    <result property="a" column="a"/>
    <result property="b" column="b"/>
  </resultMap>

  <parameterMap id="map_par_4" type="models.Result">
    <parameter property="start_from" jdbcType="INTEGER" mode="IN"  />
    <parameter property="models" jdbcType="CURSOR" mode="OUT" resultMap="map_res_4"  />
  </parameterMap>

  <select id="select_4" parameterMap="map_par_4" statementType="CALLABLE">
    {CALL get_data(?, ?)}
  </select>

</mapper>

以及用mybatis调用过程get_data的示例:

您会注意到selectOne 方法返回null,因为我们执行了一个可调用语句。与过程调用的所有交互都使用第二个参数:我们传递start_from 并接收models 作为 Result 对象的字段(MyBatis 可以通过反射获取和设置它们)。所以 Result 对象在方法调用之前和之后是相同的:你甚至可以将字段设为私有(这里我将它们设为公开以保持代码更短)。

import models.Result;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;

import java.io.IOException;
import java.io.Reader;

/**
 *
 */
public class Main {
    private static SqlSessionFactory sessionFactory = null;
    private static String CONFIGURATION_FILE = "mybatis-config.xml";

    static {
        try {
            Reader reader = Resources.getResourceAsReader(CONFIGURATION_FILE);
            sessionFactory = new SqlSessionFactoryBuilder().build(reader);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String... args) {

        SqlSession session = sessionFactory.openSession();

        Result res = new Result();

        res.start_from = 5;

        Object obj = session.selectOne("select_4", res);

        // `obj` must be NULL
        // `res` contains all the results of Oracle procedure call
    }
}

【讨论】:

  • 我已经尝试过您提供的解决方案,但没有成功。经销商仍然是空的
  • @JohnStrong,我用完整的工作示例更新了答案。
  • 感谢更新示例,但我熟悉 parameterMapresultMaps 在使用游标时的典型行为。我的问题与当您在作为 OUT 参数返回的游标中具有嵌套游标时有关。例如该过程返回模型列的游标,其中返回的列之一也是游标。所以我们可能会返回类似这样的内容:(NUMBER, VARCHAR, CURSOR)。如何让mybatis解析嵌套游标?
  • 仅供参考:我可以通过在游标列中添加 typeHandler 来解析嵌套游标。然而,这需要一些手动映射工作。我宁愿找到一个允许我使用纯 xml 的解决方案(如果可能的话)。
  • 我已经用现在的程序更新了我的描述。也许这会让我更深入地了解我的问题。
【解决方案2】:

来自 ibatis 的例子:

<parameter property="result" javaType="java.sql.ResultSet" jdbcType="ORACLECURSOR" mode="OUT"/>

与你代码的区别:

 - javaType was specified
 - jdbcType = ORACLECURSOR

参见iBatis Oracle REF CURSOR的示例

   <sqlMap namespace="KOMUNIKA_REPORT">
   <resultMap id="BaseResultMap" class="javaapplication4.StockAreaAndWarehouse" >
   <result column="PRODUCT_CODE" property="productCode"  />
   <result column="PRODUCT_NAME" property="productName" />
   <result column="INCOMING" property="incoming" />
   <result column="UNIT_SOLD" property="unitSold" />
   <result column="TOTAL_STOCK" property="totalStock" />
   </resultMap>
   <parameterMap id="resultMap" class="java.util.Map">
   <parameter property="result" javaType="java.sql.ResultSet" jdbcType="ORACLECURSOR" mode="OUT"/>
   </parameterMap>
   <procedure id="selectStockAreaAndWarehouse"
   parameterMap="resultMap"
   resultMap="BaseResultMap"
   >
   { call KOMUNIKA.LP_STOCK_AREA_WAREHOUSE(?) }
   </procedure>
   </sqlMap>

example for ResultSet

<resultMap id="userDataResultMap" type="TestUserData"> 
<id property="userid" column="userid" /> 
<result property="firstName" column="firstName"/> 
<result property="lastName" column="lastName"/> 
<result property="zip" column="zip"/> 
<result property="summary" column="summary"/> 
<result property="specialities" column="specialities"/> 
<result property="isActive" column="isActive"/> 
<result property="country" column="country"/> 
<result property="platform" column="platforms"/> 
</resultMap>
<select id="getFullPublicData" statementType="CALLABLE" parameterType="User" >
{call p_user_public_data(#{userId}
,#{userDataList,mode=OUT,jdbcType=CURSOR,javaType=java.sql.ResultSet, resultMap=com.test.data.UserPublicViewMapper.userDataResultMap}
,#{noOfConnections,mode=OUT,jdbcType=NUMERIC,javaType=int}
,#{noOfRecommendations,mode=OUT,jdbcType=NUMERIC,javaType=int})}
</select>

【讨论】:

  • 此示例适用于parameterMap。我正在寻找在resultMap 中返回的嵌套游标的解决方案。我尝试将 jdbcType 设置为 CURSOR 并将 javaType 设置为 java.sql.ResultSet,仍然没有运气。
  • 你能把你的程序的标题添加到描述中吗?
  • 好吧,对于这种情况,我无法阅读该过程,但我知道它返回值名称(VARCHAR)和经销商(ref_cursor)的ref_cursor。经销商可以返回 N 个结果(因此我在描述中将其映射到 collection
  • 是函数还是过程? my_proc(name out varchar, dealers out ref cursor)my_proc(name out varchar) return sys_refcursor?
  • 看到这个链接issues.apache.org/jira/browse/IBATIS-745他们使用statementType="CALLABLE"#{userDataList,mode=OUT,jdbcType=CURSOR,javaType=ResultSet, resultMap=userDataResultMap}
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-11-12
  • 1970-01-01
  • 2015-03-19
  • 2020-01-27
  • 2020-03-06
  • 1970-01-01
相关资源
最近更新 更多