存储过程之七—java代码调用

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: 一、简介   jdbc调用存储过程与调用一般的sql语句有些差别。jdbc调用一般的sql语句的时候,返回的是结果集,或是sql语句后是否执行成功的标记量。而存储过程返回的可以是结果集,输出参数、返回状态和受影响行数。

一、简介

  jdbc调用存储过程与调用一般的sql语句有些差别。jdbc调用一般的sql语句的时候,返回的是结果集,或是sql语句后是否执行成功的标记量。而存储过程返回的可以是结果集,输出参数、返回状态和受影响行数。而mysql存储过程不支持return,所以只有剩下的三种返回方式。

  调用的过程一般如下:

  1、创建连接 Connection conn = DriverManager.getConnection(url, user, password);

  2、创建CallableStatement CallableStatement statement = conn.prepareCall(sql);

  3、设置参数

    statement.setInt(1, id);
    statement.registerOutParameter(2, Types.VARCHAR);
    statement.registerOutParameter(3, Types.INTEGER);
    statement.registerOutParameter(4, Types.VARCHAR);

   4、执行

    statement.execute(); 或 statement.executeUpdate();

  5、获取返回

    int age = statement.getInt(3);或ResultSet resultSet = statement.executeQuery();

二、实例

  以下存储过程表结构如下:

DROP TABLE IF EXISTS `person`;
CREATE TABLE `person` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `username` varchar(255) DEFAULT NULL,
    `age` int(11) DEFAULT NULL, 
  `password` varchar(255) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=11 DEFAULT CHARSET=utf8;

-- ----------------------------
-- Records of person
-- ----------------------------
INSERT INTO `person` VALUES ('1', 'lisi', '21', 'li123456');
INSERT INTO `person` VALUES ('2', 'zhangsan', '23', 'zhang123456');

   1、增加记录(输入输出参数的调用)

     存储过程如下:

DROP PROCEDURE IF EXISTS proc_person_insert;
CREATE PROCEDURE proc_person_insert(
    IN uusername VARCHAR(255),
    IN uage INT(11),
    IN upassword VARCHAR(255),
    OUT flag INT(11)
)
BEGIN
    START TRANSACTION;
        INSERT INTO person (username, age, password) values (uusername, uage, upassword);
        SET flag = ROW_COUNT(); -- 保存sql语句操作后影响的行数
    COMMIT;    
END;

    调用代码如下:

 1     public static void insert(String username, int age, String upassword) {
 2         String driver = "com.mysql.jdbc.Driver";
 3         String url = "jdbc:mysql://127.0.0.1:3306/test";
 4         String user = "root";
 5         String password = "";
 6         try {
 7             Class.forName(driver);
 8             Connection conn = DriverManager.getConnection(url, user, password);
 9             String sql = "{call proc_person_insert( ?, ?, ?, ? )}";
10             CallableStatement statement = conn.prepareCall(sql);
11             statement.setString(1, username); //设置输入参数username
12             statement.setInt(2, age); //设置输入参数age
13             statement.setString(3, upassword);//设置输入参数password
14             statement.registerOutParameter(4, Types.INTEGER);//设置输出参数username
15             statement.executeUpdate();
16             int flag = statement.getInt(4);//获取输出参数
17             System.out.println(flag);// 1添加成功,0添加失败
18         } catch (ClassNotFoundException e) {
19             e.printStackTrace();
20         } catch (SQLException e) {
21             e.printStackTrace();
22         }
23     }

    对于输出参数需要调用statement.registerOutParameter(4, Types.INTEGER)进行注册,执行之后通过statement.getInt()方法来获取。

   2、修改记录(输入输出参数的调用)

    存储过程如下:

