MySQL 存储

news2024/11/22 22:08:46

关系型数据库是基于关系模型的数据库, 而关系模型是通过二维表来保存的,所以关系型数据库中的数据的村方式就是行列组成的表,每一列代表一个字段,每一行代表一条记录。表可以看作某个实体的集合,实体之间存在的联系需要通过表与表之间的关联关系体现的,例如主键与外键的关系。由多个表组成数据库,就是关系型数据库

关系型数据库有多种。l例如: SQLite, MySQL, Oracle, SQL Server, DB2等

在 python2 中, 连接 MySQL 的库大多是 MySQLdb, 但是此库的官方不支持 Python3, 所以这里推荐使用的库是 PyMySQL

准备工作:

在开始之前要确保安装了 MySQL ,并保证它能正常运行

安装

官网: MySQL :: MySQL Downloads

安装路径和数据路径 , 安装路径要记住,后面配置环境变量要用

后面就是下一步了

密码设置要记住,后面会常用

完成之后,右键开始菜单------搜素---mysql -----出现的操作台,输入刚刚的密码

顺利进入,就说明安装完成了

配置环境变量

找到安装路径bin 目录 , 点进去复制路径。在我的电脑---右键--- 属性----高级设置---环境变量----path ----新建---粘贴刚刚复制的路径---一路确定回去---win+R ----输入 mysql -uroot -p 

然后输入密码---成功进入

然后安装 : pymysql

pip install pymysql 

连接数据库

首先尝试连接数据,假设当前 MySQL 运行在本地, 用户名为 root 密码为 123456, 运行端口为 3306 。 这里利用 PyMySQL 先连接 MySQL ,然后创建一个新的数据库, 叫作 spiders 

import pymysql
db = pymysql.connect(host='localhost', user='root', password='123456', port=3306)
cursor = db.cursor()
cursor.execute('SELECT VERSION()')
data = cursor.fetchone()
print('Database version:', data)
cursor.execute("CREATE DATABASE spiders DEFAULT CHARSET utf8mb4")
db.close()

Database version: ('8.0.38',)

这里通过 PyMySQL 的 connect 方法声明了一个 MySQL连接对象 db , 此时需要传入的第一个参数是 MySQL 运行的  host(即 IP) ,由于 MySQL 运行在本地, 所以传入的是 loaclhost , 如果 MySQL在远程运行, 则传入其公网 IP 地址。 后续参数分别是 user 用户名, password 密码 和 port 端口号 默认为 3306

连接成功后调用 cursor 方法获得了 MySQL 的操作游标, 利用游标可以执行 SQL 语句。 这里执行了两个 SQL 语句, 直接调用 execute 方法即可执行。 第一个 SQL 语句用于获取 MySQL 的当前版本, 然后调用 fetchone 方法就得到了第一条数据, 即版本号。 第二个 SQL 语句用于创建数据库 spiders , 默认编码为 UTF-8, 由于该语句不是查询语句, 所以执行后成功创建了 spiders 数据库

创建表

创建一个 students 表

import pymysql

# 创建数据库后, 在连接的时候需要额外指定一个参数 db
db = pymysql.connect(host='localhost', user='root', password='123456', port=3306, db='spiders')
cursor = db.cursor()
sql = 'CREATE TABLE IF NOT EXISTS students (id VARCHAR(255) NOT NULL, name VARCHAR(255) NOT NULL, age INT NOT NULL, PRIMARY KEY(id))'
cursor.execute(sql)
db.close()

插入数据

import pymysql

id = '20120001'
user = 'Bob'
age = 20

db = pymysql.connect(host='localhost', user='root', password='123456', port=3306, db='spiders')
cursor = db.cursor()
sql = 'INSERT INTO students(id, name, age) values(%s, %s, %s)'
try:
    cursor.execute(sql, (id, user, age))
    db.commit()
except:
    db.rollback()

db.close()

这里首先定义了需要插入的数据,然后构造了一个SQL 语句,这个语句还有另一种写法

sql = 'INSERT INTO students(id, name, age) values('+ id +', '+ name +','+ age +')'

这样的写法繁琐不直观,所以我们直接用格式化符 %s 来构造, 有几个 value 就写几个 %s 。 我们需要在 execute 方法的第一个参数传入该 SQL 语句,。 value 值统一用元组传过来就好。这样的写法既可以避免字符串拼接的麻烦,又可以避免引号冲突的问题

