mybatis3调用oracle存储过程
来源:互联网 发布:预科生的贩毒网络 srt 编辑:程序博客网 时间:2024/06/06 05:52
oracle的存储过程,如果要查询数据必须有一个游标供使用
先看下简单的存储过程
- CREATE OR REPLACE PROCEDURE zdrqlx_PROC
- (
- V_TEMP OUT zdrqlxPackage.zdrqlx_cursor
- )
- AS
- BEGIN
- OPEN V_TEMP FOR SELECT LXMC FROM TBGW_ZHDRQFL ;
- END zdrqlx_PROC;
这个存储过程查询后 结果是 一列数据 列名 LXMC
下面贴查询mybatis的配置
- <resultMap id="teshurenqunMap" type="bean.admin.menus.ZhongDianRenQunBean" >
- <result column="lxmc" property="lxmc" />
- </resultMap>
- <select id="selectByMap" parameterType="map" statementType="CALLABLE" >
- { call zdrqlx_PROC(#{listinfo,mode=OUT,jdbcType=CURSOR,javajavaType=java.sql.ResultSet,resultMap=teshurenqunMap})}
- </select>
程序调用的是selectByMap 这个方法 在调用的时候 需要传入一个参数 这里参数名是map ,map内可以什么属性都没有, 在mybatis调用存储过程后,会把数据 回塞给这个map
所以 如果你的程序是 Map resultmap = dao.selectByMap(map);
然后去读取resultmap ,结果会是null的 什么都没有,而数据实际上在被存在参数的map里面 而不是resultmap (map 是参数map,resultmap 是返回接收的) 这点很重要,我被卡住了半天!!!
下面是调用 看了上面一段话 下面的程序一目了然
- Map map = new HashMap();
- menusService.getToolsBarMunus(map);//这里调用selectByMap
- map.put("success", true);
- /这边数据会存在这个参数map里,是被回填的了
- datas.append(com.alibaba.fastjson.JSON.toJSONString(map));
com.alibaba.fastjson.JSON.toJSONString(map)
这句话是用的json组件转化map得到string,结果是下面的
{"listinfo":[{"lxmc":"张三"},{"lxmc":"李四"},{"lxmc":"王五"}{"lxmc":"赵六"}]}
根据结果,反过来看调用存储过程
{ call zdrqlx_PROC(#{listinfo,mode=OUT,jdbcType=CURSOR,javaType=java.sql.ResultSet,resultMap=teshurenqunMap})}
1. listinfo 这个是第一个参数,我是随意写的,不和任何东西管理,看下上面json转化后的map结果就明白了
2 mode=OUT 声明此参数作为输出类型
3 jdbcType=CURSOR 说明他是游标 因为这个存储过程主要是查询,需要传入一个游标参数,不信的话你在plsql里面test存储过程 看下面就明白了
4 javaType=java.sql.ResultSet, 貌似固定要求这样,求大神解答
5 resultMap 必须有这个参数 我这里配置的 是resultMap id="teshurenqunMap" 对应的
最后是我的bean 也就是resultMap 设置的type=
- package bean.admin.menus;
- import java.util.List;
- public class ZhongDianRenQunBean {
- private String lxmc;
- public String getLxmc() {
- return lxmc;
- }
- public void setLxmc(String lxmc) {
- this.lxmc = lxmc;
- }
- }
-------------------------------------------------------------------------------------------------------------------------------------------------------------------
mybatis调用存储过程
参数形式:- create procedure sptest.adder(in addend1 integer, in addend2 integer, out theSum integer)
- begin atomic
- set theSum = addend1 + addend2;
- end
- go
- <parameterMap type="map" id="testParameterMap">
- <parameter property="addend1" jdbcType="INTEGER" mode="IN"/>
- <parameter property="addend2" jdbcType="INTEGER" mode="IN"/>
- <parameter property="sum" jdbcType="INTEGER" mode="OUT"/>
- </parameterMap>
- lt;update id="adderWithParameterMap" parameterMap="testParameterMap" statementType="CALLABLE">
- {call sptest.adder(?, ?, ?)}
- </update>
- public void testAdderAsUpdateWithParameterMap() {
- SqlSession sqlSession = sqlSessionFactory.openSession();
- try {
- Map<String, Object> parms = new HashMap<String, Object>();
- parms.put("addend1", 3);
- parms.put("addend2", 4);
- SPMapper spMapper = sqlSession.getMapper(SPMapper.class);
- spMapper.adderWithParameterMap(parms);
- assertEquals(7, parms.get("sum"));
- parms = new HashMap<String, Object>();
- parms.put("addend1", 2);
- parms.put("addend2", 3);
- spMapper.adderWithParameterMap(parms);
- assertEquals(5, parms.get("sum"));
- } finally {
- sqlSession.close();
- }
带输入输出参数的存储过程:
sql代码:
- create procedure sptest.getnames(in lowestId int, out totalrows integer)
- reads sql data
- dynamic result sets 1
- BEGIN ATOMIC
- declare cur cursor for select * from sptest.names where id >= lowestId;
- select count(*) into totalrows from sptest.names where id >= lowestId;
- open cur;
- END
- go
- <select id="getNamesAndItems" statementType="CALLABLE"
- <select id="getNames" parameterType="java.util.Map" statementType="CALLABLE"
- resultMap="nameResult">
- {call sptest.getnames(
- #{lowestId,jdbcType=INTEGER,mode=IN},
- #{totalRows,jdbcType=INTEGER,mode=OUT})}
- </select>
- </select>
- public void testCallWithResultSet2_a1() {
- SqlSession sqlSession = sqlSessionFactory.openSession();
- try {
- SPMapper spMapper = sqlSession.getMapper(SPMapper.class);
- Map<String, Object> parms = new HashMap<String, Object>();
- parms.put("lowestId", 1);
- List<Name> names = spMapper.getNamesAnnotated(parms);
- assertEquals(3, names.size());
- assertEquals(3, parms.get("totalRows"));
- } finally {
- sqlSession.close();
- }
- }
返回多个结果集
sql代码:
- create procedure sptest.getnamesanditems()
- reads sql data
- dynamic result sets 2
- BEGIN ATOMIC
- declare cur1 cursor for select * from sptest.names;
- declare cur2 cursor for select * from sptest.items;
- open cur1;
- open cur2;
- END
- go
- <resultMap type="org.apache.ibatis.submitted.sptests.Name" id="nameResult">
- <result column="ID" property="id"/>
- <result column="FIRST_NAME" property="firstName"/>
- <result column="LAST_NAME" property="lastName"/>
- </resultMap>
- <resultMap type="org.apache.ibatis.submitted.sptests.Item" id="itemResult">
- <result column="ID" property="id"/>
- <result column="ITEM" property="item"/>
- </resultMap>
- <select id="getNamesAndItems" statementType="CALLABLE"
- resultMap="nameResult,itemResult">
- {call sptest.getnamesanditems()}
- </select>
- @Test
- public void testGetNamesAndItems() throws SQLException {
- SqlSession sqlSession = sqlSessionFactory.openSession();
- try {
- SPMapper spMapper = sqlSession.getMapper(SPMapper.class);
- List<List<?>> results = spMapper.getNamesAndItems();
- assertEquals(2, results.size());
- assertEquals(4, results.get(0).size());
- assertEquals(3, results.get(1).size());
- } finally {
- sqlSession.close();
- }
- }
注意:
上面就是几种常用的了。
1 sqlserver oracle sqlserver返回结果集是可以不要out参数的。如果sql中用的是select出结果,不需要配置out参数。多个结果集/结果集可以配置resultMap 来返回LIST,主要是调用selectList方法会自动把结果集加入到list中去的。
2 sql有返回值 用select标签
3 注意sql参数和mybatis参数的对应关系,这个这里就不讲了。
4 注意参数个数
我遇到的异常:
参数不匹配的原因,因为sqlserver 中我是直接返回select临时表结果,不需要配置存储过程输出参数。
list中的内容形式:
------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
存储过程都是一样的,只是根据自己的喜好,可以用MAP或者JAVABEAN传递参数。
- -- --------------------------------------------------------------------------------
- -- Routine DDL
- -- Note: comments before and after the routine body will not be stored by the server
- -- --------------------------------------------------------------------------------
- DELIMITER $$
- CREATE DEFINER=`root`@`localhost` PROCEDURE `selectCount`(
- IN pcsId int,
- IN drId int,
- IN partnerId int,
- IN customerId int,
- OUT pcsCount int,
- OUT drCount int
- )
- BEGIN
- select count(md.id) into @pcsC from mdm_device md
- left join mdm_device_security mds on mds.device_id = md.id
- where mds.device_rooted = pcsId
- and md.partner_id = partnerId and md.customer_id = customerId;
- set pcsCount = @pcsC;
- select count(md.id) into @drC from mdm_device md
- where md.managed_status = drId and DATE_SUB(CURDATE(), INTERVAL 7 DAY) <= date(md.un_manage_date)
- and md.partner_id = partnerId and md.customer_id = customerId;
- set drCount = @drC;
- END
1.java调用传入MAP。获取通过MAP获取。
1.1 mapper文件写法
- <parameterMap type="map" id="homeVO">
- <parameter property="pcsId" jdbcType="INTEGER" mode="IN"/>
- <parameter property="drId" jdbcType="INTEGER" mode="IN"/>
- <parameter property="partnerId" jdbcType="INTEGER" mode="IN"/>
- <parameter property="customerId" jdbcType="INTEGER" mode="IN"/>
- <parameter property="pcsCount" jdbcType="INTEGER" mode="OUT"/>
- <parameter property="drCount" jdbcType="INTEGER" mode="OUT"/>
- </parameterMap>
- <select id="selectForHome" parameterMap="homeVO"
- statementType="CALLABLE">
- {call selectCount(
- ?,?,?,?,?,?
- )}
- </select>
1.2 java调用写法
- @Override
- public StringselectHomeCount(HomeVO home) throws Exception {
- Map<String, Object> map = new HashMap<String, Object>();
- map.put("pscId", 0);
- map.put("drId", 1);
- map.put("partnerId", 25);
- map.put("customerId", 50);
- map.put("isolation", 1);
- selectOne("Mapper.selectForHome", map);
- System.out.println(map.get("pcsCount"));
- return map.get("drCount");
- }
2.java调用传入javaBean。返回值通过javaBean属性获取
2.1 通过javabean传递参数 mapper写法
- <parameterMap type="com.polysaas.mdm.device.entity.HomeVO" id="home">
- <parameter property="pcsId" jdbcType="INTEGER" mode="IN"/>
- <parameter property="drId" jdbcType="INTEGER" mode="IN"/>
- <parameter property="partnerId" jdbcType="INTEGER" mode="IN"/>
- <parameter property="customerId" jdbcType="INTEGER" mode="IN"/>
- <parameter property="pcsCount" jdbcType="INTEGER" mode="OUT"/>
- <parameter property="drCount" jdbcType="INTEGER" mode="OUT"/>
- </parameterMap>
javaBean可以通过两种来进行映射
- <resultMap type="com.polysaas.mdm.device.entity.HomeVO" id="home">
- <result column="partnerId" property="partnerId" jdbcType="INTEGER"/>
- <result column="customerId" property="customerId" jdbcType="INTEGER" />
- <result column="pcsId" property="pcsId" jdbcType="INTEGER" />
- <result column="drId" property="drId" jdbcType="INTEGER" />
- <result column="pcsCount" property="pcsCount" jdbcType="INTEGER" />
- <result column="drCount" property="drCount" jdbcType="INTEGER" />
- </resultMap>
- <select id="selectForHome2" parameterType="com.polysaas.mdm.device.entity.HomeVO"
- statementType="CALLABLE">
- {call selectCount(
- #{pcsId,jdbcType=INTEGER, mode=IN},
- #{drId, jdbcType=INTEGER,mode=IN},
- #{partnerId, jdbcType=INTEGER,mode=IN},
- #{customerId, jdbcType=INTEGER,mode=IN},
- #{pcsCount, jdbcType=INTEGER,mode=OUT},
- #{drCount, jdbcType=INTEGER,mode=OUT}
- )}
- </select>
2.2 通过javabean传递参数 java调用写法
- @Override
- public HomeVO selectHomeCount(HomeVO home) throws Exception {
- HomeVO vo = new HomeVO();
- vo.setPcsId(0);
- vo.setDrId(0);
- vo.setPartnerId(25);
- vo.setCustomerId(50);
- vo.setPcsCount(0);
- vo.setDrCount(0);
- selectOne("com.polysaas.mdm.device.mapper.MdmDeviceMapper.selectForHome2", vo);
- return vo;
- }
遇到异常很多。。记得的写下
1.通过javaBean的时候,#{pcsId, mode=IN, javaType=INTEGER},javaType这个属性是必须的。
2.使用javaBean传递。输出参数不需要初始值。个人喜好并建议使用javaBean,因为定义更清晰,封装性。
3.map可以用占位符问号,javaBean不可以。 会有异常 No value specified for parameter 1
4. 说不认识某个参数。。可能是没有创建存储过程,这个发生在多个库的时候
5. read Only 这个可能是事务控制只读。。我是因为存储过程有set 语句,而方法叫做selectAaaa()。
- mybatis3调用oracle存储过程
- myBatis3 存储过程调用
- myBatis3执行存储过程
- oracle调用存储过程
- Oracle调用存储过程
- oracle调用存储过程
- oracle 存储过程调用
- Oracle存储过程调用存储过程
- oracle 存储过程 调用存储过程
- oracle存储过程调用存储过程
- ORACLE存储过程调用存储过程
- oracle 存储过程中调用存储过程
- java调用oracle存储过程
- java 调用 oracle 存储过程
- ASP调用oracle存储过程
- C#调用oracle存储过程
- java调用存储过程(oracle)
- C#调用oracle存储过程
- 第十五章作业~~~
- SNMP--Windows API--SnmpUtilOidFree
- javaSE-单例设计模式的线程安全
- Binary Tree Preorder Traversal
- VC与Matlab混合编程
- mybatis3调用oracle存储过程
- 创建子程序的理由
- SWT自定义布局
- squid+nginx+varnish资料整理
- PowerDesigner15连接Oracle失败的解决办法
- VirtualBox安装及使用说明和虚拟机安装XP系统图文教程
- 使用wsimport生成本地调用代码
- 整洁编程
- C# WebBrowser实现网页自动填表