【数据库——MySQL(实战项目1)】(4)图书借阅系统——触发器

目录

  • 1. 简述
  • 2. 功能代码
    • 2.1 创建两个触发器,分别在借出或归还图书时,修改借阅人表中的已借数目(附加:借阅人表的总借书数、图书表的借阅次数以及更新图书表的图书状态为(已借出/在架上))字段;
    • 2.2 创建触发器,当借阅者已借阅的书籍数目达到限额(附加:当借阅者证件号不存在、借阅者星级为 `0`、重复借入同一本书籍【未还同一本书的情况下】、图书已经被他人借了(此情况只发生在线上预约的时候-目前构建的数据库只适用于线下)、书籍不存在时)时,禁止借入新的书籍;
    • 2.3 创建触发器,当要从借阅人表中删除借阅人时(毕业了),需要先判断此借阅人是否有书未还,有的话禁止从借阅人表中删除此借阅人。
  • 结尾

1. 简述

在介绍本项目功能前,突然有感而发,讲讲我是如何对待程序员不写注释的吧(不感兴趣的小伙伴可以跳过这段话~)。作为一名合格的程序员,写注释是最基本的要求!!!它有如下好处:

  1. 帮助你回忆起这段代码在干什么?大家都写过代码吧,代码长度或长或短。可能一些短代码在仔细阅读后可以知道它在干什么,但是阅读代码的时间我相信我已经看了几个短代码的注释了!所以写注释可以提高我们读代码的效率(这是基于短代码而言的)。对于长代码,比如我写过的背单词软件(森林背词),仅仅是完成基础的背词、复习、调整各单词复习时间、利用数据库存储数据(不包含一些活动、优惠券之类)的代码就有 1w+ 的代码行数了(python 代码,如果是其它的代码可能更长~),如果不写注释,加上过了一段时间之后,连我自己都不记得某一段代码是干什么的了。。。但是有注释就不一样了,想要了解或者修改某一个功能代码时,只要搜索以下注释的关键字就能马上定位(这个也是血的教训!!!)。在重构森林背词之前,我用 python 写过一次,那时候写的功能更简单,但是要想从旧版本升级到新版本(重构),就要对原代码有一定的了解,因为之前没写过注释,所以看每个功能都很慢,最后干脆从头重写代码,这也导致我的工作量激增~
  2. 写注释除了帮助你自己了解和修改代码外,也可以帮助别人了解和修改你的代码。你也不想接手一个旧项目的代码,发现里面啥注释也没有吧,这样当你要维护的时候,时间成本是很大的
  3. 当然注释的质量也是很重要的,要是一段注释说了跟没说一样,或者说的牛头不对马尾,那只会浪费你的时间。
  4. 除了注释,还有一个重点就是功能一定要分模块来写,不要全写在一个文件里面,要不然要你修改的时候你会崩溃。这里还有一个好处就是可以多人协作编写不同的功能代码~

那么下面就正式开始图书借阅系统的最后一篇介绍文章~


这篇文章将主要完成图书借阅系统的 9~11 题(触发器),即:

  1. 创建两个触发器,分别在借出或归还图书时,修改借阅人表中的已借数目(附加:借阅人表的总借书数、图书表的借阅次数以及更新图书表的图书状态为(已借出/在架上))字段;
  2. 创建触发器,当借阅者已借阅的书籍数目达到限额(附加:当借阅者证件号不存在、借阅者星级为 0、重复借入同一本书籍【未还同一本书的情况下】、图书已经被他人借了(此情况只发生在线上预约的时候-目前构建的数据库只适用于线下)、书籍不存在时)时,禁止借入新的书籍;
  3. 创建触发器,当要从借阅人表中删除借阅人时(毕业了),需要先判断此借阅人是否有书未还,有的话禁止从借阅人表中删除此借阅人。

:临时多了一些想法,把第 10 题新增了一些功能~

2. 功能代码

2.1 创建两个触发器,分别在借出或归还图书时,修改借阅人表中的已借数目(附加:借阅人表的总借书数、图书表的借阅次数以及更新图书表的图书状态为(已借出/在架上))字段;

