MyBatis如何调用存储过程与存储函数
作者:pan_junbiao 发布时间:2023-06-14 16:10:51
标签:MyBatis,存储过程,存储函数
1、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>
【示例】创建存储过程,实现分页查询用户列表,并返回数据总数和总页数,通过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)创建存储过程,实现分页查询用户列表,并返回数据总数和总页数。
-- 将结束标志符更改为$$
DELIMITER $$
/*
-- 存储过程:分页查询用户列表,并返回数据总数和总页数
-- 输入参数:page_index:当前页码
-- 输入参数:page_size:分页大小
-- 输出参数:total_count:数据总数
-- 输出参数:total_page:总页数
*/
CREATE PROCEDURE proc_search_user(IN page_index INT,IN page_size INT, OUT total_count INT, OUT total_page INT)
BEGIN
DECLARE begin_no INT;
SET begin_no = (page_index-1)*page_size;
-- 分页查询列表
SELECT * FROM tb_user
WHERE id >= (
SELECT id FROM tb_user
ORDER BY id ASC
LIMIT begin_no,1
)
ORDER BY id ASC
LIMIT page_size;
-- 计算数据总数
SELECT COUNT(1) INTO total_count FROM tb_user;
-- 计算总页数
SET total_page = FLOOR((total_count + page_size - 1) / page_size);
END$$
-- 将结束标志符更改回分号
DELIMITER ;
(3)创建用户信息持久化类(User.java)。
package com.pjb.mybatis.po;
/**
* 用户信息的持久化类
* @author pan_junbiao
**/
public class User
{
private int id; //用户编号
private String userName; //用户姓名
private String sex; //性别
//省略getter与setter方法...
}
(4)编写SQL映射配置。
<!-- 存储过程:分页查询用户列表,并返回数据总数和总页数 -->
<select id="proc_search_user" statementType="CALLABLE" parameterType="hashmap" resultType="com.pjb.mybatis.po.User">
{CALL proc_search_user(#{page_index,mode=IN,jdbcType=INTEGER},
#{page_size,mode=IN,jdbcType=INTEGER},
#{total_count,mode=OUT,jdbcType=INTEGER},
#{total_page,mode=OUT,jdbcType=INTEGER})}
</select>
(5)编写执行方法。
/**
* 使用MyBatis调用存储过程:分页查询用户列表,并返回数据总数和总页数
* @author pan_junbiao
*/
@Test
public void procSearchUser()
{
DataConnection dataConnection = new DataConnection();
SqlSession sqlSession = dataConnection.getSqlSession();
//封装查询参数
Map params = new HashMap();
params.put("page_index",2); //输入参数:当前页码
params.put("page_size",10); //输入参数:分页大小
params.put("total_count",0); //输出参数:数据总数
params.put("total_page",0); //输出参数:总页数
//调用存储过程
List<User> userList = sqlSession.selectList("test.proc_search_user",params);
System.out.println("查询第"+ params.get("page_index") +"页的数据,每页共"+params.get("page_size")+"条数据");
//遍历用户列表
for (User user : userList)
{
System.out.println("编号:" + user.getId() +" 姓名:" + user.getUserName() + " 性别:" + user.getSex());
}
//获取输出参数
System.out.println("数据总数:" + params.get("total_count"));
System.out.println("总页数:" + params.get("total_page"));
sqlSession.close();
}
执行结果:
【示例】创建存储过程,实现新增用户信息,并返回自增主键,通过MyBatis调用该存储过程。
(1)创建存储过程。
-- 将结束标志符更改为$$
DELIMITER $$
/*
-- 存储过程:新增用户信息,返回自增主键
-- 输入参数:user_name:用户姓名
-- 输入参数:sex:性别
-- 输出参数:user_id:自增主键
*/
CREATE PROCEDURE proc_add_user(IN user_name VARCHAR(50),IN sex CHAR(2), OUT user_id INT)
BEGIN
-- 新增用户
INSERT INTO tb_user(user_name,sex) VALUE (user_name,sex);
-- 获取自增主键
SELECT LAST_INSERT_ID() INTO user_id;
END$$
-- 将结束标志符更改回分号
DELIMITER ;
(2)编写SQL映射配置。
<!-- 存储过程:新增用户信息,返回自增主键 -->
<insert id="proc_add_user" statementType="CALLABLE" parameterType="com.pjb.mybatis.po.User">
{CALL proc_add_user(#{userName,mode=IN,jdbcType=VARCHAR},
#{sex,mode=IN,jdbcType=CHAR},
#{id,mode=OUT,jdbcType=INTEGER})}
</insert>
(3)编写执行方法。
/**
* 使用MyBatis调用存储过程:新增用户信息,返回自增主键
* @author pan_junbiao
*/
@Test
public void procAddUser()
{
DataConnection dataConnection = new DataConnection();
SqlSession sqlSession = dataConnection.getSqlSession();
//新增的用户对象
User user = new User();
user.setUserName("pan_junbiao的博客");
user.setSex("男");
//调用存储过程执行新增
int reuslt = sqlSession.insert("test.proc_add_user",user);
sqlSession.commit();
//打印结果
System.out.println("执行结果:"+reuslt);
System.out.println("自增主键:"+user.getId());
sqlSession.close();
}
执行结果:
其实,新增数据后,获取自增主键是可以使用MyBatis提供的<selectKey>标签,SQL映射配置如下:
<!-- 存储过程:新增用户信息,返回自增主键 -->
<insert id="proc_add_user" statementType="CALLABLE" parameterType="com.pjb.mybatis.po.User">
<selectKey keyProperty="id" order="AFTER" resultType="java.lang.Integer">
SELECT LAST_INSERT_ID()
</selectKey>
{CALL proc_add_user(#{userName,mode=IN,jdbcType=VARCHAR},
#{sex,mode=IN,jdbcType=CHAR})}
</insert>
但上述示例是为了能让该存储过程拥有一个返回的参数。
2、MyBatis调用存储函数
【示例】创建存储函数,根据用户编号,获取用户名称,通过MyBatis调用该存储函数。
(1)创建存储函数,根据用户编号,获取用户名称。
-- 将结束标志符更改为$$
DELIMITER $$
/*
-- 存储函数:根据用户编号,获取用户名称
-- 输入参数:in_id:用户编号
-- 返回结果:用户名称
*/
CREATE FUNCTION func_get_user_name(in_id INT)
RETURNS VARCHAR(50)
BEGIN
-- 定义返回变量
DECLARE out_name VARCHAR(50);
-- 查询用户信息,获取用户名称
SELECT user_name INTO out_name FROM tb_user WHERE id = in_id;
-- 返回结果
RETURN out_name;
END$$
-- 将结束标志符更改回分号
DELIMITER ;
(2)编写SQL映射配置。
<!-- 存储函数:根据用户编号,获取用户名称 -->
<select id="func_get_user_name" statementType="CALLABLE" parameterType="hashMap" >
{#{userName,mode=OUT,jdbcType=VARCHAR} = CALL func_get_user_name(#{userId,mode=IN,jdbcType=INTEGER})}
</select>
(3)编写执行方法。
/**
* 使用MyBatis调用存储函数:根据用户编号,获取用户名称
* @author pan_junbiao
*/
@Test
public void funcGetUserName()
{
DataConnection dataConnection = new DataConnection();
SqlSession sqlSession = dataConnection.getSqlSession();
//封装参数
Map userMap = new HashMap();
userMap.put("userName","");
userMap.put("userId",8);
sqlSession.selectOne("test.func_get_user_name",userMap);
System.out.println("用户名称:" + userMap.get("userName"));
sqlSession.close();
}
执行结果:
来源:https://blog.csdn.net/pan_junbiao/article/details/103575982


猜你喜欢
- 一般而言,Android 应用在请求数据时都是以 Get 或 Post 等方式向远程服务器发起请求,那你有没有想过其实我们也可以在 Andr
- 项目结构:pom.xml文件: <parent>
- 本文实例讲述了C#实现的xml操作类,分享给大家供大家参考,具体如下:using System;using System.Data;usin
- using System;using System.Collections.Generic;using System.IO;using Sy
- 文件复制和文件上传最近在看文件和IO流相关的东西,写了一些代码,发现这个有很多很有趣的地方。特别是对 File 和 IO 流的使用之后,我对
- 需求:当时间在凌晨0点至0点5分之间程序不执行。也就是实现判断当前时间点是否在
- 一、异步模型的基本概述异步编程的核心是 Task 和 Task<T> 对象,这两个对象对异步操作建模。 它们受关键字 async
- 设计模式要进行共性与可变性的分析,对共性进行抽象,同时对可变性进行封装,没有完美的设计模式,作为一名开发者要懂得取舍,触类旁通,开发出高内聚
- C++中的动态数组(Dynamic Array)是指动态分配的、可以根据需求动态增长占用内存的数组。为了实现一个动态数组类的封装,我们需要考
- 最近正好也没什么可忙的,就回过头来鼓捣过去的知识点,到Servlet部分时,以前学习的时候硬是把从上到下的继承关系和接口实现记得乱七八糟。这
- 今天遇到了一个比较有意思的问题,从服务器上封装好的java.sql.timestamp对象返回到本地客户端程序后与数据库中的时间相差了整整1
- java 出现Zipexception 异常的解决办法1 异常描述在从 SVN 检出项目并配置完成后,启动 Tomcat 服务器,报出如下错
- 本文实例为大家分享Android通过Movie展示Gif格式图片的相关代码,供大家参考,具体内容如下public class CommonG
- 运行在TCP之上常见的网络应用协议有比如HTTP、FTP、SMTP、POP3、IMAP。TCP是TCP/IP体系中最重要的传输协议,它提供全
- 前言工作中是否遇到这样的场景?1、需要异步线程执行,而且需要获取到线程执行返回的结果。2、如果执行过程异常,可以按照自定义方式消费异常信息。
- 背景:在平时的开发中,我们时常会遇到下列场景公司的组织架构的数据存储与展示文件夹层级的数据存储与展示评论系统中,父评论与诸多子评论的数据存储
- 引言java 7提供了另外一个很有用的线程池框架,Fork/Join框架理论Fork/Join框架主要有以下两个类组成. * ForkJoi
- 在前面几篇文章中,我们详细介绍了Androi
- 异常方法//返回此可抛出对象的详细信息消息字符串public String getMessage() //将此可抛发对象及其回溯到标准错误流
- RTF文档即富文本格式(Rich Text Format)的文档。我们在处理文件时,遇到需要对文档格式进行转换时,可以将RTF转为其他格式,