MySQL数据库基础练习系列:科研项目管理系统

DDL

CREATE TABLE Users (user_id INT AUTO_INCREMENT PRIMARY KEY COMMENT '用户ID',username VARCHAR(50) NOT NULL UNIQUE COMMENT '用户名',password VARCHAR(255) NOT NULL COMMENT '密码',gender ENUM('男', '女') NOT NULL COMMENT '性别',email VARCHAR(100) UNIQUE COMMENT '邮箱'
);CREATE TABLE Roles (role_id INT AUTO_INCREMENT PRIMARY KEY COMMENT '角色ID',role_name VARCHAR(50) NOT NULL UNIQUE COMMENT '角色名称'
);CREATE TABLE UserRoles (user_id INT COMMENT '用户ID',role_id INT COMMENT '角色ID',PRIMARY KEY (user_id, role_id),FOREIGN KEY (user_id) REFERENCES Users(user_id),FOREIGN KEY (role_id) REFERENCES Roles(role_id)
);CREATE TABLE Projects (project_id INT AUTO_INCREMENT PRIMARY KEY COMMENT '项目ID',project_name VARCHAR(100) NOT NULL COMMENT '项目名称',project_description TEXT COMMENT '项目描述',principal_investigator_id INT COMMENT '主研人ID',start_date DATE NOT NULL COMMENT '开始日期',end_date DATE NOT NULL COMMENT '结束日期',status ENUM('申请中', '审批中', '执行中', '结题') NOT NULL COMMENT '项目状态',FOREIGN KEY (principal_investigator_id) REFERENCES Users(user_id)
);CREATE TABLE Funds (fund_id INT AUTO_INCREMENT PRIMARY KEY COMMENT '资金ID',project_id INT NOT NULL COMMENT '项目ID',source VARCHAR(100) NOT NULL COMMENT '资金来源',amount DECIMAL(10, 2) NOT NULL COMMENT '资金金额',FOREIGN KEY (project_id) REFERENCES Projects(project_id)
);CREATE TABLE Achievements (achievement_id INT AUTO_INCREMENT PRIMARY KEY COMMENT '成果ID',project_id INT NOT NULL COMMENT '项目ID',achievement_name VARCHAR(100) NOT NULL COMMENT '成果名称',achievement_type ENUM('论文', '专利', '获奖', '其他') NOT NULL COMMENT '成果类型',description TEXT COMMENT '成果描述',FOREIGN KEY (project_id) REFERENCES Projects(project_id)
);CREATE TABLE ProjectLogs (log_id INT AUTO_INCREMENT PRIMARY KEY COMMENT '日志ID',project_id INT NOT NULL COMMENT '项目ID',user_id INT NOT NULL COMMENT '用户ID',log_date DATETIME NOT NULL COMMENT '日志日期',log_content TEXT NOT NULL COMMENT '日志内容',FOREIGN KEY (project_id) REFERENCES Projects(project_id),FOREIGN KEY (user_id) REFERENCES Users(user_id)
);

DML