之后值得注意的是,需要执行 db 对象的 commit 方法才可以实现数据插入, 这个方法才是真正将语句提交到数据库的方法。 对于数据 插入,更行, 删除操作都需要调用该方法才能生效。

接下来,我们加了一层异常处理。 如果执行失败, 则调用 rollback 执行数据回滚, 相当于什么都没有发生过

这里涉及事务问题。事务机制能够确保数据的一致性,也就是一件事要发生完整了,要么完全没有发生。 例如插入一条数据, 不会存在插入一半的情况

事务的4个属性: 原子性, 一直性,隔离性, 持久性

上面的数据插入有一个很明显的不足,就是如果需要插入新字段 gender,就需要重新改动插入语句

sql = 'INSERT INTO students(id, name, age, gender) values(%s, %s, %s, %s)'

元组参数

(id, user, age, gender)

这显然不是我们要的, 在很多情况下, 我们要达到的效果是插入方法无需做改动, 其做为通用方法,只需传入要给动态字典就好

{

        'id': '20120001',

        'name': 'Bob',

        'age': 20

}

然后 SQL 语句会根据这个动态字典构造出来, 元组也是这样才能实现通用插入方法

import pymysql

data = {
    'id': '20120003',
    'name': 'Bob',
    'age': 21
}


db = pymysql.connect(host='localhost', user='root', password='123456', port=3306, db='spiders')
cursor = db.cursor()
table = 'students'
keys = ','.join(data.keys())
values = ','.join(['%s'] * len(data))
# sql = 'INSERT INTO students(id, name, age) values(%s, %s, %s)'
sql = 'INSERT INTO {table}({keys}) VALUES({values})'.format(table=table, keys=keys, values=values)
print(sql)
try:
    if cursor.execute(sql, tuple(data.values())):
        print('Successful')
        db.commit()
except:
    print('Faile')
    db.rollback()
db.close()

这里我们传入的数据是字典, 将其定义为了 data 变量, 将 students 表定义为了变量 table 。 接下来构造了一个动态 SQL语句

首先,需要构造插入的字段: id , name, age 。 这里只要将  data 的键名拿过来,并且用逗号分隔即可。 所以 ','.join(dat.keys()) 的结果就是 id,name,age, 然后,需要构造多个 %s 当作占位符, 由几个字段就构造几个。 例如,这里有三个字段, 就需要构造 %s, %s, %s 这里先是定义了长度为 1 的数组 ['%s'], 然后用乘法将其扩充为 ['%s', '%s', '%s'], 再调用 join 方法, 就变成了 %s, %s, %s, 最后利用字符串的 format 方法将表明, 字段名和占位符构造出来。于是SQL语句就构造出来了

最后,为 execute 方法的第一个参数传入 sql 变量, 第二个参数传入由 data 的键值构造的元组,就可以成功插入数据了

更新数据

更新数据实际上也是执行 SQL语句

import pymysql
db = pymysql.connect(host='localhost', user='root', password='123456', port=3306, db='spiders')
cursor = db.cursor()
sql = 'UPDATE students SET age = %s WHERE name = %s'
try:
    cursor.execute(sql, (25, 'Bob'))
    db.commit()
except:
    db.rollback()
db.close()

这里同样用占位符的方式构造 SQL , 然后执行 execute 方法, 传入元组形式的参数,同样执行 commit 方法,如果做的是简单的数据更新,完全可以用此方法

但是在实际的数据抓取过程中, 大部分情况下需要插入数据,我们关心的是会不会出现重复数据,如果出现了,我们希望更新数据,而不是重复保存一次。另外,就像前面所说的动态构造SQL的问题,所以这里可以再实现一次去重的方法: 如果保存的诗句存在,就更新数据,如果不存在,则插入数据。另外这样做法支持灵活的字典传值

import pymysql
db = pymysql.connect(host='localhost', user='root', password='123456', port=3306, db='spiders')
cursor = db.cursor()
data = {
    'id': '20120002',
    'name': 'Box',
    'age': 22
}
table = 'students'
keys = ', '.join(data.keys())
values = ', '.join(['%s'] * len(data))

