【数据库——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.rhkb.cn/news/159043.html

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

相关文章

相似性搜索:第 3 部分--混合倒排文件索引和产品量化

接续前文&#xff1a;相似性搜索&#xff1a;第 2 部分&#xff1a;产品量化 SImilarity 搜索是一个问题&#xff0c;给定一个查询的目标是在所有数据库文档中找到与其最相似的文档。 一、介绍 在数据科学中&#xff0c;相似性搜索经常出现在NLP领域&#xff0c;搜索引擎或推…

Codeforces Round 903 (Div. 3)

D. Divide and Equalize Example input Copy 7 5 100 2 50 10 1 3 1 1 1 4 8 2 4 2 4 30 50 27 20 2 75 40 2 4 4 3 2 3 1 output Copy YES YES NO YES NO YES NONote The first test case is explained in the problem statement. 很重要很重要的知识点&a…

Windows端口号被占用的查看方法及解决办法

Windows端口号被占用的查看方法及解决办法 Error starting ApplicationContext. To display the conditions report re-run your application with debug enabled. 2023-10-14 22:58:32.069 ERROR 6488 --- [ main] o.s.b.d.LoggingFailureAnalysisReporter : ***…

CustomNavBar 自定义导航栏视图

1. 创建偏好设置键 CustomNavBarTitlePreferenceKey.swift import Foundation import SwiftUI//State private var showBackButton: Bool true //State private var title: String "Title" //"" //State private var subtitle: String? "SubTitl…

算法练习13——跳跃游戏II

LeetCode 45 跳跃游戏 II 给定一个长度为 n 的 0 索引整数数组 nums。初始位置为 nums[0]。 每个元素 nums[i] 表示从索引 i 向前跳转的最大长度。换句话说&#xff0c;如果你在 nums[i] 处&#xff0c;你可以跳转到任意 nums[i j] 处: 0 < j < nums[i] i j < n 返回…

螺杆支撑座对注塑机的生产过程有哪些重要影响?

螺杆支撑座对注塑机的生产过程具有重要影响&#xff0c;主要体现在以下几个方面&#xff1a; 1、精度和稳定性&#xff1a;螺杆支撑座能够提高注塑机的精度和稳定性&#xff0c;从而保证塑料制品的品质和一致性。通过提供稳定的支撑和承载&#xff0c;螺杆支撑座可以减少机器运…

React18入门(第三篇)——React Hooks详解,React内置Hooks、自定义Hooks使用

文章目录 概述一、内置 Hook——useState1.1 响应式数据更新1.2 什么是 state1.3 state 特点&#xff08;一&#xff09;——异步更新1.4 state 特点&#xff08;二&#xff09;——可能会被合并1.5 state 特点&#xff08;三&#xff09;——不可变数据&#xff08;重要&#…

MySQL的各种锁

1. MySQL有遇到过死锁的问题吗&#xff1f;你是如何解决的&#xff1f; 死锁&#xff0c;就是两个或两个以上的线程在执行过程中&#xff0c;去争夺同一个共享资源导致互相等待的现象&#xff0c;在没有外部干预的情况下&#xff0c;线程会一直处于阻塞状态&#xff0c;无法往下…

【RocketMQ系列二】通过docker部署单机RocketMQ

您好&#xff0c;我是码农飞哥&#xff08;wei158556&#xff09;&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f4aa;&#x1f3fb; 1. Python基础专栏&#xff0c;基础知识一网打尽&#xff0c;9.9元买不了吃亏&#xff0c;买不了上当。 Python从入门到精…

goland安装教程

安装版本&#xff1a; goland-2023.2.3.exe

spring boot+ vue位置信息大数据综合管理平台源码

spring boot vue位置信息大数据综合管理平台源码 UWB技术的人员定位系统源码 智慧工厂是产业升级的外在表现形式&#xff0c;利用物联网技术加强信息管理的新模式&#xff0c;人员定位管理通过物联网技术、位置信息大数据的综合处理应用&#xff0c;在智慧工厂人员管理方面具有…

git强制删除本地分支 git branch -D

git强制删除本地分支 git branch -D git删除本地分支_zhangphil的博客-CSDN博客git branch -d <分支名>可以通过: git branch 查看所有本地分支及其名字&#xff0c;然后删除特定分支。https://blog.csdn.net/zhangphil/article/details/82255002 使用git branch -d删除…

MacBook/MacOS如何更新到指定的版本

背景 现在是A版本&#xff0c;想要更新到B&#xff0c;而目前能最新更新到C。 是可以做到的&#xff0c;不一定更新就得更新到最新的。 只要下载好B之后更新即可。 方法 思路是下载好目标的版本后更新&#xff0c;这里可以下载&#xff1a; https://support.apple.com/zh-…

Arbitrum Stylus 的工作原理

理解 Arbitrum 如何协调 EVM 和 WASM 的共存是至关重要的。这不仅仅是拥有两个独立的引擎&#xff1b;这是一种增强两者优势的协同关系。 Arbitrum 的独特架构允许 EVM 和 WASM 之间进行无缝和同步的操作&#xff0c;这要归功于其统一的状态、跨 VM 调用和兼容的经济模型。 用…

uniapp小程序实现绘制内容,生成海报并保存截图(Painter和Canvas两种方式举例)

一、Painter方法 Painter插件传送门 1.下载资源包 2.将资源包的如下部分 3.使用页面引入组件 ui样式 <paintercustomStyle=margin-left: 40rpx; height: 1000rpx;palette="{{palette}}"bind:imgOK="onImgOK"/>data 中数据(绘制内容,替换区域) pai…

8.简易无线通信

预备知识 Zigbee无线通信&#xff0c;需要高频的载波来提供发射效率&#xff0c;Zigbee模块之间要可以正常的收发&#xff0c;接收模块必须把接收频率设置和发射模块的载波频率一致。Zigbee有27个载波可以进行通信&#xff0c;载波叫做信道&#xff08;无线通信的通道&#xf…

Umi + React + Ant Design Pro + TS 项目搭建

新建项目目录 mkdir 【项目名称】在对应目录 D:\react\demo 中&#xff0c;安装 Umi 脚手架&#xff1a; yarn create umi接下来&#xff0c;安装将要用到的相关依赖 umijs/plugins&#xff1a; npm i umijs/plugins -Dumijs/plugins 是 Umi 的官方插件集&#xff0c;包含了…

STM32单片机入门学习(六)-光敏传感器控制LED

光敏传感器模块和LED接线 LED负极接B12,正极接VCC 光敏传感模块一DO端接B13,GND接GND&#xff0c;VCC接VCC,AO不接。 如图&#xff1a; 主程序代码&#xff1a;main.c #include "stm32f10x.h" #include "Delay.h" //delay函数所在头文件 #include …

HTTPS 加密全过程

加密协议以前是SSL,现在都是TLS, 而证书现在大多数都是SSL证书 抓包流程: TCP三次握手过后, 客户端发送Client Hello 服务器相应Server Hello 服务器再次响应发送证书: 服务器再发送公钥:

凉鞋的 Godot 笔记 109. 专题一 小结

109. 专题一 小结 在这一篇&#xff0c;我们来对第一个专题做一个小的总结。 到目前为止&#xff0c;大家应该能够感受到此教程的基调。 内容的难度非常简单&#xff0c;接近于零基础的程度&#xff0c;不过通过这些零基础内容所介绍的通识内容其实是笔者好多年的时间一点点…