INSERT INTO Roles (role_name) VALUES
('管理员'),
('项目负责人'),
('项目成员');
INSERT INTO Users (username, password, gender, email) VALUES
('诸葛亮', '123', '男', 'zhugeliang@example.com'),
('孙悟空', '123', '男', 'sunwukong@example.com'),
('林黛玉', '123', '女', 'lindaiyu@example.com');
INSERT INTO UserRoles (user_id, role_id) VALUES
(1, 1), -- 诸葛亮是管理员
(2, 2), -- 孙悟空是项目负责人
(2, 3), -- 孙悟空也是项目成员
(3, 3);  -- 林黛玉是项目成员
INSERT INTO Projects (project_name, project_description, principal_investigator_id, start_date, end_date, status) VALUES
('三国历史研究项目', '研究三国历史背景', 1, '2023-01-01', '2023-12-31', '执行中'),
('西游记文化研究', '探究西游记的文学价值', 2, '2023-02-01', '2024-01-31', '申请中'),
('红楼梦解读', '分析红楼梦的深层含义', 2, '2023-03-01', '2023-11-30', '审批中');
INSERT INTO Funds (project_id, source, amount) VALUES
(1, '国家社会科学基金', 50000.00),
(2, '企业赞助', 30000.00),
(3, '学校科研基金', 45000.00),
(1, '地方政府资助', 20000.00); -- 同一个项目可以有多个经费来源
INSERT INTO Achievements (project_id, achievement_name, achievement_type, description) VALUES
(1, '三国历史研究报告', '论文', '详细分析了三国时期的历史事件'),
(2, '西游记文化解读', '论文', '深入探讨了西游记的文化内涵'),
(3, '红楼梦人物分析', '论文', '对红楼梦中的主要人物进行了深入剖析'),
(2, '西游记新发现', '专利', '发现了西游记中的新文学元素'); -- 同一个项目可以有多个成果
INSERT INTO ProjectLogs (project_id, user_id, log_date, log_content) VALUES
(1, 1, '2023-01-10 10:00:00', '项目启动会议召开'),
(2, 2, '2023-02-15 15:30:00', '提交项目申请书至学院'),
(3, 3, '2023-03-20 09:45:00', '开始收集红楼梦相关资料'),
(1, 1, '2023-04-01 14:15:00', '第一阶段研究成果汇报');

ER图 

 ER图

 模型图

简单查询

一、查询用户信息,仅显示用户的姓名与项目名称,用中文显示列名

SELECT DISTINCTu.username AS 用户名,p.project_name AS 项目名称
FROMUsers u
JOINProjects p ON u.user_id = p.principal_investigator_id;

二、根据项目名称进行模糊查询,模糊查询要进行索引,需要给出explain语句

EXPLAIN SELECT project_id, project_name
FROM Projects
WHERE project_name LIKE '%三国%';

三、统计用户的项目信息,查询所有用户的项目数量,并进行倒序排列

SELECT u.username AS 用户名,COUNT(p.project_id) AS 项目数量
FROM Users u
LEFT JOIN Projects p ON u.user_id = p.principal_investigator_id
GROUP BY u.user_id, u.username
ORDER BY 项目数量 DESC;

复杂查询

一、查询用户的基本信息,项目信息

SELECT u.user_id,u.username,u.gender,u.email,p.project_id,p.project_name,p.project_description,p.start_date,p.end_date,p.status
FROM Users u
LEFT JOIN Projects p ON u.user_id = p.principal_investigator_id;

二、查看项目中项目阶段最多的项目对应的类型

SELECT p.project_name,p.status
FROM Projects p
WHERE (SELECT COUNT(*) FROM Projects p2 WHERE p2.status = p.status) = (SELECT MAX(cnt) FROM (SELECT status, COUNT(*) as cnt FROM Projects GROUP BY status) as subquery);

三、查询项目最多的用户,并且查询用户的全部信息与当前项目阶段

SET @MostProjectsUserId = (SELECT principal_investigator_idFROM ProjectsGROUP BY principal_investigator_idORDER BY COUNT(*) DESCLIMIT 1
);SELECT u.*,p.project_id,p.project_name,p.status AS current_project_status
FROM Users u
JOIN Projects p ON u.user_id = p.principal_investigator_id
WHERE u.user_id = @MostProjectsUserId;

触发器

触发器一:项目状态更新时记录日志

DELIMITER $$
CREATE TRIGGER trg_after_project_status_update
AFTER UPDATE ON Projects
FOR EACH ROW
BEGINIF NEW.status <> OLD.status THENINSERT INTO ProjectLogs (project_id, user_id, log_date, log_content)VALUES (NEW.project_id, @CURRENT_USER_ID, NOW(), '项目状态已更新');END IF;
END;
$$
DELIMITER ;

触发器二:用户角色变更时记录日志

DELIMITER $$
CREATE TRIGGER trg_after_fund_insert
AFTER INSERT ON Funds
FOR EACH ROW
BEGININSERT INTO ProjectLogs (project_id, user_id, log_date, log_content)VALUES (NEW.project_id, @CURRENT_USER_ID, NOW(), CONCAT('项目获得资金:', NEW.source, ',金额为:', NEW.amount));
END;
$$
DELIMITER ;

