【Java】【Mybatis】如何调用存储过程和存储函数
https://www.jb51.net/article/230756.htm
Mybatis调用存储过程
MyBatis支持使用存储过程的配置。当使用存储过程时,需要设置一个参数“mode”,其值有IN(输入参数)、OUT(输出参数)和INOUT(输入/输出参数)。
Mybatis定义存储过程如下:
<!-- 存储过程 -->
<select id="selectSomeThing" statementType="CALLABLE" parameterType="hashmap" resultType="com.pjb.mybatis.po.User">
{CALL PROC_FOR_INPUT(#{information,mode=IN,jdbcType=VARCHAR})}
</select>
示例1
创建存储过程,实现分页查询用户列表,并返回数据总数和总页数,通过MyBatis调用该存储过程。
(1)在MySQL数据库中创建用户信息表(tb_user)。
-- 创建“用户信息”数据表
CREATE TABLE IF NOT EXISTS tb_user
(
id INT AUTO_INCREMENT PRIMARY KEY COMMENT '用户编号',
user_name VARCHAR(50) NOT NULL COMMENT '用户姓名',
sex CHAR(2) DEFAULT '男' COMMENT '性别'
) COMMENT = '用户信息表';
(2)创建存储过程,实现分页查询用户列表,并返回数据总数和总页数。

1 -- 将结束标志符更改为$$
2 DELIMITER $$
3
4 /*
5 -- 存储过程:分页查询用户列表,并返回数据总数和总页数
6 -- 输入参数:page_index:当前页码
7 -- 输入参数:page_size:分页大小
8 -- 输出参数:total_count:数据总数
9 -- 输出参数:total_page:总页数
10 */
11 CREATE PROCEDURE proc_search_user(IN page_index INT,IN page_size INT, OUT total_count INT, OUT total_page INT)
12 BEGIN
13 DECLARE begin_no INT;
14 SET begin_no = (page_index-1)*page_size;
15
16 -- 分页查询列表
17 SELECT * FROM tb_user
18 WHERE id >= (
19 SELECT id FROM tb_user
20 ORDER BY id ASC
21 LIMIT begin_no,1
22 )
23 ORDER BY id ASC
24 LIMIT page_size;
25
26 -- 计算数据总数
27 SELECT COUNT(1) INTO total_count FROM tb_user;
28
29 -- 计算总页数
30 SET total_page = FLOOR((total_count + page_size - 1) / page_size);
31 END$$
32
33 -- 将结束标志符更改回分号
34 DELIMITER ;
proc_search_user
(3)创建用户信息持久化类(User.java)。
1 package com.pjb.mybatis.po;
2
3 /**
4 * 用户信息的持久化类
5 * @author pan_junbiao
6 **/
7 public class User
8 {
9 private int id; //用户编号
10 private String userName; //用户姓名
11 private String sex; //性别
12
13 //省略getter与setter方法...
14 }
(4)编写SQL映射配置。
1 <!-- 存储过程:分页查询用户列表,并返回数据总数和总页数 -->
2 <select id="proc_search_user" statementType="CALLABLE" parameterType="hashmap" resultType="com.pjb.mybatis.po.User">
3 {CALL proc_search_user(#{page_index,mode=IN,jdbcType=INTEGER},
4 #{page_size,mode=IN,jdbcType=INTEGER},
5 #{total_count,mode=OUT,jdbcType=INTEGER},
6 #{total_page,mode=OUT,jdbcType=INTEGER})}
7 </select>
(5)编写执行方法。

1 /**
2 * 使用MyBatis调用存储过程:分页查询用户列表,并返回数据总数和总页数
3 * @author pan_junbiao
4 */
5 @Test
6 public void procSearchUser()
7 {
8 DataConnection dataConnection = new DataConnection();
9 SqlSession sqlSession = dataConnection.getSqlSession();
10 //封装查询参数
11 Map params = new HashMap();
12 params.put("page_index",2); //输入参数:当前页码
13 params.put("page_size",10); //输入参数:分页大小
14 params.put("total_count",0); //输出参数:数据总数
15 params.put("total_page",0); //输出参数:总页数
16 //调用存储过程
17 List<User> userList = sqlSession.selectList("test.proc_search_user",params);
18 System.out.println("查询第"+ params.get("page_index") +"页的数据,每页共"+params.get("page_size")+"条数据");
19 //遍历用户列表
20 for (User user : userList)
21 {
22 System.out.println("编号:" + user.getId() +" 姓名:" + user.getUserName() + " 性别:" + user.getSex());
23 }
24 //获取输出参数
25 System.out.println("数据总数:" + params.get("total_count"));
26 System.out.println("总页数:" + params.get("total_page"));
27 sqlSession.close();
28 }
procSearchUser
示例2
创建存储过程,实现新增用户信息,并返回自增主键,通过MyBatis调用该存储过程
(1)创建存储过程。

1 -- 将结束标志符更改为$$
2 DELIMITER $$
3
4 /*
5 -- 存储过程:新增用户信息,返回自增主键
6 -- 输入参数:user_name:用户姓名
7 -- 输入参数:sex:性别
8 -- 输出参数:user_id:自增主键
9 */
10 CREATE PROCEDURE proc_add_user(IN user_name VARCHAR(50),IN sex CHAR(2), OUT user_id INT)
11 BEGIN
12 -- 新增用户
13 INSERT INTO tb_user(user_name,sex) VALUE (user_name,sex);
14
15 -- 获取自增主键
16 SELECT LAST_INSERT_ID() INTO user_id;
17 END$$
18
19 -- 将结束标志符更改回分号
20 DELIMITER ;
proc_add_user
(2)编写SQL映射配置。
1 <!-- 存储过程:新增用户信息,返回自增主键 -->
2 <insert id="proc_add_user" statementType="CALLABLE" parameterType="com.pjb.mybatis.po.User">
3 {CALL proc_add_user(#{userName,mode=IN,jdbcType=VARCHAR},
4 #{sex,mode=IN,jdbcType=CHAR},
5 #{id,mode=OUT,jdbcType=INTEGER})}
6 </insert>
(3)编写执行方法。
1 /**
2 * 使用MyBatis调用存储过程:新增用户信息,返回自增主键
3 * @author pan_junbiao
4 */
5 @Test
6 public void procAddUser()
7 {
8 DataConnection dataConnection = new DataConnection();
9 SqlSession sqlSession = dataConnection.getSqlSession();
10 //新增的用户对象
11 User user = new User();
12 user.setUserName("pan_junbiao的博客");
13 user.setSex("男");
14 //调用存储过程执行新增
15 int reuslt = sqlSession.insert("test.proc_add_user",user);
16 sqlSession.commit();
17 //打印结果
18 System.out.println("执行结果:"+reuslt);
19 System.out.println("自增主键:"+user.getId());
20 sqlSession.close();
21 }
执行结果:

其实,新增数据后,获取自增主键是可以使用MyBatis提供的<selectKey>标签,SQL映射配置如下:
1 <!-- 存储过程:新增用户信息,返回自增主键 -->
2 <insert id="proc_add_user" statementType="CALLABLE" parameterType="com.pjb.mybatis.po.User">
3 <selectKey keyProperty="id" order="AFTER" resultType="java.lang.Integer">
4 SELECT LAST_INSERT_ID()
5 </selectKey>
6 {CALL proc_add_user(#{userName,mode=IN,jdbcType=VARCHAR},
7 #{sex,mode=IN,jdbcType=CHAR})}
8 </insert>
但上述示例是为了能让该存储过程拥有一个返回的参数。
MyBatis 调用存储过程
【示例2】
创建存储函数,根据用户编号,获取用户名称,通过MyBatis调用该存储函数。
(1)创建存储函数,根据用户编号,获取用户名称。

1 -- 将结束标志符更改为$$
2 DELIMITER $$
3
4 /*
5 -- 存储函数:根据用户编号,获取用户名称
6 -- 输入参数:in_id:用户编号
7 -- 返回结果:用户名称
8 */
9 CREATE FUNCTION func_get_user_name(in_id INT)
10 RETURNS VARCHAR(50)
11 BEGIN
12 -- 定义返回变量
13 DECLARE out_name VARCHAR(50);
14
15 -- 查询用户信息,获取用户名称
16 SELECT user_name INTO out_name FROM tb_user WHERE id = in_id;
17
18 -- 返回结果
19 RETURN out_name;
20 END$$
21
22 -- 将结束标志符更改回分号
23 DELIMITER ;
func_get_user_name
(2)编写SQL映射配置。
1 <!-- 存储函数:根据用户编号,获取用户名称 -->
2 <select id="func_get_user_name" statementType="CALLABLE" parameterType="hashMap" >
3 {#{userName,mode=OUT,jdbcType=VARCHAR} = CALL func_get_user_name(#{userId,mode=IN,jdbcType=INTEGER})}
4 </select>
(3)编写执行方法。
1 /**
2 * 使用MyBatis调用存储函数:根据用户编号,获取用户名称
3 * @author pan_junbiao
4 */
5 @Test
6 public void funcGetUserName()
7 {
8 DataConnection dataConnection = new DataConnection();
9 SqlSession sqlSession = dataConnection.getSqlSession();
10 //封装参数
11 Map userMap = new HashMap();
12 userMap.put("userName","");
13 userMap.put("userId",8);
14 sqlSession.selectOne("test.func_get_user_name",userMap);
15 System.out.println("用户名称:" + userMap.get("userName"));
16 sqlSession.close();
17 }
【Java】【Mybatis】如何调用存储过程和存储函数的更多相关文章
- day70-oracle 12-Java调用存储过程和存储函数
我们现在调用的是存储过程和存储函数.用CallableSatement调用存储函数和存储过程. RDBMS:关系数据库.使用标准方式调用存储过程.也就是说:在mysql中调用和在oracle中调用的写 ...
- java程序调用存储过程和存储函数
java程序调用存储过程 jdbcUtil.java文件 package cn.itcast.oracle.utils; import java.sql.Connection; import java ...
- java调用存储过程、存储函数
需要用到的接口 接口 CallableStatement JDK文档对改接口的说明: public interface CallableStatement extends PreparedStatem ...
- MySQL-快速入门(8)存储过程、存储函数
1.存储过程 1>创建存储过程:create procedure create procedure sp_name ([in | out | inout] param_name type) [c ...
- Oracle03——游标、异常、存储过程、存储函数、触发器和Java代码访问Oracle对象
作者: kent鹏 转载请注明出处: http://www.cnblogs.com/xieyupeng/p/7476717.html 1.游标(光标)Cursor 在写java程序中有集合的概念,那么 ...
- Java数据库连接--JDBC调用存储过程,事务管理和高级应用
相关链接:Jdbc调用存储过程 一.JDBC常用的API深入详解及存储过程的调用 1.存储过程的介绍 我们常用的操作数据库语言SQL语句在执行的时候要先进行编译,然后执行,而存储过程是在大型数据库系统 ...
- 存储过程,存储函数(Oracle)
存储过程和存储函数 指存储在数据库中供所有用户程序调用的子程序叫存储过程.存储函数. 存储过程和存储函数的区别? 存储函数:可以通过return 语句返回函数值. 存储过程:不能 除此之外我们可以认为 ...
- MySQL数据库之存储过程与存储函数
1 引言 存储过程和存储函数类似于面向对象程序设计语言中的方法,可以简化代码,提高代码的重用性.本文主要介绍如何创建存储过程和存储函数,以及存储过程与函数的使用.修改.删除等操作. 2 存储过程与存储 ...
- 编程开发之--Oracle数据库--存储过程和存储函数(2)
上一小结我们简单介绍了存储过程和存储函数,对存储过程和存储函数有了一个基本的了解,接下来介绍在java程序中如何调用我们创建的存储过程和存储函数 1.在应用程序中调用我们的存储过程 创建一个简单的Ja ...
- oracle存储过程和存储函数
存储过程 1.存储过程简介 下面先来简单介绍一下oracle的存储过程的语法,如下: create or replace procedure Tony_Process ( num in number, ...
随机推荐
- 以EEPROM为例的硬件IIC的使用
目录 参考调试MPU6050与EEPROM的经验,整合了目标内存/寄存器地址是否为16位的情况,合并了单字节与多字节间的操作,添加了返回值与读写超时功能:硬件IIC的7位从机地址查询方式读写参考代码 ...
- centos7安装php8
原文: http://www.manongjc.com/detail/25-qpyxndyogppmfdf.html 前言 centos7默认源的php版本只有5.4,版本太老,而mediawiki需 ...
- Windows 11 调整工具 TweakNow WinSecret for Windows 11 3.2.0 中文汉化版
Windows 11 调整工具 TweakNow WinSecret for Windows 11 中文版由大眼仔旭(www.dayanzai.me)发布.适用于 Windows 11 的 Tweak ...
- Linux 第六节( 磁盘系统,挂载,分区,格式化)
/dev/st0 磁带机 /dev/lp 打印机 /dev/cdrom 光驱 /dev/sd scsi接口硬盘 sata接口硬盘 U盘(sda,sdb,sdc 分别对应 ...
- git多分支-git远程仓库-ssh方式连接远程仓库-协同开发-冲突解决-线上分支合并-远程仓库回滚
目录 git多分支-git远程仓库-ssh方式连接远程仓库-协同开发-冲突解决-线上分支合并-远程仓库回滚 昨日内容回顾 今日内容概要 今日内容详细 1 git多分支 2 git远程仓库 3 ssh方 ...
- 【转载】Adobe Acrobat XI Pro闪退原因及解决办法
https://www.cnblogs.com/zohoo/p/12704689.html https://www.cnblogs.com/zohoo/p/12704689.html
- tidb 杂记
tidb_biuil_stats_concurrency 执行analyze table时会分成多个小任务,可以同时执行的任务数量.tidb_distsql_scan_concurrency 在执行分 ...
- winform高dpi问题探索
winform的高dpi适应问题由来已久,属于一个历史遗留问题.主要是由于winform对不同尺寸.不同分辨率的屏幕适配不足造成的.接下来我们简单说一下如何解决,最后我们探索一下解决此问题的原理. ...
- 浅谈组件二封-vue
目录 组件二封不是换一种写法 组件二封应当具备哪些条件 我认为的二封应当有哪些作用 二封的好处 先来一个列表页demo来看看效果(Vue2) 本文仅仅针对vue系列做探讨, 项目倾向于大量增删改查的后 ...
- Qt 一键部署脚本
echo "begin deploying..." echo Setting up environment for Qt usage... set PATH=C:\Qt\Qt5.8 ...