Python操作MySQL数据库详细案例
Python操作MySQL数据库详细案例
- 一、前言
- 二、数据准备
- 三、建立数据库
- 四、处理和上传数据
- 五、下载数据
- 六、完整项目数据和代码
一、前言
本文通过案例讲解如何使用Python操作MySQL数据库。具体任务为:假设你已经了解MySQL和知识图谱标注工具Brat,将Brat标注的结果上传到MySQL。
在知识图谱的文本标注任务中,需要将数据按照事先决定的标注规则进行人工标注。Brat是一种比较著名的标注工具,但是目前不支持Windows系统,你可以安装虚拟机使用该工具。本文已经完成了一项基于Windows、Python3.7的标注软件制作工作,你可以点击 实体关系文本标注工具 进行查看。
二、数据准备
本文从知网下载了一些关于医学文章的摘要,提前设置了标注规则,例如实体类别、关系等。通过Brat工具标注该文本数据并得到输出标注好的结果文件,该文件共有2042行,以T开头表示实体,以R开头表示关系,该文件是对网络图结构的文本表述。
三、建立数据库
在开始Python代码任务之前,你需要对数据库有基础的了解,比如你已经具备使用SQL语言操作MySQL数据库的能力。
1、将MySQL中的操作定义为python函数(基于MySQL80),那么就可以直接调用该函数,来操作数据库。
import pandas as pd
import pymysqlprint("1")def mycursor(db_name = 'mysql80'):connection = pymysql.connect(host='localhost',user='root',port = 3308,password='',#123456database= db_name,charset='utf8mb4',cursorclass=pymysql.cursors.DictCursor)cursor = connection.cursor()return cursor, connectiondef use(db_name):'''切换数据库,返回游标'''return mycursor(db_name)def create_database(db_name):'''新建数据库'''sql = f'create database if not exists {db_name};'cursor.execute(sql)def create_table(tbl_name):'''新建数据表'''sql = f'create table if not exists {tbl_name};'cursor.execute(sql) def drop_database(db_name):'''删除数据库'''sql = f'drop database if exists {db_name};'cursor.execute(sql)def drop_table(tbl_name):'''删除数据表'''sql = f'drop table if exists {tbl_name};'cursor.execute(sql)def query(sql):'''以数据框形式返回查询据结果'''cursor.execute(sql)data = cursor.fetchall() # 以元组形式返回查询数据header = [t[0] for t in cursor.description]df = pd.DataFrame(list(data), columns=header) # pd.DataFrem 对列表具有更好的兼容性# return dfprint(df)def show_databases():'''查看服务器上的所有数据库'''sql = 'show databases;'return query(sql)def select_database():'''查看当前数据库'''sql = 'select database();'return query(sql)def show_tables():'''查看当前数据库中所有的表'''sql = 'show tables;'return query(sql)
2、创建数据库
#选择需要使用的数据库,此前需要创建数据库cursor, db = use('mysql') #默认设置,不要更改!dbname = 'kgg' #此处更改为需要创建的数据库名
create_database(dbname)
cursor, db = use(dbname)
3、建立数据库中各表
# 建实体类表
sqltb1 = ''' create table entity(id int auto_increment primary key comment '实体类编号',name varchar(20) comment '实体类名'
) comment '实体类表';
'''
cursor.execute(sqltb1)# 插入数据
sqlin1 = '''insert into entity values (1, '病症'),(2, '病名'),(3, '诊断方案'),(4, '治疗方案'),(5, '药名'),(6, '其它');
'''
cursor.execute(sqlin1)
db.commit() #事务
# 检查是否传递成功
sqlset = "select * from entity ;"
query(sqlset)#-------------------------------------------------------------------------# 创建关系表格
sqltb2 = ''' create table relation(id int auto_increment primary key comment '关系编号',name varchar(20) comment '关系名'
) comment '关系表';
'''
cursor.execute(sqltb2)#插入数据
sqlin2 = '''insert into relation values (1, '包含'),(2, '治疗'),(3, '危险因素'),(4, '辅助诊断'),(5, '特征'),(6, '并发'),(7, '别名'),(8, '作用'),(9, '条件');'''cursor.execute(sqlin2)
db.commit() #事务
#检查是否传递成功
sqlset = "select * from relation ;"
query(sqlset)#-------------------------------------------------------------------------# 创建实体表
sqltb3 = ''' create table entitymin(id int auto_increment primary key comment '编号',name varchar(50) comment '实体名'
) comment '实体表';
'''
cursor.execute(sqltb3)
db.commit() #事务#-------------------------------------------------------------------------
4、建立一些交叉表
#建立 实体-关系库,并设置主键外键关联# 创建实体表
sqltb5 = ''' create table entityrela(id int auto_increment primary key comment '编号',headclass int comment '头实体类',headentity int comment '头实体',relation int comment '关系',tailentity int comment '尾实体',tailclass int comment '尾实体类'
) comment '实体和关系表';
'''
cursor.execute(sqltb5)
db.commit() #事务#-------------------------------------------------------------------------
#创建链接该表(子表、外键)与另外三张表(父表、主键)链接sqlkey = '''
alter table entityrela add constraint encl_he_en_id foreign key (headclass) references entity (id);
'''
cursor.execute(sqlkey)sqlkey = '''
alter table entityrela add constraint enla_he_enmin_id foreign key (headentity) references entitymin (id);
'''
cursor.execute(sqlkey)sqlkey = '''
alter table entityrela add constraint enla_re_re_id foreign key (relation) references relation (id);
'''
cursor.execute(sqlkey)sqlkey = '''
alter table entityrela add constraint enla_ta_enmin_id foreign key (tailentity) references entitymin (id);
'''
cursor.execute(sqlkey)sqlkey = '''
alter table entityrela add constraint enla_ta_en_id foreign key (tailclass) references entity (id);
'''
cursor.execute(sqlkey)
db.commit() #事务#-------------------------------------------------------------------------
5、查看创建情况
# 查看该库中所有表格、查看表是否创建成功
show_tables()#-------------------------------------------------------------------------
# end
6、此外,若你在任何操作过程中,需要清空数据库,你可以:
# # #格式化数据库,取消下面三行注释,运行后便清空数据库。
# drop_database(dbname)
# create_database(dbname)
# print('数据库格式化')
四、处理和上传数据
创建好数据库后,便可以对数据进行处理,并将数据上传到数据库中。
1、同上,连接数据库,并定义工具函数。
# 将数据添加到数据库(数据库已经提前建好)import pandas as pd
import pymysql
import redef mycursor(db_name = 'mysql80'):connection = pymysql.connect(host='localhost',user='root',password='',#123456port = 3308,database= db_name,charset='utf8mb4',cursorclass=pymysql.cursors.DictCursor)cursor = connection.cursor()return cursor, connectiondef use(db_name):'''切换数据库,返回游标'''return mycursor(db_name)def query(sql):'''以数据框形式返回查询据结果'''cursor.execute(sql)data = cursor.fetchall() # 以元组形式返回查询数据header = [t[0] for t in cursor.description]df = pd.DataFrame(list(data), columns=header) # pd.DataFrem 对列表具有更好的兼容性return df# print(df)def show_databases():'''查看服务器上的所有数据库'''sql = 'show databases;'return query(sql)def select_database():'''查看当前数据库'''sql = 'select database();'return query(sql)def show_tables():'''查看当前数据库中所有的表'''sql = 'show tables;'return query(sql)cursor, db = use('mysql') #默认设置,不要更改!
dbname = 'kgg'
cursor, db = use(dbname)#-------------------------------------
2、数据处理-实体
# 将ann文件数据传入(暂以txt格式传入)
txtfile = open(r'C:\Users\DELL\Desktop\mysql数据库\0001-0500.ann', 'r', encoding='utf-8').readlines()
# print(txtfile[:3])#将实体与关系数据分开
entity = []
relation = []
for i in txtfile:if i[0] == 'T':entity.append(i)else:relation.append(i)# print(relation)#将实体名称、类别与编号提取出来
entity = [i.strip('\n').split('\t') for i in entity]
entity = [[i[0],i[1].split(' ')[0], i[-1]] for i in entity]
# print(entity[:5])
3、上传实体表中数据
#-------------------------------------------------------------------------
#在实体表中插入数据
# #提取实体名,去重;需要提前查看数据库已有的实体名称 需要表名称
entitymin = [i[-1] for i in entity]
entitymin = list(set(entitymin))
# print(entitymin[:5])for q in entitymin:sqlq = "select name from entitymin where name = (%s);"TF = cursor.execute(sqlq, q)if TF == 0: #存在则为1,不存在为0;不存在则添加 sqlin2 = "insert into entitymin values (null, %s) ;"cursor.execute(sqlin2, q)
db.commit() #事务
# 查看数据效果
sqlq = "select * from entitymin;"
query(sqlq)
4、数据处理-5元组关系
)#-------------------------------------------------------------------------
#将三元组提取出来
relation = [re.split("[\tA:' ']",i) for i in relation]
relation = [[i[1],i[4],i[7]] for i in relation]
# print(relation[:5])dicen = dict([('病症',1),('病名',2),('诊断方案',3),('治疗方案',4),('药名',5),('其它',6)])
dicre = dict([('包含',1),('治疗',2),('危险因素',3),('辅助诊断',4),('特征',5),('并发',6),('别名',7),('作用',8),('条件',9)])
# print(dicre['包含'])# #将三元组中的实体编号替换成实体名称
for r in relation:r[0] = dicre[r[0]]for e in entity:if r[1] == e[0]:r[1] = e[-1]r.insert(0,e[1])if r[-1] == e[0]:r[-1] = e[-1]r.append(e[1])
# print(relation[:5])#编码化---['头实体类', '头实体','关系类','尾实体', '尾实体类']
enre = []
for j in relation:j[0] = dicen[j[0]]j[-1] = dicen[j[-1]]sqlchaen = "select id from entitymin where name = (%s);"cursor.execute(sqlchaen, j[2])j[2] = cursor.fetchone()['id']sqlchaen = "select id from entitymin where name = (%s);"cursor.execute(sqlchaen, j[3])j[3] = cursor.fetchone()['id']enre.append([j[0],j[2],j[1],j[3],j[4]])
print(enre[-5:]) #传入实体-关系库
5、上传5元组数据
#插入实体-关系库数据,并创建链接该表(子表、外键)与另外三张表(父表、主键)链接# #插入数据
for en in enre:sqlin2 = "insert into entityrela values (null, %s, %s, %s, %s, %s) ;"cursor.execute(sqlin2, (en[0],en[1],en[2],en[3],en[4]))
db.commit() #事务
print('ok')#-------------------------------------------------------------------------
#清除重复数据并id排序
sql = '''
delete p1
from entityrela p1,entityrela p2
where (p1.headclass = p2.headclassand p1.headentity = p2.headentityand p1.relation = p2.relationand p1.tailentity = p2.tailentityand p1.tailclass = p2.tailclassand p1.id > p2.id);
'''
cursor.execute(sql)sql = '''
ALTER TABLE entityrela DROP id;
'''
cursor.execute(sql)
sql ='''
ALTER TABLE entityrela ADD id MEDIUMINT( 8 ) NOT NULL FIRST;
'''
cursor.execute(sql)
sql ='''
ALTER TABLE entityrela MODIFY COLUMN id MEDIUMINT( 8 ) NOT NULL AUTO_INCREMENT,ADD PRIMARY KEY(id);
'''
cursor.execute(sql)
db.commit() #事务
cursor.close()
#-------------------------------------------------------------------------
#end
其中,db.commit()
表示该事务要么全部成功,要么全部不执行。cursor.close()
表示关闭数据库。
6、在MySQL中查看执行结果
(1)在MySQL中查看所有数据库
(2)使用本文创建的kgg数据库,并查看库内所有表
(3)检索任意表,查看数据
以上3步表示数据库创建成功并成功写入数据。
五、下载数据
假设我们已经创建好数据库以及上传了数据。很久以后,我们需要从数据库中下载数据,进行分析或者分享给他人,那么本文在这里考虑了这一情况。
1、连接数据库,定义工具函数
# 从数据库中调取数据,以中文显示;import pandas as pd
import pymysqldef mycursor(db_name = 'mysql80'):connection = pymysql.connect(host='localhost',user='root',password='', #123456port = 3308,database= db_name,charset='utf8mb4',cursorclass=pymysql.cursors.DictCursor)cursor = connection.cursor()return cursor, connectiondef use(db_name):'''切换数据库,返回游标'''return mycursor(db_name)def query(sql):'''以数据框形式返回查询据结果'''cursor.execute(sql)data = cursor.fetchall() # 以元组形式返回查询数据header = [t[0] for t in cursor.description]df = pd.DataFrame(list(data), columns=header) # pd.DataFrem 对列表具有更好的兼容性return df# print(df)def show_databases():'''查看服务器上的所有数据库'''sql = 'show databases;'return query(sql)def select_database():'''查看当前数据库'''sql = 'select database();'return query(sql)def show_tables():'''查看当前数据库中所有的表'''sql = 'show tables;'return query(sql)
dbname = 'mysql'
cursor, db = use(dbname)
dbname = 'kgg'
cursor, db = use(dbname)# print(show_tables())
2、选择数据,并设计输出结构
#a.headentity ,
sql = '''select m.name from entityrela a left outer join entitymin m on a.headentity = m.id'''
i_s = list(query(sql)['name'])sql = '''select m.name from entityrela a left outer join entitymin m on a.tailentity = m.id'''
j_s = list(query(sql)['name'])sql = '''select m.name from entityrela a left outer join relation m on a.relation = m.id'''
k_s = list(query(sql)['name'])eql = '''select m.name from entityrela a left outer join entity m on a.headclass = m.id'''
ek_1 = list(query(eql)['name'])eq2 = '''select m.name from entityrela a left outer join entity m on a.tailclass = m.id'''
ek_2 = list(query(eq2)['name'])result = [[i,j,k,w,p] for i,j,k,w,p in zip(i_s,ek_1,k_s,ek_2,j_s)]
print(result[:5])
cursor.close()
3、设置保存文件地址并下载
f = open(r"C:\Users\DELL\Desktop\neo4j_python\data\4.csv",'w', encoding='utf-8')
for i in result:f.write('\''+i[0]+'\',\''+i[1]+'\',\''+i[2]+'\',\''+i[3]+'\',\''+i[4]+'\'\n')
f.close()
print('ok')
六、完整项目数据和代码
以下任意方式皆可:
1、评论、留言邮箱账号,博主定期回复。
2、点击:GitHub-python_mysql (或者访问: https://github.com/chenyiadam/python_mysql.git ) 进行下载
相关文章:

Python操作MySQL数据库详细案例
Python操作MySQL数据库详细案例一、前言二、数据准备三、建立数据库四、处理和上传数据五、下载数据六、完整项目数据和代码一、前言 本文通过案例讲解如何使用Python操作MySQL数据库。具体任务为:假设你已经了解MySQL和知识图谱标注工具Brat,将Brat标注…...
MicroBlaze系列教程(8):AXI_CAN的使用
文章目录 @[toc]CAN总线概述AXI_CAN简介MicroBlaze硬件配置常用函数使用示例波形实测参考资料工程下载本文是Xilinx MicroBlaze系列教程的第8篇文章。 CAN总线概述 **CAN(Controller Area Network)**是 ISO 国际标准化的串行通信协议,是由德国博世(BOSCH)公司在20世纪80年代…...
网络安全领域中八大类CISP证书
CISP注册信息安全专业人员 注册信息安全专业人员(Certified Information Security Professional),是经中国信息安全产品测评认证中心实施的国家认证,对信息安全人员执业资质的认可。该证书是面向信息安全企业、信息安全咨询服务…...

stm32学习笔记-5EXIT外部中断
5 EXIT外部中断 [toc] 注:笔记主要参考B站 江科大自化协 教学视频“STM32入门教程-2023持续更新中”。 注:工程及代码文件放在了本人的Github仓库。 5.1 STM32中断系统 图5-1 中断及中断嵌套示意图 中断 是指在主程序运行过程中,出现了特定…...

MySQL Workbench 图形化界面工具
Workbench 介绍 MySQL官方提供了一款免费的图形工具——MySQL Workbench,它是一款功能强大且易于使用的数据库设计、管理和开发工具,总之,MySQL Workbench是一款非常好用的MySQL图形工具,可以满足大多数MySQL用户的需求。 目录 W…...

雪花算法(SnowFlake)
简介现在的服务基本是分布式、微服务形式的,而且大数据量也导致分库分表的产生,对于水平分表就需要保证表中 id 的全局唯一性。对于 MySQL 而言,一个表中的主键 id 一般使用自增的方式,但是如果进行水平分表之后,多个表…...

Linux防火墙
一、Linux防火墙Linux的防火墙体系主要在网络层,针对TCP/IP数据包实施过滤和限制,属于典型的包过滤防火墙(或称为网络层防火墙)。Linux系统的防火墙体系基于内核编码实现,具有非常稳定的性能和极高的效率,因…...
网络安全系列-四十七: IP协议号大全
IP协议号列表 这是用在IPv4头部和IPv6头部的下一首部域的IP协议号列表。 十进制十六进制关键字协议引用00x00HOPOPTIPv6逐跳选项RFC 246010x01ICMP互联网控制消息协议(ICMP)RFC 79220x02IGMP...

HTTP协议格式以及Fiddler用法
目录 今日良言:焦虑和恐惧改变不了明天,唯一能做的就是把握今天 一、HTTP协议的基本格式 二、Fiddler的用法 1.Fidder的下载 2.Fidder的使用 今日良言:焦虑和恐惧改变不了明天,唯一能做的就是把握今天 一、HTTP协议的基本格式 先来介绍一下http协议: http 协议(全称为 &q…...

自动写代码?别闹了!
大家好,我是良许。 这几天,GitHub 上有个很火的插件在抖音刷屏了——Copilot。 这个神器有啥用呢?简单来讲,它就是一款由人工智能打造的编程辅助工具。 我们来看看它有啥用。 首先就是代码补全功能,你只要给出函数…...

项目心得--网约车
一、RESTFULPost:新增Put:全量修改Patch:修改某个值Delete: 删除Get:查询删除接口也可以用POST请求url注意:url中不要带有敏感词(用户id等)url中的名词用复数形式url设计:api.xxx.co…...

【二叉树广度优先遍历和深度优先遍历】
文章目录一、二叉树的深度优先遍历0.建立一棵树1. 前序遍历2.中序遍历3. 后序遍历二、二叉树的广度优先遍历层序遍历三、有关二叉树练习一、二叉树的深度优先遍历 学习二叉树结构,最简单的方式就是遍历。 所谓二叉树遍历(Traversal)是按照某种特定的规则ÿ…...

Spring Cloud微服务架构必备技术
单体架构 单体架构,也叫单体应用架构,是一个传统的软件架构模式。单体架构是指将应用程序的所有组件部署到一个单一的应用程序中,并统一进行部署、维护和扩展。在单体架构中,应用程序的所有功能都在同一个进程中运行,…...

TCP三次握手与四次挥手(一次明白)
TCP基本信息 默认端口号:80 LINUX中TIME_WAIT的默认时间是30s TCP三次握手 三次握手过程:每行代表发起握手到另一方刚刚收到数据包时的状态 客户端服务端客户端状态服务端状态握手前CLOSELISTEN客户端发送带有SYN标志的数据包到服务端一次握手SYN_SENDLISTEN二次握手服务端发送…...

pyside6@Mouse events实例@QApplication重叠导致的报错@keyboardInterrupt
文章目录报错内容鼠标事件演示报错内容 在pyside图形界面应用程序开发过程中,通常只允许运行一个实例 假设您重复执行程序A,那么可能会导致一些意向不到的错误并且,从python反馈的信息不容易判断错误的真正来源 鼠标事件演示 下面是一段演示pyside6的鼠标事件mouseEvent对象…...

订单30分钟未支付自动取消怎么实现?
目录了解需求方案 1:数据库轮询方案 2:JDK 的延迟队列方案 3:时间轮算法方案 4:redis 缓存方案 5:使用消息队列了解需求在开发中,往往会遇到一些关于延时任务的需求。例如生成订单 30 分钟未支付࿰…...

< 开源项目框架:推荐几个开箱即用的开源管理系统 - 让开发不再复杂 >
文章目录👉 SCUI Admin 中后台前端解决方案👉 Vue .NetCore 前后端分离的快速发开框架👉 next-admin 适配移动端、pc的后台模板👉 django-vue-admin-pro 快速开发平台👉 Admin.NET 通用管理平台👉 RuoYi 若…...

内网渗透-基础环境
解决依赖,scope安装 打开要给cmd powershell 打开远程 Set-ExecutionPolicy RemoteSigned -scope CurrentUser; 我试了好多装这东西还是得科学上网,不然不好用 iwr -useb get.scoop.sh | iex 查看下载过的软件 安装sudo 安装git 这里一定要配置bu…...

Go语言学习的第一天(对于Go学习的认识和工具选择及环境搭建)
首先学习一门新的语言,我们要知道这门语言可以帮助我们做些什么?为什么我们要学习这门语言?就小wei而言学习这门语言是为了区块链,因为自身是php出身,因为php的一些特性只能通过一些算法模拟的做一个虚拟链,…...
C和C++到底有什么关系
C++ 读作”C加加“,是”C Plus Plus“的简称。顾名思义,C++是在C的基础上增加新特性,玩出了新花样,所以叫”C Plus Plus“,就像 iPhone 6S 和 iPhone 6、Win10 和 Win7 的关系。 C语言是1972年由美国贝尔实验室研制成功的,在当时算是高级语言,它的很多新特性都让汇编程序…...
云原生核心技术 (7/12): K8s 核心概念白话解读(上):Pod 和 Deployment 究竟是什么?
大家好,欢迎来到《云原生核心技术》系列的第七篇! 在上一篇,我们成功地使用 Minikube 或 kind 在自己的电脑上搭建起了一个迷你但功能完备的 Kubernetes 集群。现在,我们就像一个拥有了一块崭新数字土地的农场主,是时…...
Ubuntu系统下交叉编译openssl
一、参考资料 OpenSSL&&libcurl库的交叉编译 - hesetone - 博客园 二、准备工作 1. 编译环境 宿主机:Ubuntu 20.04.6 LTSHost:ARM32位交叉编译器:arm-linux-gnueabihf-gcc-11.1.0 2. 设置交叉编译工具链 在交叉编译之前&#x…...

2025年能源电力系统与流体力学国际会议 (EPSFD 2025)
2025年能源电力系统与流体力学国际会议(EPSFD 2025)将于本年度在美丽的杭州盛大召开。作为全球能源、电力系统以及流体力学领域的顶级盛会,EPSFD 2025旨在为来自世界各地的科学家、工程师和研究人员提供一个展示最新研究成果、分享实践经验及…...

(二)原型模式
原型的功能是将一个已经存在的对象作为源目标,其余对象都是通过这个源目标创建。发挥复制的作用就是原型模式的核心思想。 一、源型模式的定义 原型模式是指第二次创建对象可以通过复制已经存在的原型对象来实现,忽略对象创建过程中的其它细节。 📌 核心特点: 避免重复初…...

华为OD机试-食堂供餐-二分法
import java.util.Arrays; import java.util.Scanner;public class DemoTest3 {public static void main(String[] args) {Scanner in new Scanner(System.in);// 注意 hasNext 和 hasNextLine 的区别while (in.hasNextLine()) { // 注意 while 处理多个 caseint a in.nextIn…...

微服务商城-商品微服务
数据表 CREATE TABLE product (id bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 商品id,cateid smallint(6) UNSIGNED NOT NULL DEFAULT 0 COMMENT 类别Id,name varchar(100) NOT NULL DEFAULT COMMENT 商品名称,subtitle varchar(200) NOT NULL DEFAULT COMMENT 商…...

ArcGIS Pro制作水平横向图例+多级标注
今天介绍下载ArcGIS Pro中如何设置水平横向图例。 之前我们介绍了ArcGIS的横向图例制作:ArcGIS横向、多列图例、顺序重排、符号居中、批量更改图例符号等等(ArcGIS出图图例8大技巧),那这次我们看看ArcGIS Pro如何更加快捷的操作。…...

Selenium常用函数介绍
目录 一,元素定位 1.1 cssSeector 1.2 xpath 二,操作测试对象 三,窗口 3.1 案例 3.2 窗口切换 3.3 窗口大小 3.4 屏幕截图 3.5 关闭窗口 四,弹窗 五,等待 六,导航 七,文件上传 …...

计算机基础知识解析:从应用到架构的全面拆解
目录 前言 1、 计算机的应用领域:无处不在的数字助手 2、 计算机的进化史:从算盘到量子计算 3、计算机的分类:不止 “台式机和笔记本” 4、计算机的组件:硬件与软件的协同 4.1 硬件:五大核心部件 4.2 软件&#…...
第7篇:中间件全链路监控与 SQL 性能分析实践
7.1 章节导读 在构建数据库中间件的过程中,可观测性 和 性能分析 是保障系统稳定性与可维护性的核心能力。 特别是在复杂分布式场景中,必须做到: 🔍 追踪每一条 SQL 的生命周期(从入口到数据库执行)&#…...