-- 图书借阅
数据库设计-- 记录每一本图书借出和还回的 *** 作-- 书的信息DROP DATABASE IF EXISTS db_libCREATE DATABASE db_libuse db_libSET @READER_MAX = 10SET @EVERY_DAY_FEE=0.1SELECT 10 INTO @temp-- 新建相关的表CREATE TABLE books (id INT PRIMARY KEY AUTO_INCREMENT,name CHAR(255) NOT NULL,price DECIMAL(8,2) DEFAULT 0)CREATE TABLE readers (id INT PRIMARY KEY AUTO_INCREMENT,name CHAR(255) NOT NULL,borrowed INT DEFAULT 0)CREATE TABLE lib_books (id INT PRIMARY KEY AUTO_INCREMENT,info INT,FOREIGN KEY(info) REFERENCES books(id),stat CHAR(10) NOT NULL DEFAULT "在馆")-- 记录
借书和还书CREATE TABLE lib_loan (id INT AUTO_INCREMENT,who INT,which INT,check_out DATETIME NOT NULL DEFAULT NOW(),check_in DATETIME DEFAULT NULL,renew BOOLEAN NOT NULL DEFAULT 0,latest DATE NOT NULL,fee DECIMAL(8,2) DEFAULT NULL,PRIMARY KEY(id),FOREIGN KEY(who) REFERENCES readers(id),FOREIGN KEY(which) REFERENCES lib_books(id))-- 给馆藏图书添加10本书-- 存储过程DELIMITER $$$CREATE PROCEDURE add_books( IN n INT) BEGIN-- 需要有多行数据放入变量,需要用 游标 cursor-- i 局部变量DECLARE i INT DEFAULT 1-- @temp全局变量SELECT COUNT(*) INTO @tempFROM booksWHILE i<=n DOINSERT INTO lib_books (info) VALUES ( FLOOR(RAND()*@temp)+1 )SET i = i+1END WHILEEND $$$DELIMITER -- 触发器 trigger ,管理图书被借出后,藏书和读者信息的维护-- 表中数据相关的 *** 作: 增 删 改 (查)-- 一个表中某条记录被修改:OLD 代表原有的那条记录 NEW代表新的那条记录-- [BEFORE | AFTER] [INSERT | UPDATE | (DELETE) ]DELIMITER $$$CREATE TRIGGER before_insert_loanBEFORE INSERT ON lib_loanFOR EACH ROWBEGIN-- OLD NEW 已知DECLARE v1 CHAR(100)DECLARE v2 INTSET NEW.latest =DATE_ADD(NEW.check_out,INTERVAL 1 month)-- 只能修改NEWSELECT stat INTO v1FROM lib_booksWHERE id=NEW.whichSELECT borrowed INTO v2FROM readersWHERE id=NEW.whoIF ( v1<>"在馆" ) THENSIGNAL SQLSTATE '45000'SET MESSAGE_TEXT = "图书已借出!"END IF IF (v2=@READER_MAX) THENSIGNAL SQLSTATE '45000'SET MESSAGE_TEXT = "读者已经到达最大借阅数!"END IF END $$$DELIMITER -- 书借出的数据维护DELIMITER $$$CREATE TRIGGER after_insert_loanAFTER INSERT ON lib_loanFOR EACH ROWBEGINUPDATE lib_booksSET stat = "已借出"WHERE id = NEW.which UPDATE readersSET borrowed = borrowed + 1WHERE id = NEW.whoEND $$$DELIMITER -- 书归还的数据维护DELIMITER $$$CREATE TRIGGER after_update_loanAFTER UPDATE ON lib_loanFOR EACH ROWBEGINIF(OLD.check_in is NULL AND NEW.check_in is NOT NULL) THENUPDATE lib_booksSET stat="在馆"WHERE id= NEW.whichUPDATE readersSET borrowed=borrowed -1WHERE id=OLD.whoEND IFEND $$$DELIMITER DELIMITER $$$CREATE TRIGGER before_update_loanBEFORE UPDATE ON lib_loanFOR EACH ROWBEGINDECLARE x INTIF(OLD.renew=0 AND NEW.renew=1 ) THENSET NEW.latest=DATE_ADD(OLD.latest,INTERVAL 1 month)END IF-- 计算超期费用IF(OLD.check_in is NULL AND NEW.check_in is NOT NULL) THENSET x=DATEDIFF(NEW.check_in,NEW.latest)IF(x<=0 )THENSET NEW.fee =0 ELSESET NEW.fee=@EVERY_DAY_FEE*xEND IFEND IFEND $$$DELIMITER -- 添加若干的测试数据INSERT INTO books (name, price)VALUES("javascript程序设计", 19.8),("数据库系统开发", 20.8),("mysql数据库必知必会", 28.8)INSERT INTO readers ( name )VALUES("Alice"),("Bob"),("Carl")-- 测试语句SHOW TABLES-- 根据书的信息添加10本馆藏图书CALL add_books( 10) SELECT * from lib_books-- 借书-- 2号读者借了2号书INSERT INTO lib_loan (who , which ) VALUES (2,2)INSERT INTO lib_loan (who , which ) VALUES (2,3)INSERT INTO lib_loan (who , which ) VALUES (2,4)INSERT INTO lib_loan (who , which ) VALUES (1,2)INSERT INTO lib_loan (who , which ) VALUES (1,6)INSERT INTO lib_loan (who , which ) VALUES (2,1)INSERT INTO lib_loan (who , which ) VALUES (1,1)-- 还书UPDATE lib_loanSET check_in='2022-12-15 12:00:00'WHERE id = 1-- 续借UPDATE lib_loanSET renew = 1WHERE id = 4SELECT * FROM lib_booksSELECT * FROM readersSELECT * FROM lib_loan
Mysql是目前互联网使用最广的关系数据库,关系数据库的本质是将问题分解为多个分类然后通过关系来查询。 一个经典的问题是用户借书,三张表,一个用户,一个书,一个借书的关系表。当需要查询某个用户借书情况或者是书被那些人借了,就用关系查询来实现。
关系数据库范式
来自英文Normal form,简称NF。要想设计—个好的关系,必须使关系满足一定的约束条件,满足这些规范的数据库是简洁的、结构明晰的,同时,不会发生插入(insert)、删除(delete)和更新(update) *** 作异常。总共有六种范式:第一范式(1NF)、第二范式(2NF)、 第三范式 (3NF)、巴斯-科德范式(BCNF)、 第四范式 (4NF)和 第五范式 (5NF,又称完美范式)。
1NF是指数据库表的每一列都是不可分割的原子数据项。2NF必须满足1NF,要求数据库表中的每行记录必须可以被唯一地区分。3NF在2NF基础上,任何非主 属性 不依赖于其它非主属性(在2NF基础上消除传递依赖)。BCNF是在3NF基础上,任何非主属性不能对主键子集依赖(在3NF基础上消除对主码子集的依赖), 满足BCNF不再会有任何由于函数依赖导致的异常,但是我们还可能会遇到由于多值依赖导致的异常。4NF的定义很简单:已经是BC范式,并且不包含多值依赖关系。5NF处理的是无损连接问题,这个范式基本没有实际意义,因为无损连接很少出现,而且难以察觉。而域键范式试图定义一个终极范式,该范式考虑所有的依赖和约束类型,但是实用价值也是最小的,只存在理论研究中。
Catalog和Schema
是数据库对象命名空间中的层次,主要用来解决命名冲突的问题。从概念上说,一个数据库系统包含多个Catalog,每个Catalog又包含多个Schema,而每个Schema又包含多个数据库对象(表、视图、字段等)。但是Mysql的数据库名就是Schema,不支持Catalog。
Mysql的数据库引擎主要有两种MyISAM和InnoDB,MyISAM支持全文检索,InnoDB支持事务。
SQL中的通配符‘%’代表任意字符出现任意次数。‘_’代表任意字符出现一次。SQL与正则表达式结合查询一般用在WHERE table_name REGEXP '^12.34'。子查询是从里到外执行。
数据库联结(join)涉及到外键,外键是指一个表的列是另一个表的主键,那么它就是外键。笛卡尔积联结(不指定联结条件时)生成的记录条目是单纯的第一个表的行乘以第二个表的列数。用得最多的是等值联结也叫内部联结。
高级联结还有自连接,是指查询中的两张表是同一张表,它通常作为外部语句用来代替从相同表中检索数据时使用的子查询。自然联结使每个列只返回一次。外部联结是指联结包含了那些在相关表中没有关联行的行。例如列出所有产品及其订购数量,包括没有人订购的产品。LEFT OUTER JOIN指选择左边表的所有行。
组合查询是指采用UNION等将两个查询结果取并集。
视图是查看存储在别处的数据的一种工具,它本身并不包含数据,因此表的数据修改了,视图返回的数据也将随之修改,因此如果使用了复杂或嵌套视图会对性能有较大的影响。视图的作用之一是隐藏复杂的SQL通常会涉及到联结查询。
存储过程类似于批处理,包含了一条或多条SQL语句。语法:
CREATE PROCEDURE name()
BEGIN
SQL
END
-------------------------
CALL name()//来调用存储过程
游标有DECLARE定义,游标与存储过程是绑定的,存储过程处理完成,游标就会消失。游标被打开后可以使用FETCH语句访问每一行。
触发器是在某个时间发生时自动执行某条SQL语句。语法:
CREATE TRIGGER name AFTER INSERT ON talbe_name FOR EACH ROW
事务处理可以维护数据库的完整性,保证批量的 *** 作要么完全执行,要么完全不执行。包括事务、回退、提交、保留点几个关键术语。ROLLBACK只能在一个事务处理内使用。他不能回退CREATE和DROP *** 作。使用COMMIT保证事务提交。复杂的事务处理需要部分提交或回退,因此我们需要使用保留点SAVEPOINT。可以使用ROLLBACK TO savepoint_name。保留点越多越好。保留点在事务执行完成后自动释放。
应该还有一个借书记录的表记录谁什麼时候借出了哪本书。假设为A表
则可以这样写
select t1.*,t3.* from people t1 left join A t2 on t1.id = t2.id left join books t3 on t2.isbn = t3.isbn where t1.id = '你想要查找的借书人'
评论列表(0条)