DROP PROCEDURE IF EXISTS proc_person_update;
CREATE PROCEDURE proc_person_update(
    IN uid INT(11),
    IN uusername VARCHAR(255),
    IN uage INT(11),
    IN upassword VARCHAR(255),
    OUT flag INT(11)
)
BEGIN
    START TRANSACTION;
        UPDATE person SET username = uusername, age = uage, password = upassword WHERE id = uid;
        SET flag = ROW_COUNT(); -- 保存sql语句操作后影响的行数
    COMMIT;
END;

    调用代码如下:

 1 public static void update(int id, String username, int age, String upassword) {
 2         String driver = "com.mysql.jdbc.Driver";
 3         String url = "jdbc:mysql://127.0.0.1:3306/test";
 4         String user = "root";
 5         String password = "";
 6         try {
 7             Class.forName(driver);
 8             Connection conn = DriverManager.getConnection(url, user, password);
 9             String sql = "{call proc_person_update( ?, ?, ?, ?, ? )}";
10             CallableStatement statement = conn.prepareCall(sql);
11             statement.setInt(1, id);
12             statement.setString(2, username);
13             statement.setInt(3, age); 
14             statement.setString(4, upassword); 
15             statement.registerOutParameter(5, Types.INTEGER);
16             statement.executeUpdate();
17             int flag = statement.getInt(5);
18             System.out.println(flag);// 1添加成功,0添加失败
19         } catch (ClassNotFoundException e) {
20             e.printStackTrace();
21         } catch (SQLException e) {
22             e.printStackTrace();
23         }
24     }

  3、删除记录(使用statement.getUpdateCount()获取影响行数)

    存储过程如下:

DROP PROCEDURE IF EXISTS proc_person_del;
CREATE PROCEDURE proc_person_del(
    IN pid INT(11) 
)
BEGIN 
    DELETE FROM person WHERE id = pid;  
END;

    调用代码如下:

 1 public  static void delete(int id){
 2         String driver = "com.mysql.jdbc.Driver";
 3         String url = "jdbc:mysql://127.0.0.1:3306/test";
 4         String user = "root";
 5         String password = "";
 6         try {
 7             Class.forName(driver);
 8             Connection conn = DriverManager.getConnection(url, user, password);
 9             String sql = "{call proc_person_del( ?)}";
10             CallableStatement statement = conn.prepareCall(sql); 
11             statement.setInt(1, id);
12             statement.executeUpdate();  
13             int flag = statement.getUpdateCount();//使用该语句的时候存储过程DELETE语句外不能出现 START TRANSACTION;COMMINT;
14             System.out.println(flag);
15         } catch (ClassNotFoundException e) { 
16             e.printStackTrace();
17         } catch (SQLException e) { 
18             e.printStackTrace();
19         }
20     }

   4、查询一条记录(输入输出)

     存储过程如下:

DROP PROCEDURE IF EXISTS proc_person_find;
CREATE PROCEDURE proc_person_find(
    IN pid INT(11),
    OUT pusername VARCHAR(255),
    OUT page INT(11),
    OUT ppassword VARCHAR(255)
)
BEGIN
    SELECT username, age, password INTO pusername, page, ppassword FROM person WHERE id = pid;
END;

    使用在SELECT语句中使用INTO给输出参数赋值

    调用代码如下:

 1     public  static void find(int id){
 2         String driver = "com.mysql.jdbc.Driver";
 3         String url = "jdbc:mysql://127.0.0.1:3306/test";
 4         String user = "root";
 5         String password = "";
 6         try {
 7             Class.forName(driver);
 8             Connection conn = DriverManager.getConnection(url, user, password);
 9             String sql = "{call proc_person_find( ?, ?, ?, ?)}";
10             CallableStatement statement = conn.prepareCall(sql);
11             statement.setInt(1, id);  
12             statement.registerOutParameter(2, Types.VARCHAR);
13             statement.registerOutParameter(3, Types.INTEGER);
14             statement.registerOutParameter(4, Types.VARCHAR);
15             statement.execute();
16             String username = statement.getString(2);
17             int age = statement.getInt(3);
18             String ppassword = statement.getString(4);
19             System.out.println("id:" + id);
20             System.out.println("username:" + username);
21             System.out.println("age:" + age);
22             System.out.println("ppassword:" + ppassword);
23         } catch (ClassNotFoundException e) {
24             e.printStackTrace();
25         } catch (SQLException e) {
26             e.printStackTrace();
27         }
28     }

  5、查询一个集合(返回一个集合)

    存储过程如下:

