《大型数据库技术》MySQL 的进阶开发技巧,java 基础知识重点总结 pdf
INSERT INTO salerecords VALUES(a, b, c, d);
CALL pr_add(1, "雷蛇哆啦 A 梦 50 周年限定版机械键盘", 499, 80);
CALL pr_add(2, "雷蛇哆啦 A 梦 50 周年限定版鼠标", 279, 70);
CALL pr_add(3, "雷蛇哆啦 A 梦 50 周年限定版耳机", 349, 60);
1.4 创建一个存储过程,名称自订,通过输入商品名称往表中插入 100 条记录。其中,所插入第一条记录的商品 ID 由现有表中商品 ID 的最大值+1 构成,后续记录中商品 ID 依次递增 1。所插入第一条记录的商品价格与现有表中商品价格的最小值相同,后续记录的商品价格依次递增 1。商品销售数量随机生成。
delimiter
, 因为 ; 会影响命令中语句的执行
CREATE PROCEDURE add100(name VARCHAR(20))
BEGIN
DECLARE a INT; # 商品 ID
DECLARE b DOUBLE; # 商品价格
DECLARE c INT; # 商品销售数量
DECLARE n INT; # 循环变量
#################################################
SELECT MAX(id) INTO a FROM salerecords; # ID 最大值
SET a = a + 1; # 第一个商品 ID 为 ID 最大值 + 1
#################################################
SELECT MIN(price) INTO b FROM salerecords; # 第一个商品价格为价格最小值
#################################################
SET n = 0; # 循环 100 次
WHILE n < 100 DO
SET c = rand(); # 商品销售数量随机
INSERT INTO salerecords VALUES(a, name, b, c);
SET a = a + 1; # 后续商品 ID+1
SET b = b + 1; # 后续商品价格+1
SET n = n + 1;
END WHILE;
END
$$
delimiter ; # 将分隔符修改回 ;
调用存储过程:
CALL add100("C++ Primer Plus");
表中前 10 条查询结果:
SELECT * FROM salerecords limit 0, 10;
补充:删除存储过程:
ALTER PROCEDURE add100
1.5 创建一个存储过程,名称自定,通过输入商品 ID 的最小值,最大值,将表中商品 ID 处于(最小值,最大值)范围内,且为偶数的记录删除,包括最小值与最大值本身。此处不考虑商品 ID 不存在的情况,在实验时请选择 1.4 中创建的商品 ID 范围的子集。
delimiter
, 因为 ; 会影响命令中语句的执行
CREATE PROCEDURE dele
teEven(a int, b int)
BEGIN
DELETE FROM salerecords
WHERE MOD(id,2)=0 # 偶数
AND id BETWEEN a AND b;
END
$$
delimiter ; # 将分隔符修改回 ;
CALL deleteEven(1,10); # 删除前 10 条数据中的偶数数据
SELECT * FROM salerecords LIMIT 0,10; # 分页查询前 10 条数据
1.6 创建两个事件调度器,第一个每 3 秒往 1.2 的表中插入一条记录,第二个每 30 秒清空 1.2 的表中的所有记录。
注意事项:
全局的 event 是关闭的:通过下列语句查看是否开启
SHOW variables LIKE 'event_scheduler';
如未开启,通过下列语句开启:
SET GLOBAL event_scheduler = 1;
CREATE EVENT sale_insert
ON SCHEDULE EVERY 3 SECOND
DO INSERT INTO salerecords(name, price, number) VALUES (
'海洋之心', 888, 10
);
CREATE EVENT sale_delete
ON SCHEDULE EVERY 30 SECOND
DO DELETE FROM salerecords;
每过 3 秒查询一下表中数据数量,可以看到确实发生了变化:
SELECT COUNT(*) FROM salerecords;
==============================================================================
定义一个触发器,实现如下功能,在往 1.2 的表中插入记录的时候,将记录同时也插入到一张新的表 sale_backup.
CREATE TABLE sale_backup(id INT,name VARCHAR(20),price INT,number INT);
CREATE TRIGGER tri AFTER INSERT
ON salerecords FOR EACH ROW
INSERT INTO sale_backup VALUES (new.id, new.name, new.price, new.number);
SELECT * FROM sale_backup LIMIT 0,10;
===============================================================================
3.1 启动一个事务往 1.2 的表中插入任意三条记录,提交在第二条和第三条记录中定义一个 savepoint,在插入完成后回滚到定义的 savepoint。
注:做这个任务时要把前面定义的 EVENT 删除,顺便清空表,效果更明显;
DELETE EVENT sale_insert;
DELETE EVENT sale_delete;
DELETE FROM salerecords;
START TRANSACTION; # 开启事务
INSERT INTO salerecords VALUES (1000, '考研数学', 40, 3); # 插入数据 1
INSERT INTO salerecords VALUES (1001, '考研英语', 35, 3); # 插入数据 2
SAVEPOINT sp; # 保存点 sp
INSERT INTO salerecords VALUES (1003, '考研政治', 30, 2); # 插入数据 3
SELECT * FROM salerecords; # 此时应当有 3 个数据
ROLLBACK TO SAVEPOINT sp; # 回滚到保存点 sp
SELECT * FROM salerecords; # 此时应当有 2 个数据
COMMIT; # 提交事务
SELECT * FROM salerecords; # 此时应当只有 2 个数据
共享锁(S 锁) 又称 读锁。若事务 T 对数据对象 A 加上 S 锁,则事务 T 可以读 A 但不能修改 A,其他事务只能再对 A 加 S 锁,而不能加 X 锁,直到 T 释放 A 上的 S 锁。这保证了其他事务可以读 A,但在 T 释放 A 上的 S 锁之前不能对 A 做任何修改。
排他锁(X 锁) 又称 写锁。若事务 T 对数据对象 A 加上 X 锁,事务 T 可以读 A 也可以修改 A,其他事务不能再对 A 加任何锁,直到 T 释放 A 上的锁。这保证了其他事务在 T 释放 A 上的锁之前不能再读取和修改 A。
=============================================================================
评论