1分钟插入10亿行数据!抛弃Python,写脚本请使用Rust
off999 2024-10-21 06:51 16 浏览 0 评论
来源:Avinash
编辑:好困
【新智元导读】近日,一位程序员急需在一分钟之内生成十亿行的测试数据库,然而在用Python写了脚本之后发现「大失败」。怎么办?当然是用Rust了!
最近,一位程序员表示自己急需一个「也就」十亿行数据的测试数据库,并且还得在一分钟之内生成。
于是,他做了一个所有程序员都会做的事:写一个Python脚本来生成数据库。
然而,很不幸的是,这个脚本非 常 慢。
于是,他又做了一个所有程序员都会做的事:进一步学习关于SQLite、Python以及不知道为什么还有Rust的知识。
项目已开源:https://github.com/avinassh/fast-sqlite3-inserts
目标
作者需要在他2019年的MacBook Pro(2.4GHz四核i5)上,一分钟内生成一个有10亿行的SQLite数据库。
表的模式
要求:
- 生成的数据是随机的;
- 「area」列将包含六位数的地区代码(任何六位数都可以,不需要验证);
- 「age」列是5、10或15中的任何一个;
- 「active」列是0或1。
不过,作者表示,对脚本的要求也不用太高,还是可以妥协的:
- 如果进程崩溃,所有的数据都丢失也没有问题,再次运行脚本就可以了;
- 允许充分利用电脑的资源:100%的CPU,8GB的内存和剩余的SSD储存;
- 不需要使用真正的随机方法,来自stdlib的伪随机方法就可以。
Python原型
在最开始的脚本中,作者试图在一个for循环中逐一插入1000万条记录,而这让用时直接达到了15分钟。
显然,这太慢了。
在SQLite中,每次插入都是一个事务,每个事务都保证它被写入磁盘,作者推断可能问题就来自这里。
于是作者开始尝试不同规模的批量插入,发现10万是一个最佳点,而运行时间减少到了10分钟。
SQLite优化
作者认为自己写的代码已经很简练了,并没有什么可以优化的空间。
于是他将下一个目标转到了数据库的优化。
根据各种关于SQLite优化的建议,作者做了一些改进。
- 关闭「journal_mode」将禁用回滚日志,也就是说,如果任何事务失败,都无法回滚。
- 关闭「synchronous」,将使SQLite不再关心是否能可靠地写入磁盘,而是把这个责任交给操作系统。也就是说,可能会出现SQLite并没有成功写入磁盘的情况。
- 「cache_size」指定了SQLite在内存中可以保留多少个内存页。
- 当「locking_mode」为「EXCLUSIVE」模式时,SQLite锁住的连接将永远不会被释放。
- 将「temp_store」设置为「MEMORY」可以让其表现像一个内存数据库。
此处作者提醒,请不要把这些操作用到生产上去。
重新审视Python
作者再次重写了Python脚本,这次包括了微调的SQLite参数,这次带来了巨大的提升,运行时间大幅减少:
- 原始的for循环版本用时大约10分钟。
- 批处理版本用时大约8.5分钟。
PyPy
PyPy在其主页上强调它比CPython快4倍,于是作者决定尝试一下。
令作者有些意外的是,竟然不需要对现有的代码进行任何改动,只需要在PyPy运行就可以了。
批处理版本只需要2.5分钟,也就是速度快了接近3.5倍。
Busy Loop?
莫非是在Python的循环上耗费了太多时间?于是作者删除了SQL指令之后再次跑了一遍代码:
- 批处理版本在CPython中用时5.5分钟。
- 批处理版本在PyPy中用时1.5分钟(又是3.5倍的速度提升)。
然而用Rust重写了相同的内容之后,循环只需要17秒。
于是,作者果断抛弃Python,转投Rust的怀抱。
Rust
像Python一样,作者先写了一个原始的Rust版本,一个循环执行一行数据的插入。
然而,即便使用了所有SQLite的优化,也依然消耗了大约3分钟。于是作者进行了进一步的测试:
- 尝试把「rusqlite」换成异步运行的「sqlx」,这让用时直接被拉到了14分钟。作者表示,这比自己迄今为止写的任何一个Python迭代都要差。
- 在执行原始SQL语句时,使用准备好的语句。这个版本的用时只有1分钟。
最优的版本
使用准备好的语句,以50行为一个批次插入,最终用时34.3秒。
作者又写了一个线程版本,其中一个线程从通道接收数据,还有四个线程向通道推送数据。
这个也是目前性能最好的版本,最终用时大约32.37秒。
IO时间
SQLite论坛上的网友提出了一个有趣的想法:测量内存数据库所需的时间。
于是作者又跑了一遍代码,将数据库的位置设定为「:memory:」,rust版本完成的时间少了两秒(29秒)。
也就是说将1亿条记录写入到磁盘上需要2秒,这个用时似乎也是合理的。
这也说明,可能没有更多的SQLite优化可以以更快的方式写入磁盘,因为99%的时间都花在生成和添加数据上。
排行榜
插入1亿行数据的用时:
Rust33秒 PyPy126秒 CPython210秒
总结
- 尽可能使用SQLite PRAGMA语句
- 使用准备好的语句
- 进行分批插入
- PyPy确实比CPython快4倍
- 异步不一定更快
目前,第二快的版本是单线程运行的,而作者的电脑有4个核心,于是他在一分钟内可以得到8亿行数据。然后再经过几秒钟的数据合并,时间仍然可以少于一分钟。
网友评论
博主的这一番研究获得了网友们的一致好评。
真的很喜欢这些观点:
学习了更多关于PRAGMA语句。
PyPy的效率和灵活性可以通过即插即用的方式体现(将来一定会给它一个机会)。
文章的排版非常简单,有适当的源代码链接。很有趣,很容易上手。
Rust高光时刻又来了!
参考资料:
https://avi.im/blag/2021/fast-sqlite-inserts/
https://github.com/avinassh/fast-sqlite3-inserts
相关推荐
- python列表(List)必会的13个核心技巧(附实用方法)
-
列表(List)是Python入门的关键步骤,因为它是编程中最常用的数据结构之一。以下是高效掌握列表的核心技巧和实用方法:一、理解列表的本质可变有序集合:可随时修改内容,保持元素顺序混合类型:一个列表...
- Python列表(List)一文全掌握:核心知识点+20实战练习题
-
Python列表(List)知识点教程一、列表的定义与特性定义:列表是可变的有序集合,用方括号[]定义,元素用逗号分隔。list1=[1,"apple",3.14]lis...
- python编程中列表常见的9大问题,你知道吗?
-
Python列表常见错误及解决方案列表(list)是Python中最常用的数据结构之一,但在使用过程中经常会遇到各种问题。以下是Python列表使用中的常见错误及其解决方法:一、索引越界错误1.访问...
- python之列表操作(python列表操作函数大全)
-
常用函数函数名功能说明append将一个元素添加到列表中names=['tom']用法:names.append('tommy')注意事项:被添加的元素只会被添加到...
- 7 种在 Python 中反转列表的智能方法
-
1.使用reverse()方法(原地)my_list=[10,12,6,34,23]my_list.reverse()print(my_list)#output:[23,34,6,12,...
- Python教程-列表复制(python中列表copy的用法)
-
作为软件开发者,我们总是努力编写干净、简洁、高效的代码。Python列表是一种多功能的数据结构,它允许你存储一个项目的集合。在Python中,列表是可变的,这意味着你可以在创建一个列表后改变它的...
- 「Python程序设计」基本数据类型:列表(数组)
-
列表是python程序设计中的一个基本的,也是重要的数据结构。我们可以把列表数据结构,理解为其它编程语言中的数组。定义和创建列表列表中的数据元素的索引,和数组基本一致,第一个元素的索引,或者是下标为0...
- Python中获取列表最后一个元素的方法
-
技术背景在Python编程中,经常会遇到需要获取列表最后一个元素的场景。Python提供了多种方法来实现这一需求,不同的方法适用于不同的场景。实现步骤1.使用负索引-1这是最简单和最Pythoni...
- Python学不会来打我(11)列表list详解:用法、场景与类型转换
-
在Python编程中,列表(list)是最常用且功能最强大的数据结构之一。它是一个有序、可变、支持重复元素的集合,可以存储任意类型的对象,包括整数、字符串、布尔值、甚至其他列表。本文将从基础语法开始...
- 零起点Python机器学习快速入门-4-4-列表操作
-
Python列表的基本操作展开。首先,定义了两个列表zlst和vlst并将它们的内容打印出来。接着,使用切片操作从这两个列表中提取部分元素,分别得到s2、s3和s4三个新的列表,并打...
- python入门 到脱坑 基本数据类型—列表
-
以下是Python列表(List)的入门详解,包含基础操作、常用方法和实用技巧,适合初学者系统掌握:一、列表基础1.定义列表#空列表empty_list=[]#包含不同类型元素的列表...
- Python 列表(List)完全指南:数据操作的利器
-
在Python中,列表(list)是一种可变序列(mutablesequence),它允许我们存储和操作一组有序数据(ordereddata)。本教程将从基础定义(basicdefiniti...
- 如何快速掌握 Python中列表的使用
-
学习python知识,好掌握Python列表的使用。从概念上来讲,Python中的列表list是一种有序、可变的容器,可以存储任意类型的数据(包括其他列表)。以下是列表的常用的操作和知识:1....
- Python中的列表详解及示例(python中列表的用法)
-
艾瑞巴蒂干货来了,数据列表,骚话没有直接来吧列表(List)是Python中最基本、最常用的数据结构之一,它是一个有序的可变集合,可以包含任意类型的元素。列表的基本特性有序集合:元素按插入顺序存储可变...
- python数据类型之列表、字典、元组、集合及操作
-
Python数据类型进阶:列表、字典与集合在Python中,数据类型是编程的基础,熟练掌握常用数据结构是成为高级开发者的关键。上一篇文章我们学习到了Python的数据类型:字符串(string)、数...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python自定义函数 (53)
- python进度条 (67)
- python吧 (67)
- python字典遍历 (54)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python串口编程 (60)
- python读取文件夹下所有文件 (59)
- java调用python脚本 (56)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python人脸识别 (54)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)