sql = 'INSERT INTO {table}({keys}) VALUES({values}) ON DUPLICATE KEY UPDATE '.format(table=table,keys=keys,values=values)
update = ','.join(["{key} = %s".format(key=key) for key in data])
sql += update
try:
    if cursor.execute(sql, tuple(data.values())*2):
        print('Successful')
        db.commit()
except:
    print('Failed')
    db.rollback()
db.close()

这里构造的SQL 语句其实是插入语句, 但是我们再后面加了 ON DUPLICATE KEY UPDATE。这行代码的意思是如果主键已经存在,就执行更新操作。例如 我们传入的数据的 id 在表里已经存在,只是 年龄(age) 有变化,此时就不会插入这条数据,而是直接更新 age 为新数据,这里构造出来的 SQL 语句有 6 个 %s

INSERT INTO students(id, name, age) VALUES(%s, %s, %s) ON DUPLICATE KEY UPDATE id = %s,name = %s,age = %s

所以后面 execute 方法的第二个参数要 乘以 2 ,是长度变成原来的两倍

如此一来, 我们就可以实现主键不存在插入数据, 主键存在则更新数据

删除数据

删除数据相对比较简单 直接使用 DELETE语句即可, 只是需要指定要删除的目标表名和删除条件,而且仍然要使用 db 的 commit 方法才能生效

import pymysql
db = pymysql.connect(host='localhost', user='root', password='123456', port=3306, db='spiders')
cursor = db.cursor()
table = 'students'
condition = 'age > 20'

sql = 'DELETE FROM {table} WHERE {condition}'.format(table=table, condition=condition)
print(sql)
try:
    cursor.execute(sql)
    db.commit()
except:
    db.rollback()
db.close()

由于删除的条件多种多样,运算符有 大于,小于, 等于, LIKE等, 条件连接符有 AND OR等

查询数据

SELECT 语句

import pymysql
db = pymysql.connect(host='localhost', user='root', password='123456', port=3306, db='spiders')
cursor = db.cursor()

sql = 'SELECT * FROM students WHERE age > 10'
try:
    cursor.execute(sql)
    print('Count:',cursor.rowcount)
    one = cursor.fetchone()
    print('One', one)
    results = cursor.fetchall()
    print('Results:', results)
    print('Rsults Type:', type(results))
    for row in results:
        print(row)
except:
    print('Error')

Count: 2
One ('20120001', 'Bob', 20)
Results: (('20120002', 'Box', 22),)
Rsults Type: <class 'tuple'>
('20120002', 'Box', 22)

要想看到所有的效果,students 表里面至少要有两条以上的数据,且 符合 gae > 10 的条件

这里我们构造了一个 SQL 语句, 查询年龄为10 以上的学生, 然后将其传给 execute 方法,注意这里不再需要 db 的 commit 方法, 接着 调用 cursor 的 rowcount 属性获取查询结果的条数

然后我们调用了 fetchone 方法, 这个方法可以获取结果的第一条数据, 返回结果以元组形式呈现,元组中的元素的顺序跟字段一 一对应, 即第一个元素就是第一个字段 id , 第二个元素就是第二个字段的 name , 以此类推。 随后,我们调用了 fetchall 方法, 可以得到结果的所有数据。之后将其结果和类型打印出来,是一个二重元组, 其中每个元素都是一条记录,我们遍历这些元素并输出

但是需要注意一个问题,这里结果显示的是 比总数据少一条的, 这是因为 fetchall 方法内部有一个偏移指针,没获取一次数据就会向后移动一次, 而在前面我们调用过一次 fetchone 方法,指针已经向后移动了一次, 所以这里遍历开始的时候已经指向了第二条

我们也可以用 while 循环加 fetchone 方法组合来获取所有数据。因为 fetchall 会将结果以元组形式全部返回,如果数据量很大,那么占用的开销也会非常高。因此推荐逐条获取数据

import pymysql
db = pymysql.connect(host='localhost', user='root', password='123456', port=3306, db='spiders')
cursor = db.cursor()

sql = 'SELECT * FROM students WHERE age > 10'
try:
    cursor.execute(sql)
    print('Count:',cursor.rowcount)
    row = cursor.fetchone()
    while row:
        print('Row:', row)
        row = cursor.fetchone()
except:
    print('Error')

这样没循环一次,指针就会偏移一条数据,随用随取,简单高效

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

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

相关文章