触发器三:用户角色变更时记录日志

CREATE TABLE UserRoleLogs (log_id INT AUTO_INCREMENT PRIMARY KEY COMMENT '日志ID',user_id INT NOT NULL COMMENT '用户ID',old_role_id INT COMMENT '旧角色ID',new_role_id INT COMMENT '新角色ID',log_date DATETIME NOT NULL COMMENT '日志日期',log_content TEXT COMMENT '日志内容',FOREIGN KEY (user_id) REFERENCES Users(user_id),FOREIGN KEY (old_role_id) REFERENCES Roles(role_id),FOREIGN KEY (new_role_id) REFERENCES Roles(role_id)
);DELIMITER $$
CREATE TRIGGER trg_after_user_role_change
AFTER INSERT ON UserRoles
FOR EACH ROW
BEGINDECLARE old_role_name VARCHAR(50);DECLARE new_role_name VARCHAR(50);SELECT role_name INTO new_role_name FROM Roles WHERE role_id = NEW.role_id;IF new_role_name IS NOT NULL THENINSERT INTO UserRoleLogs (user_id, new_role_id, log_date, log_content)VALUES (NEW.user_id, NEW.role_id, NOW(), CONCAT('用户', NEW.user_id, '被赋予了新角色:', new_role_name));END IF;
END;
$$
DELIMITER ;

存储过程 

存储过程 1: 分配用户角色

DELIMITER $$
CREATE PROCEDURE AssignUserRole(IN userId INT, IN roleName VARCHAR(50))
BEGINDECLARE roleId INT;DECLARE CONTINUE HANDLER FOR NOT FOUND SET @errorMsg = 'Role not found.';SELECT role_id INTO roleId FROM Roles WHERE role_name = roleName;IF roleId IS NULL THENSIGNAL SQLSTATE '45000' SET MESSAGE_TEXT = @errorMsg;END IF;IF NOT EXISTS (SELECT 1 FROM UserRoles WHERE user_id = userId AND role_id = roleId) THENINSERT INTO UserRoles (user_id, role_id) VALUES (userId, roleId);END IF;SELECT 'Role assigned successfully.' AS message;
END $$
DELIMITER ;

存储过程 2: 更新项目状态并记录日志

DELIMITER $$
CREATE PROCEDURE UpdateProjectStatus(IN projectId INT, IN newStatus ENUM('申请中', '审批中', '执行中', '结题'))
BEGINDECLARE OLD_STATUS ENUM('申请中', '审批中', '执行中', '结题');SELECT status INTO OLD_STATUS FROM Projects WHERE project_id = projectId;UPDATE Projects SET status = newStatus WHERE project_id = projectId;INSERT INTO ProjectLogs (project_id, user_id, log_date, log_content)VALUES (projectId, USER(), NOW(), CONCAT('Project status changed from ', OLD_STATUS, ' to ', newStatus));SELECT 'Project status updated successfully.' AS message;
END $$
DELIMITER ;

存储过程 3: 分配项目资金并检查预算

ALTER TABLE Projects ADD total_budget DECIMAL(10, 2) NOT NULL DEFAULT 0 COMMENT '项目总预算';
DELIMITER $$
CREATE PROCEDURE AllocateProjectFunds(IN p_project_id INT,IN p_source VARCHAR(100),IN p_amount DECIMAL(10, 2)
)
BEGINDECLARE v_total_budget DECIMAL(10, 2);DECLARE v_allocated_funds DECIMAL(10, 2);SELECT total_budget INTO v_total_budget FROM Projects WHERE project_id = p_project_id;SELECT SUM(amount) INTO v_allocated_funds FROM Funds WHERE project_id = p_project_id;IF v_total_budget >= (v_allocated_funds + p_amount) THENINSERT INTO Funds (project_id, source, amount) VALUES (p_project_id, p_source, p_amount);INSERT INTO ProjectLogs (project_id, user_id, log_date, log_content)VALUES (p_project_id, USER(), NOW(),CONCAT('资金已成功分配给项目', p_project_id, ',金额:', p_amount, ',来源:', p_source));SELECT '资金分配成功' AS message;ELSESELECT '预算不足,无法分配资金' AS message;END IF;
END $$
DELIMITER ;

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

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

