SQL进阶知识:五、存储过程和函数

今天介绍下关于存储过程和函数的详细介绍,并结合MySQL数据库提供实际例子。

在MySQL中,存储过程(Stored Procedures)和函数(Functions)是数据库编程的重要组成部分,它们可以封装SQL语句,提高代码的复用性和安全性,同时减少网络传输。以下是关于存储过程和函数的详细介绍,以及基于MySQL的实际例子。


一、存储过程(Stored Procedures)

1. 存储过程的基本概念

存储过程是一组SQL语句的集合,封装在一个名称下,可以带参数调用。存储过程可以包含复杂的逻辑,支持条件语句、循环语句和变量声明。

2. 存储过程的优点

  • 代码复用:存储过程可以被多次调用,避免重复编写SQL语句。
  • 性能提升:存储过程在服务器端执行,减少了网络传输。
  • 安全性:通过存储过程可以限制用户直接访问表,提高数据的安全性。
  • 事务管理:存储过程可以包含事务控制语句,确保操作的原子性。

3. 创建存储过程

语法:

CREATE PROCEDURE procedure_name (IN|OUT|INOUT param1 type, ...)
BEGIN-- SQL语句
END;
  • IN参数:输入参数,用于传递值到存储过程。
  • OUT参数:输出参数,用于返回值。
  • INOUT参数:既可以输入也可以输出。

4. 调用存储过程

CALL procedure_name(param1, param2, ...);

5. 删除存储过程

DROP PROCEDURE procedure_name;

二、函数(Functions)

1. 函数的基本概念

函数是一组SQL语句的集合,封装在一个名称下,可以带参数调用,并返回一个值。函数通常用于计算和返回单个值。

2. 函数的优点

  • 代码复用:函数可以被多次调用,避免重复编写SQL语句。
  • 性能提升:函数在服务器端执行,减少了网络传输。
  • 逻辑封装:函数可以封装复杂的计算逻辑。

3. 创建函数

语法:

CREATE FUNCTION function_name (param1 type, ...)
RETURNS return_type
BEGIN-- SQL语句RETURN value;
END;

4. 调用函数

SELECT function_name(param1, param2, ...);

5. 删除函数

DROP FUNCTION function_name;

三、实际例子

示例1:存储过程(插入数据并返回插入的行数)

场景:插入一条用户数据,并返回插入的行数
-- 创建存储过程
DELIMITER //CREATE PROCEDURE InsertUser(IN username VARCHAR(50), IN email VARCHAR(100), OUT rows_affected INT)
BEGININSERT INTO users (username, email) VALUES (username, email);SET rows_affected = ROW_COUNT();
END //DELIMITER ;-- 调用存储过程
CALL InsertUser('Alice', 'alice@example.com', @rows_affected);-- 查看返回的行数
SELECT @rows_affected;

解释

  • 创建了一个存储过程InsertUser,接收用户名和邮箱作为输入参数,返回插入的行数。
  • 使用ROW_COUNT()函数获取插入的行数。
  • 调用存储过程时,使用OUT参数rows_affected接收返回值。

示例2:存储过程(更新数据并返回受影响的行数)

场景:更新用户邮箱,并返回受影响的行数
-- 创建存储过程
DELIMITER //CREATE PROCEDURE UpdateUserEmail(IN user_id INT, IN new_email VARCHAR(100), OUT rows_affected INT)
BEGINUPDATE users SET email = new_email WHERE id = user_id;SET rows_affected = ROW_COUNT();
END //DELIMITER ;-- 调用存储过程
CALL UpdateUserEmail(1, 'new_email@example.com', @rows_affected);-- 查看返回的行数
SELECT @rows_affected;

解释

  • 创建了一个存储过程UpdateUserEmail,接收用户ID和新邮箱作为输入参数,返回受影响的行数。
  • 使用ROW_COUNT()函数获取受影响的行数。
  • 调用存储过程时,使用OUT参数rows_affected接收返回值。

示例3:存储过程(事务控制)

场景:从用户A的账户转账到用户B的账户
-- 创建存储过程
DELIMITER //CREATE PROCEDURE TransferMoney(IN from_id INT, IN to_id INT, IN amount DECIMAL(10, 2))
BEGINDECLARE from_balance DECIMAL(10, 2);DECLARE to_balance DECIMAL(10, 2);-- 开始事务START TRANSACTION;-- 检查用户A的余额是否足够SELECT balance INTO from_balance FROM accounts WHERE id = from_id FOR UPDATE;IF from_balance < amount THENSIGNAL SQLSTATE '45000' SET MESSAGE_TEXT = 'Insufficient balance';END IF;-- 扣款UPDATE accounts SET balance = balance - amount WHERE id = from_id;-- 加款UPDATE accounts SET balance = balance + amount WHERE id = to_id;-- 提交事务COMMIT;
END //DELIMITER ;-- 调用存储过程
CALL TransferMoney(1, 2, 100.00);