DROP PROCEDURE IF EXISTS proc_person_findAll;
CREATE PROCEDURE proc_person_findAll( )
BEGIN
    SELECT id, username, age, password  FROM person;
END;

    调用代码如下:

 1 public  static void findAll(){
 2         String driver = "com.mysql.jdbc.Driver";
 3         String url = "jdbc:mysql://127.0.0.1:3306/test";
 4         String user = "root";
 5         String password = "";
 6         try {
 7             Class.forName(driver);
 8             Connection conn = DriverManager.getConnection(url, user, password);
 9             String sql = "{call proc_person_findAll()}";
10             CallableStatement statement = conn.prepareCall(sql); 
11             ResultSet resultSet = statement.executeQuery();
12             while(resultSet.next()){
13                 int id = resultSet.getInt("id");
14                 String username =resultSet.getString("username");
15                 int age = resultSet.getInt("age");
16                 String ppassword = resultSet.getString("password");
17                 System.out.println(id + " " + username + " " + age + " " + ppassword);
18             } 
19         } catch (ClassNotFoundException e) { 
20             e.printStackTrace();
21         } catch (SQLException e) {
22              e.printStackTrace();
23         }
24     }

     通过statement.executeQuery()方法返回已经ResultSet,再对ResultSet进行遍历。

  关于存储过程存在into参数的时候,java代码中在设置输入参数之后,还需要注册输出参数,调用之后通过statement.get..方法拿到输出参数的值。如下:

    String sql = "{call proc_person_del( ? )}";
    CallableStatement statement = conn.prepareCall(sql);
    statement.setInt(1, id); 
    statement.registerOutParameter(1, Types.INTEGER);  
    // 注意此次注册out 的index 和上面的in 参数index 相同  
    statement.execute(); 
    int flag = statement.getInt(1); 
相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助     相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
相关文章
|
2月前
|
Java
在 Java 中捕获和处理自定义异常的代码示例
本文提供了一个 Java 代码示例,展示了如何捕获和处理自定义异常。通过创建自定义异常类并使用 try-catch 语句,可以更灵活地处理程序中的错误情况。
80 1
|
2月前
|
Java
在Java中实现接口的具体代码示例
可以根据具体的需求,创建更多的类来实现这个接口,以满足不同形状的计算需求。希望这个示例对你理解在 Java 中如何实现接口有所帮助。
92 38
|
13天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
37 3
|
2月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
57 24
|
21天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
56 2
|
1月前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
82 5
|
1月前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
57 5
|
2月前
|
Java API 开发者
Java中的Lambda表达式:简洁代码的利器####
本文探讨了Java中Lambda表达式的概念、用途及其在简化代码和提高开发效率方面的显著作用。通过具体实例,展示了Lambda表达式如何在Java 8及更高版本中替代传统的匿名内部类,使代码更加简洁易读。文章还简要介绍了Lambda表达式的语法和常见用法,帮助开发者更好地理解和应用这一强大的工具。 ####
|
2月前
|
XML 安全 Java
Java反射机制:解锁代码的无限可能
Java 反射(Reflection)是Java 的特征之一,它允许程序在运行时动态地访问和操作类的信息,包括类的属性、方法和构造函数。 反射机制能够使程序具备更大的灵活性和扩展性
50 5
Java反射机制:解锁代码的无限可能
|
2月前
|
Java API Maven
商汤人像如何对接?Java代码如何写?
商汤人像如何对接?Java代码如何写?
51 5