【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第三篇 嵌入式Linux驱动开发篇-第五十三章 设备树下的platform驱动

i.MX8MM处理器采用了先进的14LPCFinFET工艺&#xff0c;提供更快的速度和更高的电源效率;四核Cortex-A53&#xff0c;单核Cortex-M4&#xff0c;多达五个内核 &#xff0c;主频高达1.8GHz&#xff0c;2G DDR4内存、8G EMMC存储。千兆工业级以太网、MIPI-DSI、USB HOST、WIFI/BT…

基于开源大模型的问答系统本地部署实战教程

1. 现有大模型产品使用感受 在开始教你搭建属于自己的本地大模型之前&#xff0c;先说下自己日常使用大模型的感受。常用的比较好的国外大模型工具&#xff0c;经常会遇到网络卡或者token数量、使用次数限制的问题。而国内的大模型工具虽然不存在网络的问题&#xff0c;但…

简单几步,教你使用scikit-learn做分类和回归预测

经常听到初学python的小伙伴在抱怨&#xff0c;python安装第三方库太慢&#xff0c;很容易失败报错&#xff0c;如果安装pandas、tensorflow这种体积大的库&#xff0c;简直龟速。 为什么pip会很慢&#xff1f; 先来了解下pip&#xff0c;pip是一个非常流行的python包管理工具…

基于微信小程序+SpringBoot+Vue的资料分享系统(带1w+文档)

基于微信小程序SpringBootVue的资料分享系统(带1w文档) 基于微信小程序SpringBootVue的资料分享系统(带1w文档) 校园资料分享微信小程序可以实现论坛管理&#xff0c;教师管理&#xff0c;公告信息管理&#xff0c;文件信息管理&#xff0c;文件收藏管理等功能。该系统采用了Sp…

初识C++ · AVL树(2)

目录 前言&#xff1a; 1 左右旋 2 右左旋 3 部分细节补充 3.1 单旋和插入 3.2 部分小函数 前言&#xff1a; AVL树作为一种结构&#xff0c;理解树的本身是不大难的&#xff0c;难的在于&#xff0c;树旋转之后的连接问题&#xff0c;写AVL树的代码大部分都是在旋转部分…

在jeesite开源平台上写了一个SQL命令中心的功能

实现目的: 这个SQL命令中心,是因为老项目就有这个页面,主要的功能是根据写出的SQL语句查询数据,并且在查出的数据基础上直接修改更新,还有新增和删除的功能,这么一说跟plsql就一样一样的了;这页面本来是给运维的同事来用,而且他们还会用plsql和Navicat等SQL语言操作工…

[OJ]平均串问题,存在超时问题未解决

众所周知&#xff0c;两个数a和b的平均数计算公式为(ab)/2。 实际上平均数也可以描述为&#xff1a;从较小的数依次遍历到较大的数&#xff0c;将遍历的数放入一个列表中&#xff0c;该列表的中心元素。例如&#xff1a;求 3和7的平均数&#xff0c;列表为{3,4,5,6,7}&#xff…

FPGA与ASIC:深入解析芯片设计的双子星

前言 在半导体世界里&#xff0c;FPGA&#xff08;Field-Programmable Gate Array&#xff0c;现场可编程门阵列&#xff09;与ASIC&#xff08;Application-Specific Integrated Circuit&#xff0c;专用集成电路&#xff09;是两种截然不同的芯片设计策略&#xff0c;各自在…

短视频矩阵系统,一键智能成片

在信息爆炸的时代&#xff0c;短视频以其短平快的特点迅速崛起&#xff0c;成为人们获取信息、娱乐消遣的重要渠道。然而&#xff0c;如何在这个竞争激烈的领域中脱颖而出&#xff0c;制作出吸引眼球的爆款视频呢&#xff1f;今天&#xff0c;我们就来揭秘一款神奇的短视频矩阵…

76.WEB渗透测试-信息收集- WAF、框架组件识别(16)

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 内容参考于&#xff1a; 易锦网校会员专享课 上一个内容&#xff1a;75.WEB渗透测试-信息收集- WAF、框架组件识别&#xff08;15&#xff09; Php的Laraverl…

C语言 之 理解指针(4)