解释

  • 创建了一个存储过程TransferMoney,接收转账的用户ID、收款的用户ID和转账金额。
  • 使用START TRANSACTIONCOMMIT管理事务。
  • 使用FOR UPDATE锁定用户A的账户,防止并发问题。
  • 如果用户A的余额不足,抛出异常。

示例4:函数(计算用户年龄)

场景:根据用户的出生日期计算年龄
-- 创建函数
DELIMITER //CREATE FUNCTION CalculateAge(birthdate DATE)
RETURNS INT
BEGINDECLARE age INT;SET age = TIMESTAMPDIFF(YEAR, birthdate, CURDATE());RETURN age;
END //DELIMITER ;-- 调用函数
SELECT id, name, birthdate, CalculateAge(birthdate) AS age FROM users;

解释

  • 创建了一个函数CalculateAge,接收出生日期作为参数,返回年龄。
  • 使用TIMESTAMPDIFF函数计算当前日期与出生日期之间的年数。
  • 调用函数时,可以直接在SELECT语句中使用。

示例5:函数(计算订单总价)

场景:根据订单中的商品数量和单价计算总价
-- 创建函数
DELIMITER //CREATE FUNCTION CalculateOrderTotal(quantity INT, unit_price DECIMAL(10, 2))
RETURNS DECIMAL(10, 2)
BEGINDECLARE total DECIMAL(10, 2);SET total = quantity * unit_price;RETURN total;
END //DELIMITER ;-- 调用函数
SELECT order_id, quantity, unit_price, CalculateOrderTotal(quantity, unit_price) AS total_price
FROM order_items;

解释

  • 创建了一个函数CalculateOrderTotal,接收商品数量和单价作为参数,返回总价。
  • 调用函数时,可以直接在SELECT语句中使用。

四、总结

存储过程和函数是MySQL中强大的工具,可以封装复杂的逻辑,提高代码的复用性和安全性。存储过程支持事务控制和多种参数类型,适用于复杂的数据操作。函数则专注于计算和返回单个值,适用于简单的逻辑封装。通过合理使用存储过程和函数,可以显著提升数据库的可维护性和性能。

以上就是基于Mysql,有关查询相关的进阶知识,希望对你有所帮助~
后续会连续发布多篇SQL进阶相关内容;
期待你的关注,学习更多知识;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/80147.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CONDA:用于 Co-Salient 目标检测的压缩深度关联学习(总结)

摘要 一 介绍 二 有关工作 三 提出的方法 图2&#xff1a;我们的凝聚式深度关联&#xff08;CONDA&#xff09;模型的整体流程图。具体来说&#xff0c;凝聚式深度关联&#xff08;CONDA&#xff09;模型首先利用图像特征来计算超关联。然后&#xff0c;全像素超关联由对应诱…

node.js 实战——(path模块 知识点学习)

path 模块 提供了操作路径的功能 说明path. resolve拼接规范的绝对路径path. sep获取操作系统的路径分隔符path. parse解析路径并返回对象path. basename获取路径的基础名称path. dirname获取路径的目录名path. extname获得路径的扩展名 resolve 拼接规范的绝对路径 const…

Kimi做内容社区,剑指小红书?

原创科技新知AI新科技组作者丨樱木编辑丨江蓠 主编丨九黎 对于当前融资形势并不明朗的大模型六小龙来说&#xff0c;该如何生存下去&#xff0c;似乎成了各家急需解决的问题。 根据PitchBook数据&#xff0c;今年一季度风险投资机构在中国AI领域共完成144笔交易&#xff0c;投…

opencv--图像滤波

图像滤波 含义 方法 噪声是怎么产生的 线性滤波 概念 利用窗口对图像中的像素进行加权求和的滤波方式。 图像来源于小虎教程。 图像的滤波是二维滤波的过程。 滤波器窗口&#xff1a; 滤波器窗口&#xff08;也称为卷积核或模板&#xff09;是一个小的矩阵&#xff08;通常为…

Java 实现SpringContextUtils工具类,手动获取Bean

SpringContextUtils 工具类实现 下面是一个完整的 Spring 上下文工具类实现&#xff0c;用于从 Spring 容器中获取 Bean。这个工具类考虑了线程安全、性能优化和易用性&#xff0c;并提供了多种获取 Bean 的方式。 完整实现代码 import org.springframework.beans.BeansExce…

基于 Vue 2 开发的分页卡片列表组件(带懒加载和点击事件)

功能目标&#xff1a; CardList.vue 中支持分页&#xff0c;每页显示指定数量的卡片。添加“加载中”动画。支持懒加载&#xff1a;滚动到底部自动加载下一页。点击卡片的事件逻辑由 Card.vue 内部发出&#xff0c;并由 CardList 向上传递。 主页面文件 Home.vue <templat…

【数据结构和算法】6. 哈希表

本文根据 数据结构和算法入门 视频记录 文章目录 1. 哈希表的概念1.1 哈希表的实现方式1.2 哈希函数&#xff08;Hash Function&#xff09;1.3 哈希表支持的操作 2. Java实现 在前几章的学习中&#xff0c;我们已经了解了数组和链表的基本特性&#xff0c;不管是数组还是链表…

