390 lines
15 KiB
TeX
390 lines
15 KiB
TeX
\documentclass[全部作业]{subfiles}
|
||
\input{mysubpreamble}
|
||
|
||
\begin{document}
|
||
\setcounter{chapter}{8}
|
||
\chapter{第九次作业}
|
||
|
||
\section*{数据库案例设计综合练习}
|
||
|
||
|
||
|
||
某金融数据库系统背景:
|
||
|
||
某银行为了建设金融管理系统,需要对该系统的数据库进行设计,针对该系统,主要将对象分为客户、银行卡、理财产品、保险、基金。假设该系统存在着以下关系:客户可以办理银行卡,同时客户可以购买不同的银行产品,如理财产品,基金和保险。购买后的银行产品统称为客户的资产。
|
||
|
||
|
||
|
||
数据库系统业务需求描述:
|
||
|
||
\begin{itemize}
|
||
\item 客户可以办理多张银行卡,不仅可以办理储蓄卡,如果符合要求还可以办理信用卡,客户办理成功后获得银行卡的卡号。
|
||
|
||
\item 银行提供多种理财产品,客户根据理财产品的信息(包括产品编号,产品名称,产品描述)购买理财产品,一个客户可以购买多种理财产品,不同的理财产品有不同的购买金额和理财年限。
|
||
|
||
\item 银行提供多种保险,客户根据保险的信息(包括保险编号,保险名称,适用人群,保险项目)购买理财产品,一个客户可以购买多种保险,不同的保险有不同的保险金额和保险年限。
|
||
|
||
\item 银行提供多种基金,客户根据基金的信息(包括基金编号,基金名称,基金类型,风险等级和基金管理者)购买理财产品,一个客户可以购买多种基金,不同的基金有不同的基金金额。
|
||
|
||
\item 客户购买银行产品(理财产品、保险、基金)成为自己的资产后,需要记录这些资产的状态(可用,冻结),每个资产的购买数量,每个资产的收益和购买时间。
|
||
\end{itemize}
|
||
|
||
|
||
|
||
项目具体要求:
|
||
|
||
\begin{itemize}
|
||
\renewcommand{\labelitemi}{-}
|
||
\item 创建finance数据库作为项目数据库,数据库编码为UTF-8。
|
||
|
||
\item 在finance模式下完成金融管理系统中所有数据库对象(数据表)的创建,并完成数据的填充(可以由navicat自动生成)。其中,客户数据不少于20条,银行卡数据不少于10条,其他数据不少于5条;
|
||
|
||
\item 对表添加外键约束,在银行信息表和资产信息表中,都存在每个银行卡必须有一个持卡者、每份资产必须都有一个资产拥有者这样的对应关系。因此针对这种对应关系,创建外键约束。
|
||
|
||
\item 在理财产品表、保险信息表和基金信息表中,都存在金额这个属性,在现实生活中,金额不会存在负数。因此针对表中金额的属性,增加大于0的约束条件。
|
||
|
||
\item 可以增加其他自定义约束。(可以用触发器来实现,不强制要求)
|
||
\end{itemize}
|
||
|
||
|
||
|
||
本次作业要求:
|
||
|
||
1. 分析系统功能,提交相应的E-R图(可以用画图软件(如visio)进行E-R图设计,或者手绘后拍照上传超星平台); (30分)
|
||
|
||
2. 根据设计完成的E-R图,转换对应的关系模式,包括对象的具体属性描述和对象之间的关系描述,需至少满足3NF要求;(30分)
|
||
|
||
3. 在MySQL中创建数据库,并加入适量的测试数据(某个关系截图,不用所有的关系都截图即可);(10分)
|
||
|
||
4. 根据需求创建合适的视图、存储过程、函数、触发器(提交代码);(20分)
|
||
|
||
5. 设计一个简单的用户界面,通过JDBC连接到该金融数据库,并测试完成简单的功能(测试代码和演示视频)。(10分)
|
||
|
||
|
||
|
||
注意:本次作业注重数据库设计(包括E-R图、关系模式转换,以及数据库实践SQL操作的综合练习);第5步仅做一个简单的UI界面即可(语言不限),不用太复杂。
|
||
|
||
\section*{解答}
|
||
|
||
\begin{enumerate}
|
||
\questionandanswer[]{
|
||
分析系统功能,提交相应的E-R图(可以用画图软件(如visio)进行E-R图设计,或者手绘后拍照上传超星平台); (30分)
|
||
}{
|
||
|
||
% \includesvgpdf[0.9]{9.1.1.drawio}
|
||
|
||
\includegraphics[width=0.9\linewidth]{svg-inkscape/9.1.1.png}
|
||
|
||
}
|
||
|
||
\questionandanswer[]{
|
||
根据设计完成的E-R图,转换对应的关系模式,包括对象的具体属性描述和对象之间的关系描述,需至少满足3NF要求;(30分)
|
||
}{
|
||
|
||
转换对应的关系模式为:
|
||
|
||
{
|
||
customer(\underbar{customer_id})
|
||
|
||
card(\underbar{card_id})
|
||
|
||
product(\underbar{product_id}, name, description, price, year)
|
||
|
||
insurance(\underbar{insurance_id}, name, appliacable_people, coverage, price, year)
|
||
|
||
fund(\underbar{fund_id}, name, type, risk_rating, manager, price)
|
||
|
||
customer_card(\underbar{customer_id}, \underbar{card_id}, status, number, profit, buying_time)
|
||
|
||
customer_product(\underbar{customer_id}, \underbar{product_id}, status, number, profit, buying_time)
|
||
|
||
customer_insurance(\underbar{customer_id}, \underbar{insurance_id}, status, number, profit, buying_time)
|
||
|
||
customer_fund(\underbar{customer_id}, \underbar{fund_id}, status, number, profit, buying_time)
|
||
}
|
||
|
||
检查是否满足第三范式。这里有个问题是,例如product,name是否能确定description, price, year,也就是是否允许同名产品存在。这里为了方便起见,就认为不存在这样的函数依赖,也就是允许同名产品存在。
|
||
|
||
那么这样的关系模式就满足第三范式了。
|
||
}
|
||
|
||
\questionandanswer[]{
|
||
在MySQL中创建数据库,并加入适量的测试数据(某个关系截图,不用所有的关系都截图即可);(10分)
|
||
}{}
|
||
{\kaishu
|
||
\begin{minted}{SQL}
|
||
create table customer(
|
||
customer_id int auto_increment primary key
|
||
);
|
||
|
||
create table card(
|
||
card_id int auto_increment primary key
|
||
);
|
||
|
||
create table product(
|
||
product_id int auto_increment primary key comment '产品编号',
|
||
name varchar(255) comment '产品名称',
|
||
description varchar(255) comment '产品描述',
|
||
price decimal(10, 2) comment '购买金额',
|
||
year year comment '理财年限'
|
||
);
|
||
|
||
create table insurance(
|
||
insurance_id int auto_increment primary key comment '保险编号',
|
||
name varchar(255) comment '保险名称',
|
||
applicable_people varchar(255) comment '适用人群',
|
||
coverage varchar(255) comment '保险项目',
|
||
price decimal(10, 2) comment '保险金额',
|
||
year year comment '保险年限'
|
||
);
|
||
|
||
create table fund(
|
||
fund_id int auto_increment primary key comment '基金编号',
|
||
name varchar(255) comment '基金名称',
|
||
type varchar(128) comment '基金类型',
|
||
risk_rating int(3) comment '风险等级',
|
||
manager varchar(64) comment '基金管理者',
|
||
price decimal(10, 2) comment '基金金额'
|
||
);
|
||
|
||
create table customer_card(
|
||
customer_id int,
|
||
card_id int,
|
||
status varchar(64) comment '状态',
|
||
number int comment '购买数量',
|
||
profit decimal(10, 2) comment '收益',
|
||
buying_time datetime comment '购买时间',
|
||
foreign key (customer_id) references customer(customer_id),
|
||
foreign key (card_id) references card(card_id)
|
||
);
|
||
|
||
create table customer_product(
|
||
customer_id int,
|
||
product_id int,
|
||
status varchar(64) comment '状态',
|
||
number int comment '购买数量',
|
||
profit decimal(10, 2) comment '收益',
|
||
buying_time datetime comment '购买时间',
|
||
foreign key (customer_id) references customer(customer_id),
|
||
foreign key (product_id) references product(product_id)
|
||
);
|
||
|
||
create table customer_insurance(
|
||
customer_id int,
|
||
insurance_id int,
|
||
status varchar(64) comment '状态',
|
||
number int comment '购买数量',
|
||
profit decimal(10, 2) comment '收益',
|
||
buying_time datetime comment '购买时间',
|
||
foreign key (customer_id) references customer(customer_id),
|
||
foreign key (insurance_id) references insurance(insurance_id)
|
||
);
|
||
|
||
create table customer_fund(
|
||
customer_id int,
|
||
fund_id int,
|
||
status varchar(64) comment '状态',
|
||
number int comment '购买数量',
|
||
profit decimal(10, 2) comment '收益',
|
||
buying_time datetime comment '购买时间',
|
||
foreign key (customer_id) references customer(customer_id),
|
||
foreign key (fund_id) references fund(fund_id)
|
||
);
|
||
|
||
\end{minted}
|
||
|
||
\noindent\includegraphics[width=1\linewidth]{imgs/2024-12-10-21-22-33.png}
|
||
}
|
||
\questionandanswer[]{
|
||
根据需求创建合适的视图、存储过程、函数、触发器(提交代码);(20分)
|
||
}{}
|
||
{\kaishu
|
||
外键应该已经创建好了。那还剩下金额了。
|
||
\begin{minted}{SQL}
|
||
alter table product add constraint _price_gt_0 check ( price > 0 );
|
||
alter table insurance add constraint _price_gt_0 check ( price > 0 );
|
||
alter table fund add constraint _price_gt_0 check ( price > 0 );
|
||
\end{minted}
|
||
}
|
||
\questionandanswer[]{
|
||
设计一个简单的用户界面,通过 JDBC 连接到该金融数据库,并测试完成简单的功能(测
|
||
试代码和演示视频)。(10 分)
|
||
}{}
|
||
{\kaishu
|
||
\begin{minted}{python}
|
||
# -*- coding: utf-8 -*-
|
||
|
||
################################################################################
|
||
# Form generated from reading UI file 'designerLjvxhA.ui'
|
||
##
|
||
# Created by: Qt User Interface Compiler version 5.15.2
|
||
##
|
||
# WARNING! All changes made in this file will be lost when recompiling UI file!
|
||
################################################################################
|
||
|
||
import sys
|
||
|
||
from PyQt5.QtCore import *
|
||
from PyQt5.QtGui import *
|
||
from PyQt5.QtWidgets import *
|
||
import pymysql
|
||
|
||
|
||
class Ui_MainWindow(object):
|
||
def setupUi(self, MainWindow):
|
||
if not MainWindow.objectName():
|
||
MainWindow.setObjectName(u"MainWindow")
|
||
MainWindow.resize(800, 600)
|
||
self.centralwidget = QWidget(MainWindow)
|
||
self.centralwidget.setObjectName(u"centralwidget")
|
||
self.verticalLayout_2 = QVBoxLayout(self.centralwidget)
|
||
self.verticalLayout_2.setObjectName(u"verticalLayout_2")
|
||
self.horizontalLayout = QHBoxLayout()
|
||
self.horizontalLayout.setObjectName(u"horizontalLayout")
|
||
self.label = QLabel(self.centralwidget)
|
||
self.label.setObjectName(u"label")
|
||
|
||
self.horizontalLayout.addWidget(self.label)
|
||
|
||
self.host_edit = QLineEdit(self.centralwidget)
|
||
self.host_edit.setObjectName(u"host_edit")
|
||
|
||
self.horizontalLayout.addWidget(self.host_edit)
|
||
|
||
self.label_2 = QLabel(self.centralwidget)
|
||
self.label_2.setObjectName(u"label_2")
|
||
|
||
self.horizontalLayout.addWidget(self.label_2)
|
||
|
||
self.user_edit = QLineEdit(self.centralwidget)
|
||
self.user_edit.setObjectName(u"user_edit")
|
||
|
||
self.horizontalLayout.addWidget(self.user_edit)
|
||
|
||
self.label_3 = QLabel(self.centralwidget)
|
||
self.label_3.setObjectName(u"label_3")
|
||
|
||
self.horizontalLayout.addWidget(self.label_3)
|
||
|
||
self.password_edit = QLineEdit(self.centralwidget)
|
||
self.password_edit.setObjectName(u"password_edit")
|
||
|
||
self.horizontalLayout.addWidget(self.password_edit)
|
||
|
||
self.label_4 = QLabel(self.centralwidget)
|
||
self.label_4.setObjectName(u"label_4")
|
||
|
||
self.horizontalLayout.addWidget(self.label_4)
|
||
|
||
self.database_edit = QLineEdit(self.centralwidget)
|
||
self.database_edit.setObjectName(u"database_edit")
|
||
|
||
self.horizontalLayout.addWidget(self.database_edit)
|
||
|
||
self.connect_button = QPushButton(self.centralwidget)
|
||
self.connect_button.setObjectName(u"connect_button")
|
||
|
||
self.horizontalLayout.addWidget(self.connect_button)
|
||
|
||
|
||
self.verticalLayout_2.addLayout(self.horizontalLayout)
|
||
|
||
self.verticalLayout = QVBoxLayout()
|
||
self.verticalLayout.setObjectName(u"verticalLayout")
|
||
self.sql_edit = QPlainTextEdit(self.centralwidget)
|
||
self.sql_edit.setObjectName(u"sql_edit")
|
||
|
||
self.verticalLayout.addWidget(self.sql_edit)
|
||
|
||
self.horizontalLayout_2 = QHBoxLayout()
|
||
self.horizontalLayout_2.setObjectName(u"horizontalLayout_2")
|
||
self.submit_button = QPushButton(self.centralwidget)
|
||
self.submit_button.setObjectName(u"submit_button")
|
||
sizePolicy = QSizePolicy(QSizePolicy.Maximum, QSizePolicy.Minimum)
|
||
sizePolicy.setHorizontalStretch(0)
|
||
sizePolicy.setVerticalStretch(0)
|
||
sizePolicy.setHeightForWidth(self.submit_button.sizePolicy().hasHeightForWidth())
|
||
self.submit_button.setSizePolicy(sizePolicy)
|
||
|
||
self.horizontalLayout_2.addWidget(self.submit_button)
|
||
|
||
|
||
self.verticalLayout.addLayout(self.horizontalLayout_2)
|
||
|
||
|
||
self.verticalLayout_2.addLayout(self.verticalLayout)
|
||
|
||
self.result_text = QTextBrowser(self.centralwidget)
|
||
self.result_text.setObjectName(u"result_text")
|
||
|
||
self.verticalLayout_2.addWidget(self.result_text)
|
||
|
||
MainWindow.setCentralWidget(self.centralwidget)
|
||
self.menubar = QMenuBar(MainWindow)
|
||
self.menubar.setObjectName(u"menubar")
|
||
self.menubar.setGeometry(QRect(0, 0, 800, 23))
|
||
MainWindow.setMenuBar(self.menubar)
|
||
self.statusbar = QStatusBar(MainWindow)
|
||
self.statusbar.setObjectName(u"statusbar")
|
||
MainWindow.setStatusBar(self.statusbar)
|
||
|
||
self.retranslateUi(MainWindow)
|
||
|
||
QMetaObject.connectSlotsByName(MainWindow)
|
||
|
||
self.setup_other()
|
||
# setupUi
|
||
|
||
def retranslateUi(self, MainWindow):
|
||
MainWindow.setWindowTitle(QCoreApplication.translate("MainWindow", u"MainWindow", None))
|
||
self.label.setText(QCoreApplication.translate("MainWindow", u"\u4e3b\u673a", None))
|
||
self.label_2.setText(QCoreApplication.translate("MainWindow", u"\u7528\u6237\u540d", None))
|
||
self.label_3.setText(QCoreApplication.translate("MainWindow", u"\u5bc6\u7801", None))
|
||
self.label_4.setText(QCoreApplication.translate("MainWindow", u"\u6570\u636e\u5e93\u540d", None))
|
||
self.connect_button.setText(QCoreApplication.translate("MainWindow", u"\u8fde\u63a5", None))
|
||
self.submit_button.setText(QCoreApplication.translate("MainWindow", u"\u63d0\u4ea4", None))
|
||
# retranslateUi
|
||
|
||
def setup_other(self):
|
||
self.host_edit.setText("localhost")
|
||
self.user_edit.setText("test_user2")
|
||
self.password_edit.setText("Rie4OhYi")
|
||
self.database_edit.setText("finance")
|
||
self.connect_button.clicked.connect(self.on_connect_clicked)
|
||
self.submit_button.clicked.connect(self.on_submit_clicked)
|
||
|
||
def on_connect_clicked(self):
|
||
# 打开数据库连接
|
||
self.conn = pymysql.connect(
|
||
host=self.host_edit.text(), user=self.user_edit.text(), passwd=self.password_edit.text())
|
||
self.conn.select_db(self.database_edit.text())
|
||
self.result_text.setText("连接成功")
|
||
|
||
def on_submit_clicked(self):
|
||
cur = self.conn.cursor()
|
||
cur.execute(self.sql_edit.toPlainText())
|
||
all_result = ""
|
||
while 1:
|
||
res=cur.fetchone()
|
||
if res is None:
|
||
#表示已经取完结果集
|
||
break
|
||
all_result += str(res) + "\n"
|
||
self.result_text.setText(all_result)
|
||
|
||
|
||
if __name__ == '__main__':
|
||
app = QApplication(sys.argv)
|
||
window = QMainWindow()
|
||
main_window = Ui_MainWindow()
|
||
main_window.setupUi(window)
|
||
window.show()
|
||
app.exec_()
|
||
|
||
\end{minted}
|
||
|
||
\noindent\includegraphics[width=1\linewidth]{imgs/2024-12-13-21-18-22.png}
|
||
}
|
||
\end{enumerate}
|
||
|
||
\end{document} |