在MyBatis中进行懒惰提取

时间:2016-05-23 08:25:53

标签: java oracle lazy-loading mybatis pipelined-function

我无法找到如何实现延迟加载(即使在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">
    <mapper namespace="com.example.FooMyBatisLazyFetch">
        <select id="callFooProc"
                parameterType="com.example.FooProcBundle"
                statementType="CALLABLE">
            {call FooProc(
                    #{arg1,  jdbcType=VARCHAR, mode=IN},
                    #{arg2,  jdbcType=VARCHAR, mode=IN},
                    #{arg3,  jdbcType=VARCHAR, mode=IN},
                    #{error, jdbcType=NUMERIC, mode=OUT},
                    #{res2,  jdbcType=CURSOR,  mode=OUT, resultMap=FooProcResult}
                )
            }
        </select>

        <resultMap id="FooProcResult" type="com.example.FooProcResult">
            <result property="bar1" column="barcol1"/>
            <result property="bar2" column="barcol2"/>
            <result property="bar3" column="barcol3"/>
            <result property="bar4" column="barcol4"/>
            <result property="bar5" column="barcol5"/>
        </resultMap>
    </mapper>

Pojo课程:

    public class FooProcResult {
        private String bar1;
        private String bar2;
        private String bar3;
        private String bar4;
        private String bar5;
    }        

    public class FooProcBoondle {
        private String arg1;
        private String arg2;
        private String arg3;
        private Integer error;
        private List<FooProcResult> res2;
        //getters,setters, etc
    }

和使用代码;

    FooProcBundle bundle = new FooProcBundle();
    bundle.setArg1("foo");
    bundle.setArg2("bar");
    bundle.setArg3("baz");
    fooMyBatisLazyFetch.callFooProc(bundle);
    Integer error = bundle.getError();
    if(error == 123) /*some condition*/ {
        List<FooProcResult> res2 = bundle.getRes2();
        // iterate res2
    --->// Only here CURSOR may be opened and executed
    }

即。除非我的代码明确要求,否则我不想获取res2。那个特殊的游标很重,我不想在不需要时执行它(但mybatis会这样做)。

我还想将它应用于类似于生成器的过程(Oracle将它们称为“流水线表函数”,它们产生结果,休眠并等待调用者获取下一行 - 唤醒并计算下一行。通常他们这样调用:{{1 }}

有关实现此目的所需配置的任何想法?

1 个答案:

答案 0 :(得分:0)

在类org.apache.ibatis.executor.resultset.DefaultResultSetHandler中处理过程输出游标参数 在方法 handleOutputParameters 中,然后在方法 handleRefCursorOutputParameter 中。您将注意到当前状态下的代码不允许您进行搜索,唯一使用的“自定义选项”是必须提供的 resultMap 。我也会理解更多选项,例如延迟加载,自定义结果处理程序和一些日志,以便能够监视实际执行时间和获取时间。

这可以在JDBC中实现,并且需要在框架中未实现的配置。