# 借书触发器
drop TRIGGER if exists tri_update_borrowerlend;
delimiter $
create TRIGGER tri_update_borrowerlend after insert on information for each row
beginupdate borrowerset borrowedCount = borrowedCount + 1where id = new.borrowerid;update borrowerset borrowedAll = borrowedAll + 1where id = new.borrowerid;update bookset lendCount = lendCount + 1where id = new.bookid;update bookset state = '已借出'where id = new.bookid;
end$
delimiter ;
# 还书触发器
drop TRIGGER if exists tri_update_borrowerreturn;
delimiter $
create TRIGGER tri_update_borrowerreturn after update on information for each row
beginupdate borrowerset borrowedCount = borrowedCount - 1where id = new.borrowerid;update bookset state = '在架上'where id = new.bookid;
end$
delimiter ;

检测触发器是否正确:

# (触发器1)在借出图书时,修改借阅人表中的已借数目(附加:借阅人表的总借书数、图书表的借阅次数以及更新图书表的图书状态为(已借出/在架上))字段;
select * from borrower;
select book.id as 图书编号, book.lendCount as 借阅次数, book.state as 图书状态 from book;
call p_InsertLeadInfo(2020312011047,101101699412);
select * from borrower;
select book.id as 图书编号, book.lendCount as 借阅次数, book.state as 图书状态 from book;# (触发器2)在归还图书时,修改借阅人表中的已借数目字段;
call p_UpdateLeadInfo(2020312011047,101101699412);
select * from borrower;
select book.id as 图书编号, book.lendCount as 借阅次数, book.state as 图书状态 from book;

测试结果如下:

  1. 触发器1(借书触发器)

    在这里插入图片描述

    可见此触发器是生效的,能够在借出图书时,将借阅人表中的当前借阅数目 +1,借阅人表的总借书数 +1、图书表的借阅次数 +1 以及图书表的图书状态从“在架上”更新为“已借出”。

  2. 触发器2(还书触发器)

    在这里插入图片描述
    可见此触发器是生效的,能够在归还图书时,将借阅人表中的当前借阅数目 -1 以及图书表的图书状态从“已借出”更新为“在架上”。

2.2 创建触发器,当借阅者已借阅的书籍数目达到限额(附加:当借阅者证件号不存在、借阅者星级为 0、重复借入同一本书籍【未还同一本书的情况下】、图书已经被他人借了(此情况只发生在线上预约的时候-目前构建的数据库只适用于线下)、书籍不存在时)时,禁止借入新的书籍;

drop trigger if EXISTS tri_insertinfo;
delimiter $
create TRIGGER tri_insertinfo before insert on information for each row
begin# 借阅者已借阅的书籍数目达到限额可使用存储函数f_ReBorrowTimes()完成declare a int;declare b int;declare repeat_borrow int;		# 是否重复借入书籍declare c int;		# 0:图书已借出:1:图书在架上declare d int;select count(*) into afrom borrowerwhere borrower.id = new.borrowerid;select count(*) into bfrom borrowerwhere borrower.grade = 0 and borrower.id = new.borrowerid;SELECT if(returnDateReality is null, 0, 1) INTO repeat_borrowFROM informationWHERE information.borrowerid = new.borrowerid AND information.bookid = new.bookidORDER BY returnDateReality ASCLIMIT 1;select if(book.state='已借出',0,1) into cfrom bookwhere book.id = new.bookid;select count(*) into dfrom bookwhere book.id = new.bookid;if a = 0 then SIGNAL SQLSTATE '12345' set message_text = '证件号不存在';elseif b != 0 then SIGNAL SQLSTATE '12346' set message_text = '借阅人等级为0(已列为失信人,禁止借入书籍)';elseif f_ReBorrowTimes(new.borrowerid) <= 0 then SIGNAL SQLSTATE '12347' set message_text = '已借阅的书籍数目达到限额';elseif repeat_borrow = 0 then SIGNAL SQLSTATE '12348' set message_text = '书籍禁止重复借入';elseif c = 0 then SIGNAL SQLSTATE '12349' set message_text = '图书已被他人借阅';elseif d = 0 then SIGNAL SQLSTATE '12350' set message_text = '图书不存在';end if;end$
delimiter ;