相关文章

【杂记-浅谈VRRP虚拟路由器冗余协议】

一、VRRP协议概述 VRRP&#xff0c;Virtual Router Redundancy Protocol&#xff0c;即虚拟路由器冗余协议&#xff0c;是一种用于提高网络可靠性和容错能力的协议。它能够在多个路由器之间共享一个虚拟IP地址&#xff0c;当主路由器失效时&#xff0c;备用路由器可以接管虚拟…

字节码编程ASM之idea插件asm bytecode outline的使用

写在前面 直接用ASM来编写字节码程序难度其实还是蛮大的&#xff0c;为此&#xff0c;就有热心人事开发了相关的idea插件 &#xff0c;其中比较优秀的一个是asm bytecode outline,本文就来一起看下如何使用。 1&#xff1a;安装 file->setting->plugins,搜索asm bytec…

gin-vue-amdin 新增路由

1&#xff1a;在api目录的example 下新建controller 层如下图&#xff08;&#xff09;&#xff1a; 在enter.go 中 加入 这个新建的结构体&#xff1a; 2&#xff1a;在router 的example 文件夹下 新建对应的路由文件 3&#xff1a;在initlize 的router 中 添加对应的代码&a…

PDF处理篇:有哪些免费的PDF注释工具

PDF 是一种功能强大的格式&#xff0c;广泛用于处理和传输数据。您可以创建自己的 PDF 文件&#xff0c;也可以使用其他人创建的 PDF 文件。但是&#xff0c;有时您想在 PDF 文件中包含其他文本、图形和其他元素。这就是 PDF 注释器为您提供帮助的地方。 有许多可用的 PDF 注释…

Bazel与Gradle工具差异

之前介绍Bazel文章中有同学闻到Bazel与Gradle工具的差异。这篇文章我们解答这个问题。 来自Bazel员工的说法 Bazel和Gradle强调构建体验的不同方面。在某种程度上,它们的侧重点是互斥的——Gradle对灵活性和非突出性的要求对它的构建结构进行了限制,而Bazel对可靠性和性能的…

无线领夹麦克风品牌排名,揭秘哪种领夹麦性价比高!

在直播电商和Vlog的热潮推动下&#xff0c;自媒体内容创作迎来了前所未有的繁荣。麦克风行业也因应这一趋势&#xff0c;迎来了快速的增长期。特别是无线领夹麦克风&#xff0c;以其便携性和高效的录音能力&#xff0c;迅速成为视频制作者的新宠。它不仅在直播带货和短视频制作…

allure安装教程

1、下载 allure的官网下载地址&#xff1a; https://github.com/allure-framework/allure2/releases 注意&#xff1a;官网时常访问失败&#xff0c;可以访问以下网址&#xff1a; https://repo.maven.apache.org/maven2/io/qameta/allure/allure-commandline/ 选择一个版本&…

Qt实现开机自启两种方法包含注意事项以及常见问题解决

Qt开机自启教程 Qt是一个跨平台的C++框架,用于开发图形用户界面应用程序,同时也支持其他类型的应用程序开发,如命令行工具、服务器、嵌入式设备应用程序等。用户在虚拟机开发后使用交叉编译器放到设备中设置开机自启,以下是两种常用的方法。 通过Systemd 设置开机自启 1…

C++ CMake 预定义宏与变量

宏/变量 说明 CMAKE_SOURCE_DIR 顶层 CMakeLists.txt 所在的目录 CMAKE_BINARY_DIR 构建输出目录的顶层路径 CMAKE_CURRENT_SOURCE_DIR 当前处理的 CMakeLists.txt 所在的路径 CMAKE_CURRENT_BINARY_DIR 当前处理的 CMakeLists.txt 输出的构建目录 CMAKE_CURRENT_LIS…