【python】如何将文件夹及其子文件夹下的所有word文件汇总导出到一个excel文件里?

根据你的需求,这里提供一套完整的Python解决方案,支持递归遍历子文件夹、提取Word文档内容(段落+表格),并整合到Excel中。以下是代码实现及详细说明: 一个单元格一个word的全部内容 完整代码 # -*- coding: utf-8 -*- import os from docx import Document import pand…

leetcode-位运算

位运算 371. 两整数之和 题目 给你两个整数 a 和 b &#xff0c;不使用 运算符 和 - &#xff0c;计算并返回两整数之和。 示例 1&#xff1a; 输入&#xff1a; a 1, b 2 输出&#xff1a; 3 示例 2&#xff1a; 输入&#xff1a; a 2, b 3 输出&#xff1a; 5 提示&am…

飞帆控件:在编辑模式下额外加载的库

飞帆是一个自由的控件设计平台。在飞帆中&#xff0c;我们可以很方便地创建基于 Vue 2 组件的控件&#xff0c;并使用控件来搭建网页。 他山之石&#xff0c;可以攻玉。在创建控件中&#xff0c;使用 js 、css 依赖库能让我们的控件更强大。 有些时候&#xff0c;在编辑模式下…

GPLT-2025年第十届团体程序设计天梯赛总决赛题解(共计266分)

今天偶然发现天梯赛的代码还保存着&#xff0c;于是决定写下这篇题解&#xff0c;也算是复盘一下了 L1本来是打算写的稳妥点&#xff0c;最后在L1-6又想省时间&#xff0c;又忘记了insert&#xff0c;replace这些方法怎么用&#xff0c;也不想花时间写一个文件测试&#xff0c…

编码转换器

大批量转换编码 可以将整个工程文件夹从GB18030转为UTF-8 使用Qt C制作 项目背景 比较老的工程&#xff0c;尤其是keil嵌入式的工程&#xff0c;其文本文件&#xff08;.c、.cpp、.h、.txt、……&#xff09;编码为gb2312&#xff0c;这为移植维护等带来了不便。现在uit-8用…

STL 核心模块

很好&#xff01;你想深入 STL&#xff08;Standard Template Library&#xff09;和容器算法&#xff0c;是学习 C 非常关键的一步。下面我给你整理一份STL 容器 算法的入门指南&#xff0c;适合从零起步掌握这部分内容。 &#x1f31f; 一、STL 核心模块 STL 分为三大块&am…

2024沈阳区域赛,D - Dot Product Game

题目链接 树状数组求逆序对 #include<bits/stdc.h> using namespace std; using lllong long; typedef pair<int,int>PII; typedef priority_queue<int> upq; typedef priority_queue<int,vector<int>,greater<int>> dpq; const int M99…

简易博客点赞系统实现

简易博客点赞系统 好久没写 Java 了&#xff0c;整个简单的项目进行康复训练。 基于 Spring Boot SSM MySQL Mybatis-Plus Knife4j Swagger 的一个简易博客点赞系统 开源地址&#xff1a;https://github.com/FangMoyu/simple-thumb 功能 登录获取当前登录用户获取博客…

一个既简单又诡异的问题

public class DaYaoGuai {static String s;public static void main(String[] args) {Thread t1 new Thread(){Overridepublic void run() {try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}s"深圳";}};t1.start();Thre…

使用docker在manjaro linux系统上运行windows和ubuntu

因为最近项目必须要使用指定版本的solidworks和maxwell&#xff08;都只能在win系统上使用&#xff09;, 且目前的ubuntu容器是没有桌面的&#xff0c;导致我运行不了一些带图形的ros2功能。无奈之下&#xff0c;决定使用docker-compose写一下配置文件&#xff0c;彻底解决问题…

Elasticsearch中的_source字段讲解

_source 在 Elasticsearch 查询中用于限制返回的字段,类似于 SQL 中的 SELECT 指定列。 代码示例: esSearchResults = es_service.search_documents({"query": {"terms": {"file_id":

【论文阅读20】-CNN-Attention-BiGRU-滑坡预测(2025-03)

这篇论文主要探讨了基于深度学习的滑坡位移预测模型&#xff0c;结合了MT-InSAR&#xff08;多时相合成孔径雷达干涉测量&#xff09;观测数据&#xff0c;提出了一种具有可解释性的滑坡位移预测方法。 [1] Zhou C, Ye M, Xia Z, et al. An interpretable attention-based deep…

C++ 的 IO 流

&#x1f4ac; &#xff1a;如果你在阅读过程中有任何疑问或想要进一步探讨的内容&#xff0c;欢迎在评论区畅所欲言&#xff01;我们一起学习、共同成长~&#xff01; &#x1f44d; &#xff1a;如果你觉得这篇文章还不错&#xff0c;不妨顺手点个赞、加入收藏&#xff0c;并…