检测触发器是否正确:

# 借阅者已借阅的书籍数目达到限额
call p_InsertLeadInfo(2020312011047,101102308670);
call p_InsertLeadInfo(2020312011047,101102698766);
call p_InsertLeadInfo(2020312011047,451100004441);
# 借阅者证件号不存在
call p_InsertLeadInfo(9999999999999,101101699410);
# 借阅者星级为0
update borrower set grade = 0 where id = 2018210210205;
call p_InsertLeadInfo(2018210210205,101101699410);
# 重复借入同一本书籍【未还同一本书的情况下】
call p_InsertLeadInfo(1000000000001,101101699410);
call p_InsertLeadInfo(1000000000001,101101699410);
# 图书已经被他人借了
call p_InsertLeadInfo(1000000000002,101101699410);
# 书籍不存在
call p_InsertLeadInfo(1000000000002,999999999999);

测试结果如下:

  1. 已借阅书籍达到限额(学生)【教师借阅是超过 10 本书就会触发此触发器,这里不进行演示,感兴趣的读者可自行验证~】:

    在这里插入图片描述
    从上面 触发器2(还书触发器) 的借阅人表可知证件号为“ 2020312011047 ”的借阅人(学生)已经借了 3 本书了,所以再借 2 本后,当继续借多 1 本时,显示借阅书籍数目达到上限,无法继续借阅。

  2. 证件号不存在:

    在这里插入图片描述
    证件号 9999999999999 不存在时,无法借阅书籍。

  3. 借阅人星级为 0 (已列为失信人,禁止借入书籍):

    在这里插入图片描述
    失信人(即长期逾期不还图书的用户)无法借阅书籍。

    :此系统没有写降低用户星级的代码(例如长期逾期不还图书的用户根据次数或者时长降低用户星级),同样可以按照用户恢复正常使用后多少天就增加用户星级(增加的星级不能超过 5 颗星,默认用户最多只有 5 颗星),读者可以自行编写~

  4. 书籍禁止重复借入:

    在这里插入图片描述
    书籍在第一次成功借出,在第二次借出失败。即书籍禁止重复借入(未还的情况下再次借入,即借入行为重复操作)。

  5. 图书已被他人借阅:

    在这里插入图片描述
    这本书刚刚已经借出了(在测试 4. 书籍禁止重复借入 时借出了),因此他人不可再借阅,其实在实际当中,这种情况基本上不会发生(除非用户本人一直借这本书 或者 这本书遗失被其他人借阅)。

  6. 图书不存在:

    在这里插入图片描述
    当输入的图书编号(如这里的 999999999999)不存在时,无法借入,其实这种情况也不会发生,除非是借阅时工作人员输入错误 或者 机器识别错误

2.3 创建触发器,当要从借阅人表中删除借阅人时(毕业了),需要先判断此借阅人是否有书未还,有的话禁止从借阅人表中删除此借阅人。

drop trigger if EXISTS tri_delete_borrower;
delimiter $
create TRIGGER tri_delete_borrower before delete on borrower for each row
begindeclare a int DEFAULT 0;select count(*) into afrom book join information on book.id = information.bookidjoin borrower on borrower.id = information.borroweridwhere borrower.id = old.id and information.returnDateReality is null;if a != 0 then SIGNAL SQLSTATE '12351' set message_text = '借阅人还有书籍未还,无法删除此借阅人';end if;# 如若数据库储存量过大,不想保存非在校师生数据,可以保留下述删除语句delete from information where information.borrowerid = old.id;end$
delimiter ;

检测触发器是否正确:

# 借阅人有书籍未还,无法删除
delete from borrower where id = 2020312011047;
# 借阅人无书籍未还,可以删除
select * from borrower;
delete from borrower where id = 2021312011002;
select * from borrower;

测试结果如下:

  1. 借阅人有书籍未还,无法删除:

    在这里插入图片描述

  2. 借阅人没有书籍未还,可以删除:

    在这里插入图片描述

结尾

好啦,本次的 MySQL 实战项目1 的介绍就到此结束了,后续如若有其它好的实战项目,我会继续分享~

