SQL(四) 游标实验、存储过程、函数实验

1.将视图实验中score表中的数据通过以下命令复制到新建的表score_copy中,然后通过带有游标的存储过程对成绩按下面的规则进行更新:80~100,更改为5;60~80,更改为3;低于60分更新为0。

sql 复制代码
Create table score_copy as select * from score;
DELIMITER //
CREATE PROCEDURE update_scores()
BEGIN
    DECLARE done INT DEFAULT FALSE;
    DECLARE s_id INT;
    DECLARE s_score INT;
    DECLARE cur CURSOR FOR SELECT score_id, grade FROM score_copy;
    DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;

    OPEN cur;

    read_loop: LOOP
        FETCH cur INTO s_id,s_score;
        IF done THEN
            LEAVE read_loop;
        END IF;

        IF s_score >= 80 AND s_score <= 100 THEN
            UPDATE score_copy SET grade = 5 WHERE score_id = s_id;
        ELSEIF s_score >= 60 AND s_score < 80 THEN
            UPDATE score_copy SET grade = 3 WHERE score_id = s_id;
        ELSE
            UPDATE score_copy SET grade = 0 WHERE score_id = s_id;
        END IF;
    END LOOP;

    CLOSE cur;
END//

DELIMITER ;
CALL update_scores();
SELECT * FROM score_copy;

将视图实验中score表中每个学生得最高分的元组加入score_max(结构score)表中。score_max建表命令如下:

Create table score_max as select * from score where 1=2;

特别提示:要求编写存储过程后,用call命令调用过程,并用select命令显示scoremax表中的内容。

sql 复制代码
Create table score_max as select * from score where 1=2;
DELIMITER //

CREATE PROCEDURE insert_max_scores()
BEGIN
    DECLARE done INT DEFAULT FALSE;
    DECLARE s_id INT;
    DECLARE s_sno VARCHAR(10);
    DECLARE s_cno VARCHAR(10);
    DECLARE s_score VARCHAR(10);
    DECLARE s_max_score VARCHAR(10);
    DECLARE cur CURSOR FOR SELECT score_id,sno,cno,grade FROM score;
    DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = TRUE;

    OPEN cur;

    read_loop: LOOP
        FETCH cur INTO s_id,s_sno,s_cno,s_score;
        IF done THEN
            LEAVE read_loop;
        END IF;

        select max(grade) into s_max_score from score where sno=s_sno;

        IF s_score = s_max_score THEN
            INSERT INTO score_max(score_id,sno,cno,grade) VALUES (s_id,s_sno,s_cno,s_max_score);
        END IF;
    END LOOP;

    CLOSE cur;
END//

DELIMITER ;

CALL insert_max_scores();
SELECT * FROM score_max;

库中有表fibonacci,用来储存斐波拉契数列的前n项(Fibonacci表结构说明)

|------|--------|---------------|
| 列名 | 类型 | 说明 |
| n | int | 斐波拉契数列的第n项,主码 |
| fibn | bigint | 第n项的值 |

斐波拉契数列的前5项为:0,1,1,2,3推导公式为:fib(n) = fib(n-1) + fib(n-2)。

请创建存储过程sp_fibonacci(in m int),向表fibonacci插入斐波拉契数列的前m项,及其对应的斐波拉契数。fibonacci表初始值为一张空表。保证你的存储过程可以多次运行而不出错。

(斐波拉契数列的第一项索引号设为0,第二项为1,以此类推。测试时设置参数m为9)。

sql 复制代码
delimiter //
drop procedure sp_fibonacci;
create procedure sp_fibonacci(in m int)
begin
    DECLARE i INT DEFAULT 0;
    DECLARE n1 BIGINT DEFAULT 0;
    DECLARE n2 BIGINT DEFAULT 1;
    DECLARE temp BIGINT;
    truncate table fibonacci;
    INSERT INTO fibonacci (n, fibn) VALUES (0, 0);
    INSERT INTO fibonacci (n, fibn) VALUES (1, 1);
    SET i = 2;
    WHILE i < m DO
        SET temp = n1 + n2;
        SET n1 = n2;
        SET n2 = temp;
        INSERT INTO fibonacci (n, fibn) VALUES (i, temp);
        SET i = i + 1;
    END WHILE;
END //
DELIMITER ;
CALL sp_fibonacci(9);
select *from fibonacci;

设有基于图书馆数据库的4个基本表:

图书(书号,书名,作者,出版社,单价)

读者(读者号,姓名,性别,办公电话,部门,读者类别编号)