文章目录 1. 字符指针变量2. 数组指针变量2.1 对数组指针变量的理解2.2 数组指针变量的初始化 3. 二维数组传参的本质4. 函数指针变量4.1 函数指针变量的创建4.2 函数指针变量的使用 5. 函数指针数组 1. 字符指针变量 我们在前面使用的主要是整形指针变量&#xff0c;现在要学…

php--高级反序列化

&#x1f3bc;个人主页&#xff1a;金灰 &#x1f60e;作者简介:一名简单的大一学生;易编橙终身成长社群的嘉宾.✨ 专注网络空间安全服务,期待与您的交流分享~ 感谢您的点赞、关注、评论、收藏、是对我最大的认可和支持&#xff01;❤️ &#x1f34a;易编橙终身成长社群&#…

21 Python常用内置函数——zip()

zip() 函数用来把多个可迭代对象中的元素压缩到一起&#xff0c;返回一个可迭代的 zip 对象&#xff0c;其中每个元素都是包含原来的多个可迭代对象对应位置上元素的元组&#xff0c;最终结果中包含的元素个数取决于所有参数序列或可迭代对象中最短的那个。 可以这样理解这个函…

学习笔记(数据结构:队列)4

qt加库 队列&#xff1a; 队列是只允许在一段进行插入&#xff0c;而在另一端进行删除操作的线性表。 允许插入的称谓队尾&#xff0c;允许删除的一端队头。 顺序队列。 循环队列&#xff0c; 常用操作&#xff0c;入队&#xff0c;出队。 先进先出&#xff0c;FIFO 用&#x…

安装nfs和rpcbind设置linux服务器共享磁盘

1、安装nfs和rpcbind 1.1 检查服务器是否安装nfs和rpcbind&#xff0c;执行下命令&#xff0c;检查服务器是否安装过。 rpm -qa|grep nfs rpm -qa|grep rpcbind 说明服务器以安装了&#xff0c;如果没有就需要自己安装 2、安装nfs和rpcbind 将rpm安装包&#xff1a; libtirpc-…

【kubernetes】认识K8S基础理论

目录 1.背景 2.虚拟机、物理机与容器 3.K8S 基本概念 3.1 K8S 的特点 4.Kubernetes 集群架构与组件 4.1 核心组件 4.1.1 Master组件 4.1.2 配置存储中心 4.1.3 Node 组件 5.Kubernetes 核心概念 6.总结 1.背景 服务器经历了三次演变过程&#xff1a;物理机、虚拟机…

【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第三篇 嵌入式Linux驱动开发篇-第五十八章 中断下文之tasklet

i.MX8MM处理器采用了先进的14LPCFinFET工艺&#xff0c;提供更快的速度和更高的电源效率;四核Cortex-A53&#xff0c;单核Cortex-M4&#xff0c;多达五个内核 &#xff0c;主频高达1.8GHz&#xff0c;2G DDR4内存、8G EMMC存储。千兆工业级以太网、MIPI-DSI、USB HOST、WIFI/BT…

跨越至智慧水利新时代:以科技创新为核心引擎,全面构建智能化水资源管理体系,显著提升水资源治理的智能化水平与高效治理能力,共筑可持续水生态未来

目录 一、智慧水利的概念与重要性 &#xff08;一&#xff09;智慧水利的概念 &#xff08;二&#xff09;智慧水利的重要性 二、智慧水利的关键技术 &#xff08;一&#xff09;物联网技术 &#xff08;二&#xff09;大数据技术 &#xff08;三&#xff09;云计算技术 …

Tekion 选择 ClickHouse Cloud 提升应用性能和指标监控

本文字数&#xff1a;4187&#xff1b;估计阅读时间&#xff1a;11 分钟 作者&#xff1a;ClickHouse team 本文在公众号【ClickHouseInc】首发 Tekion 由前 Tesla CIO Jay Vijayan 于 2016 年创立&#xff0c;利用大数据、人工智能和物联网等技术&#xff0c;为其汽车客户解决…

X00078-基于深度强化学习图神经网络的云工作流调度python

强化学习&#xff08;RL&#xff09;技术通过蒙特卡洛树搜索&#xff08;MCTS&#xff09;的加速&#xff0c;成功解决了多资源需求下的DAG问题&#xff0c;优化目标为缩短makespan。根据当前的计算资源和任务状态&#xff0c;MCTS算法被用来探索并选择最佳的动作&#xff0c;在…