大家感觉有帮助的可以点赞、收藏 + 关注哦,你的支持是我持续分享的动力~

上一篇文章:【数据库——MySQL(实战项目1)】(3)图书借阅系统——存储函数

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

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

相关文章

redis 哨兵 sentinel(一)配置

sentinel巡查监控后台master主机是否故障&#xff0c;如果故障根据投票数自动将某一个从库转换为新主库&#xff0c;继续对外服务 sentinel 哨兵的功能 监控 监控主从redis库运行是否正常消息通知 哨兵可以将故障转移的结果发送给客户端故障转移 如果master异常&#xff0c;则…

IOS17 轻松签全能签还能不能用?多开能否使用?升级后微信底栏消失怎么办?BY:后厂村路灯

从iphone15还没出就有小伙伴们追着问&#xff0c; 到现在也有人一直再问iOS17能不能用&#xff0c;看来换手机的人很多呀。 这里统一回答一下&#xff1a;“iOS17苹果签名可以用&#xff0c;多开也可以用”但是还是有些地方注意。 如果你是16系统直接升级刀17就可以&#xff…

使用 Splashtop 驾驭未来媒体和娱乐

在当今时代&#xff0c;数字转型不再是可选项&#xff0c;而是必选项。如今&#xff0c;媒体与娱乐业处于关键时刻&#xff0c;正在错综复杂的创意、技术和远程协作迷宫之中摸索前进。过去几年发生的全球事件影响了我们的日常生活&#xff0c;不可逆转地改变了行业的运作方式&a…

给你一个项目,你将如何开展性能测试工作?

一、性能三连问 1、何时进行性能测试&#xff1f; 性能测试的工作是基于系统功能已经完备或者已经趋于完备之上的&#xff0c;在功能还不够完备的情况下没有多大的意义。因为后期功能完善上会对系统的性能有影响&#xff0c;过早进入性能测试会出现测试结果不准确、浪费测试资…

【C++STL基础入门】list基本使用

文章目录 前言一、list简介1.1 list是什么1.2 list的头文件 二、list2.1 定义对象2.2 list构造函数2.3 list的属性函数 总结 前言 STL&#xff08;Standard Template Library&#xff09;是C标准库的一个重要组成部分&#xff0c;提供了一套丰富的数据结构和算法&#xff0c;可…

【Lombok的Bug记录】前端传的有值,但是到后端就全为空了

项目场景&#xff1a; 项目背景&#xff1a;使用Data注解标注类 问题描述 前端传的有值&#xff0c;但是到后端就全为空了 原因分析&#xff1a; AName和aName生成的set方法名是一样的&#xff0c;所以换名字就行了&#xff01; 解决方案&#xff1a; 属性不要写成xXxx的形式…

【伪彩色图像处理】将灰度图像转换为彩色图像研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

YB4014是可以对单节磷酸铁锂电池进行恒流/恒压充电管理的集成电路。

概述&#xff1a; YB4014是可以对单节磷酸铁锂电池进行恒流/恒 压充电管理的集成电路。该器件内部包括功率晶 体管&#xff0c;不需要外部的电流检测电阻和阻流二极管 YB4014只需要极少的外围元器件&#xff0c;非常适合于 便携式应用的领域。热调制电路可以在器件的功 耗比较大…

flutter开发入门,windows环境安装,耗时一天解决各种bug,终于成功

首先说明要安装的环境&#xff1a;java8必须&#xff0c;android studio&#xff0c;chrome是开发安卓和web是必须的 java8的下载地址&#xff1a;https://www.java.com/en/download/、 java8蓝奏云下载地址&#xff1a;jre-8u381-windows-x64.exe - 蓝奏云 flutter国内环境…

Spring Boot 中的 Redis 数据操作配置和使用

Spring Boot 中的 Redis 数据操作配置和使用 Redis&#xff08;Remote Dictionary Server&#xff09;是一种高性能的开源内存数据库&#xff0c;用于缓存、消息队列、会话管理和数据存储。在Spring Boot应用程序中&#xff0c;Redis被广泛用于各种用例&#xff0c;包括缓存、…

