【数据库——MySQL】(13)过程式对象程序设计——存储函数、错误处理以及事务管理

news/2024/10/23 5:38:58/

目录

  • 1. 存储函数
  • 2. 存储函数的应用
  • 3. 错误处理
  • 4. 抛出异常
  • 5. 事务处理
  • 6. 事务隔离级
  • 7. 应用实例
  • 参考书籍

1. 存储函数

  1. 创建 存储函数,需要用到 CREATE 语句:

    CREATE FUNCTION 存储函数名([参数名 类型, ...])RETURNS 类型[存储函数体]
    

    注意:存储过程名和存储函数名不能相同!

  2. 调用 存储函数,语法格式如下:

    存储函数名([参数, ...])
    

    :存储过程只能采用 CALL 语句直接调用,而存储函数则可以出现在各种语句中。

  3. 修改 存储函数,需要用到 ALTER 语句:

    ALTER FUNCTION 存储函数名 
    [COMMENT ‘string’| LANGUAGE SQL| {CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA }| SQL SECURITY { DEFINER | INVOKER }
    }
    

    :上述修改语句只能修改存储函数的属性,不能修改功能!修改存储函数的功能,与修改存储过程功能一样也只能采取先删除后重新定义的方式实现

  4. 删除 存储函数,需要用到 DROP 语句:

    DROP FUNCTION [IF EXISTS] 存储函数名
    
  5. 注意在创建自定义存储函数的时候,需要提供权限

    set GLOBAL log_bin_trust_function_creators = 1;    # 创建自定义函数需要给权限
    

    :上述语句只需要在开启数据库时输入一次即可。

2. 存储函数的应用

【例 1】在数据库 score 中创建存储函数,要求通过给定的学号输出相对应的姓名。

drop FUNCTION if exists f1;
delimiter $
create FUNCTION f1(stu_id char(20))
returns varchar(20)    # 返回学生姓名
begindeclare sname varchar(20);select stu.`name` into snamefrom stuwhere stu.id = stu_id;return sname;
end$
delimiter ;
select f1('20191001') as Name;

在这里插入图片描述

【例 2】在数据库 score 中创建存储函数,要求输出每位学生的所修学分总数。

drop FUNCTION if exists getxf;
delimiter $
create FUNCTION getxf(stu_id char(20))
returns decimal(5,1)    # 返回学生总学分
begindeclare s decimal(5,1) default 0;   # 因为之后要求和,所以给个缺省值0select sum(if(score.score >= 60, lesson.xf, 0)) into sfrom score join lesson on score.LessonId = lesson.lessonidwhere score.stuId = stu_id;return s;
end$
delimiter ;
select stu.id, stu.`name`, IFNULL(getxf(stu.id),0) as 学分
from stu;

在这里插入图片描述

3. 错误处理

在执行 SQL 语句后,可能有时会出错,那么大家可以根据报错编号在 MySQL 手册里面查找出错原因。

错误处理语句:

DECLARE 处理动作 HANDLER FOR 条件值 ,...处理语句过程体

4. 抛出异常

我们还可以自定义错误代码以及抛出异常值:

SIGNAL SQLSTATE 错误编号(自定义,不能与系统已有编号重复)
SET message_text = 错误提示信息;

示例如下:

SIGNAL SQLSTATE '12345' 
set message_text = '证件号不存在';

5. 事务处理

  1. 关闭自动提交
    自动提交常用语银行系统中,比如小邓给森林转了 1 元的写博客助力费,那么这个动作应该是同步的,即小邓银行账户扣除 1 元,与此同时,森林银行账户增加 1 元(假设小邓银行账户刚好有 1 元,足够扣除)。不能说小邓银行账户扣除 1 元,但森林银行账户余额不变(比如转钱信号传输中受到干扰导致信号未正确传输)。

    这时候 关闭自动提交 就可以实现解决这种情况。当完成一系列操作后,只有手动提交事务,才能算是实现了之前的所有操作!

    SET @@AUTOCOMMIT = 0;
    

    :默认是开启自动提交的,即 SET @@AUTOCOMMIT = 0;

  2. 开始事务

    START TRANSACTIONSQL语句
    
  3. 结束事务

    COMMIT [WORK] [AND [NO] CHAIN] [[NO] RELEASE]
    
  4. 撤销事务

    ROLLBACK [WORK] [AND [NO] CHAIN] [[NO] RELEASE]
    
  5. 回滚事务到指定点
    回滚前需要先设置一个保存点:

    SAVEPOINT 保存点名
    

    然后才可以回滚事务:

    ROLLBACK [WORK] TO SAVEPOINT 保存点名
    