借阅(读者号,书号,借出日期,归还日期)

读者类别(读者类别编号,读者类别,允许借阅册数,允许借阅天数)

编写存储过程实现:根据读者号查询该读者的图书借阅的册数。(采用读者号"1205021"测试)

sql 复制代码
DELIMITER //
CREATE PROCEDURE checkBorrowLimit(IN in_reader_id VARCHAR(50), OUT out_result VARCHAR(20))
BEGIN
    DECLARE borrowed_count INT;
    DECLARE allowed_count INT;
    SELECT COUNT(*) INTO borrowed_count FROM 借阅 WHERE 读者号 = in_reader_id;
    SELECT 允许借阅册数 INTO allowed_count FROM 读者类别 WHERE 读者类别编号 = (SELECT 读者类别编号 FROM 读者 WHERE 读者号 = in_reader_id);
    IF borrowed_count < allowed_count THEN
        SET out_result = '可以继续借阅';
    ELSE
        SET out_result = '不可以继续借阅';
    END IF;
END//
DELIMITER ;
SET @result ='0';
CALL checkBorrowLimit('1205021', @result);
SELECT @result;

设有基于图书馆数据库的4个基本表:

图书(书号,书名,作者,出版社,单价)

读者(读者号,姓名,性别,办公电话,部门,读者类别编号)

借阅(读者号,书号,借出日期,归还日期)

读者类别(读者类别编号,读者类别,允许借阅册数,允许借阅天数)

编写存储过程实现:

读者借阅图书时,需要查询是否超出允许借阅册数,如果超出则不能继续借阅图书,

试创建存储过程完成此查询功能。(提示:输入参数:读者号;输出:"可以继续借阅"或"不可以继续借阅",

使用读者号"1205021"测试)

sql 复制代码
DELIMITER //
CREATE PROCEDURE checkBorrowLimit(IN in_reader_id VARCHAR(50), OUT out_result VARCHAR(20))
BEGIN
    DECLARE borrowed_count INT;
    DECLARE allowed_count INT;
    SELECT COUNT(*) INTO borrowed_count FROM 借阅 WHERE 读者号 = in_reader_id;
    SELECT 允许借阅册数 INTO allowed_count FROM 读者类别 WHERE 读者类别编号 = (SELECT 读者类别编号 FROM 读者 WHERE 读者号 = in_reader_id);
    IF borrowed_count < allowed_count THEN
        SET out_result = '可以继续借阅';
    ELSE
        SET out_result = '不可以继续借阅';
    END IF;
END//
DELIMITER ;
SET @result ='0';
CALL checkBorrowLimit('1205021', @result);
SELECT @result;

用create function语句创建符合以下要求的函数:依据读者号计算其所借图书数量。

函数名为:get_borrow_num。使用读者号"1205021"测试结果。

sql 复制代码
set global log_bin_trust_function_creators=1;
drop function IF EXISTS get_borrow_num;
delimiter $$
create function get_borrow_num(读者号 varchar(50))
returns int
begin
    declare borrow_num int;
    select count(*) into borrow_num from 借阅 where 借阅.读者号 = 读者号;
    return borrow_num;
end$$
delimiter ;
select get_borrow_num('1205021');
相关推荐
~我爱敲代码~2 小时前
使用XSHELL远程操作数据库
数据库·adb
一般社员2 小时前
Windows导入大型sql文件到mysql
windows·sql·mysql
春风霓裳8 小时前
sql-窗口函数
大数据·数据库·sql
言之。8 小时前
【数据库】TiDB 技术选型与架构分析报告
数据库·架构·tidb
人工智能训练9 小时前
如何在 Ubuntu 22.04 中安装 Docker 引擎和 Linux 版 Docker Desktop 桌面软件
linux·运维·服务器·数据库·ubuntu·docker·ai编程
胖头鱼的鱼缸(尹海文)9 小时前
数据库管理-第386期 使用OCP部署OceanBase 4.4.1社区版集群(20251107)
数据库·oceanbase
Craaaayon9 小时前
如何选择两种缓存更新策略(写缓存+异步写库;写数据库+异步更新缓存)
java·数据库·redis·后端·缓存·mybatis
一 乐9 小时前
点餐|智能点餐系统|基于java+ Springboot的动端的点餐系统小程序(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·小程序·论文
WarriorTan9 小时前
理解PostgreSQL中的数据块
数据库·postgresql
学好statistics和DS9 小时前
三个好思路:SQL并行化处理、混淆矩阵和特征交叉
数据库·sql·矩阵