下载安装Ipa Guard

下载安装Ipa Guard 可以前往ipaguard工具官网下载&#xff0c;工具是免费下载&#xff0c;免费体验使用的。下载地址是https://www.ipaguard.com。 下载后解压工具便ok了&#xff0c;工具是绿色软件&#xff0c;无需其他安装流程。双击Ipa Guard.exe 启动ipaguard。 ipaguard…

Java语法基础案例(二)

目录 案例六&#xff1a;抢红包 案例七&#xff1a;找素数 方法一&#xff1a; 方法二&#xff1a; 方法三&#xff1a; 案例八&#xff1a;实现双色球 关于本项目所用所有方法的解释&#xff1a; 案例六&#xff1a;抢红包 一个大V直播时发起了抢红包活动&#xff0c;分…

2023年中国固废处理行业研究报告

第一章 行业概况 1.1 定义 固体废物处理是一个日益重要的领域&#xff0c;随着中国城市化进程的加速和工业产值的持续增长&#xff0c;固体废物的产生量也在不断上升。根据《固体废物污染环境防治法》的定义&#xff0c;固体废物包括了人类在生产、生活和其他活动中产生的固态…

AI低代码维格云甘特视图怎么用?

甘特视图,以日期为横轴展示任务持续时长和先后顺序,简称甘特图。 项目管理过程中,合理分配任务和资源至关重要,使用甘特图,妥当解决以下场景: 想知道整个项目的周期多长,哪些任务对项目的周期影响最大; 想知道每个任务的时间有多长,任务的优先级和依赖关系是什么; 想…

基于深度学习的“语义通信编解码技术”框架分类

目录 基于神经网络的语义提取基于神经网络的语义信源编码基于神经网络的语义信源信道联合编码基于神经网络的语义编码与数字调制联合设计参考文献 基于神经网络的语义提取 在现有的信源编码前端加上一个语义提取神经网络[53] &#xff0c;如图所示。语义提取神经网络的输入是原…

接口自动化测试框架搭建【附教程加源码】

1 接口测试 接口测试是对系统或组件之间的接口进行测试&#xff0c;主要是校验数据的交换&#xff0c;传递和控制管理过程&#xff0c;以及相互逻辑依赖关系。 接口自动化相对于UI自动化来说&#xff0c;属于更底层的测试&#xff0c;这样带来的好处就是测试收益更大&#xff…

【TensorFlow2 之015】 在 TF 2.0 中实现 AlexNet

一、说明 在这篇文章中&#xff0c;我们将展示如何在 TensorFlow 2.0 中实现基本的卷积神经网络 \(AlexNet\)。AlexNet 架构由 Alex Krizhevsky 设计&#xff0c;并与 Ilya Sutskever 和 Geoffrey Hinton 一起发布。并获得Image Net2012竞赛中冠军。 教程概述&#xff1a; 理论…

OpenCV实现答题卡自动打分!

目录 1&#xff0c;主要原理以及函数介绍 全部代码&#xff0c;以 2 &#xff0c; 实现过程 3&#xff0c;结果展示 1&#xff0c;主要原理以及函数介绍 ap argparse.ArgumentParser() 创建一个ArgumentParser对象&#xff0c;并将其赋值给变量ap。这个对象可以接受我们的脚…

gitbook使用

文章目录 前言一、安裝Node.js&#xff1a;二、安裝GitBook&#xff1a;編輯和創建內容&#xff1a;安装typora 前言 ubuntu20.04系统 一、安裝Node.js&#xff1a; 打開終端機&#xff08;Terminal&#xff09;。 使用以下命令來安裝Node.js和npm&#xff08;Node.js的包管…

Vulnhub系列靶机---Raven: 2

文章目录 信息收集主机发现端口扫描目录扫描用户枚举 漏洞发现漏洞利用UDF脚本MySQL提权SUID提权 靶机文档&#xff1a;Raven: 2 下载地址&#xff1a;Download (Mirror) 信息收集 靶机MAC地址&#xff1a;00:0C:29:15:7F:17 主机发现 sudo nmap -sn 192.168.8.0/24sudo arp…