6. 事务隔离级

  1. 脏读(dirty reads):一个事务读取了另一个未提交的并行事务写的数据。
  2. 不可重复读(non-repeatable reads):一个事务重新读取前面读取过的数据, 发现该数据已经被另一个已提交的事务修改过。
  3. 幻读(phantom read):一个事务重新执行一个查询,返回一套符合查询条件的行, 发现这些行因为其他最近提交的事务而发生了改变。

以上 3 种情况是我们不想看见的。

SET [GLOBAL | SESSION] TRANSACTION ISOLATION LEVELSERIALIZABLE					//可序列化| REPEATABLE READ				//可重复读| READ COMMITTED				//提交读| READ UNCOMMITTED				//未提交读

如果指定GLOBAL,那么定义的隔离级将适用于所有的SQL用户;如果指定SESSION,则隔离级只适用于当前运行的会话和连接。

基于 ANSI/ISO SQL 规范,MySQL 提供了 4 种隔离级:序列化、可重复读、提交读和未提交读

查看当前事务隔离级:

SELECT @@TRANSACTION_ISOLATION;

7. 应用实例

请自行创建数据库 bank,并在里面新建一个表 account,表内容如下:

在这里插入图片描述

下面完成转账操作:

set @@autocommit = 0;    # 关闭自动提交功能select * from account;update account
set ck = ck - 500
where id = '001';SAVEPOINT a;    # 保存点(rollback时只撤销后面的操作)update account
set ck = ck + 500
where id = '002';select * from account;

此时还未提交事务,大家可以在可视化界面刷新表 account,发现数据并没有更新。这是因为我们关闭了自动提交,因此要手动提交事务:

COMMIT;

在这里插入图片描述

参考书籍

《MySQL实用教程(第4版)》

上一篇文章:【数据库——MySQL】(12)过程式对象程序设计——存储过程


http://www.ppmy.cn/news/1130764.html

相关文章

深度解读F5:从企业级负载均衡到云原生应用服务

上世纪九十年代,Internet 的快速发展催生了大量在线网站,Web 访问量迅速提升。在互联网泡沫破灭以前,这个领域基本是围绕如何对 Web 网站进行负载均衡与优化。因而在早期,也会有“Web 交换机”的说法。从1997年 F5 发布了 BIG-IP …

关于Java的IO流里面的方法read()的细究

我们会经常看到上面的代码,很显然read(b),是把字节读入到b这个字节数组里,然后read()返回一个成功读取的字节长度,如果现在用不带参的read()去一个一个字节的读了?那么len bInput.read()就是把这一个节字读入到len&am…

Win10电脑任务栏没有蓝牙图标的简单解决方法

Win10电脑任务栏没有蓝牙图标怎么办?在Win10电脑中,用户有时候会发现任务栏上没有蓝牙图标了,这样就无法通过蓝牙图标快速打开蓝牙服务了。下面小编给大家介绍最简单的解决方法,帮助大家找回任务栏上面的蓝牙图标吧。 问题原因 反…

Unity调用API函数对系统桌面和窗口截图

Unity3D调用WINAPI函数对系统窗口截图 引入WINAPI函数调用WINAPI函数进行截图使用例子 引入WINAPI函数 using System; using System.Collections; using System.Runtime.InteropServices; using System.Drawing;[DllImport("user32.dll")]private static extern Int…

【计算机网络】Linux 内核网络概述

文章目的 了解 Linux 内核网络架构通过网络包过滤器或者防火墙获得使用的 IP 数据包(分组)管理技巧熟悉如何在 Linux 内核级别使用套接字 概述 网络应用程序的开发过去这些年按照指数级增长,这样增加了对系统网络子系统的速度要求和产品化要…

Python 中的 Urljoin 简介

文章目录 介绍Python中的urljoin及其使用方法使用urljoin()方法使用urljoin()模块构建URLs使用urljoin()构建URLs 本篇文章介绍了 Python urljoin 及其使用时的行为。 它还使用不同的示例代码演示了在 Python 中使用 urljoin。 介绍Python中的urljoin及其使用方法 URL通常包含…

AutoAnimate - 无需任何配置,一行代码自动为元素添加优雅的过渡动画,可以搭配 Vue / React 和 Sevelt 使用

这个动画库只要一行代码就可以自动在我们的组件中添过渡动画,为什么这么省事高效呢? AutoAnimate 是一个无需任何配置,自动为我们开发的 Web 项目添加平滑过渡动画的 JavaScript 工具库。AutoAnimate 和之前推荐的一些 js 动画库相比&#x…

Java之SpringCloud Alibaba【六】【Alibaba微服务分布式事务组件—Seata】

一、事务简介 事务(Transaction)是访问并可能更新数据库中各种数据项的一个程序执行单元(unit)。 在关系数据库中,一个事务由一组SQL语句组成。 事务应该具有4个属性: 原子性、一致性、隔离性、持久性。这四个属性通常称为ACID特性。 原子性(atomicity) ∶个事务…