Uniapp的使用

为什么要使用uniapp uniapp 可以进行多端开发&#xff0c;uniapp 在设计的时候就拥有许多兼容性代码&#xff0c;可以兼容很多的平台 如 支付宝小程序 html页面 微信小程序等&#xff0c;注重开发效率而不是运行效率时 &#xff0c;就可以考虑一下 uniapp 当然也可以去…

深入浅出:MongoDB中的背景创建索引

深入浅出&#xff1a;MongoDB中的背景创建索引 想象一下&#xff0c;你正忙于将成千上万的数据塞入你的MongoDB数据库中&#xff0c;你的用户期待着实时的响应速度。此时&#xff0c;你突然想到&#xff1a;“嘿&#xff0c;我应该给这些查询加个索引&#xff01;” 没错&…

python实现链接数据库查询方法步骤

前言 在python项目开发中&#xff0c;我们经常使用多种数据库存储数据,常用的有&#xff1a; MySQL&#xff1a;mysql-connector-python, PyMySQL PostgreSQL&#xff1a;psycopg2 SQLite&#xff1a;sqlite3&#xff08;标准库&#xff09; 下面是一个通用的步骤示例&#xff…

ABAP ALV报表性能优化 经验总结

优化ALV报表&#xff0c;最主要就是优化取数逻辑和数据库查询。因为几乎在所有的程序中都会用到数据库查询&#xff0c;所以这篇文章的内容也不仅局限于SAP、ABAP程序&#xff0c;虽然ABAP有其特殊之处。 优化的时候我遵从以下几个原则&#xff1a; 1.把数据库连接视为一种极其…

CodeBlocks 导航栏/输出栏/菜单栏消失

参考地址&#xff1a;CodeBlocks 导航栏/输出栏/菜单栏消失_codeblocks工具栏没了-CSDN博客

Vivo手机怎么录屏?分享2种录屏方法

“新换的Vivo手机还挺好用的&#xff0c;但是今天看到一个视频想录下来保存&#xff0c;但找不到录屏功能啊&#xff0c;想问问大家Vivo手机的录屏功能怎么打开啊&#xff1f;还有Vivo手机能不能录制出高质量的视频呢&#xff1f;” 随着智能手机的普及&#xff0c;录屏功能已…

Redis 高并发缓存架构实战与性能优化

前置知识 1、缓存击穿、缓存失效的基本概念 2、什么样的 数据 需要加分布式锁 3、课上代码 05-一线大厂Redis高并发缓存架构实战与性能优化 对于 公司 中 简单的增删改查 做 高性能处理 &#xff0c; 采用递进的方式一步步优化。 普通Redis用法&#xff1a; 新增、修改、删…

Bazel构建工具介绍

Android开发我们主要用到的编译器是Java的Java编译器以及C/C++的gcc/clang编译器。对应Java层构建工具最初是Ant,后面Maven以及gradle;C/C++构建工具主要是make/cmake,做过系统开发或者chromeium,可能还用过ninjia、gclient、mm等。最近在编译Android平台的tensorflow lite…

Gobject tutorial 十一

参考&#xff1a;GObject – 2.0 Generic Value Container GValue结构体是一个变量容器,它包含一个类型成员和类型对应的值。其定义如下&#xff1a; /*** GValue:* * An opaque structure used to hold different types of values.** The data within the structure has pr…

ChatTTS源码部署

感谢阅读 默认已完成的操作准备工作下载源码安装依赖下载补丁(报错在运行) 界面展示(discord上有各种补丁&#xff0c;我的加了UI补丁和音色增强)提示词常用&#xff08;这个每个音基本都能生效&#xff09;语调类语速类情感类 默认已完成的操作 python版本>3.9 cuda版本的…

supersocket的app层源码解析

基础结构 #mermaid-svg-Q1MUtx0YWtvpRVNw {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-Q1MUtx0YWtvpRVNw .error-icon{fill:#552222;}#mermaid-svg-Q1MUtx0YWtvpRVNw .error-text{fill:#552222;stroke:#552222;}#…