You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

231 lines
15 KiB
Markdown

2 years ago
# 27zipfile压缩库如何给数据压缩&加密备份?
你好,我是尹会生。
你在日常工作中,肯定和压缩文件打过交道,它们能把文件夹制作成一个体积更小的压缩文件,不仅方便数据备份,还方便作为邮件附件来传输,或者与他人共享。
但是如果你需要每天都进行数据备份或者把压缩包作为每天工作的日报发送给领导你肯定希望它能自动化的压缩。面对这个需求我们同样可以通过python来解决。我们可以用Python来自动压缩文件夹并为压缩包设置密码保证备份数据的安全。
在Python中要想实现数据的压缩一般可以采用基于标准库zipfile的方式来实现也可以采用命令行方式来实现。
当我们希望能够用Python自动压缩一个无需密码保护的文件夹时可以通过zipfile来实现它的好处是使用简单而且不用安装任何的软件包就能制作出“zip”格式的压缩包。不过zipfile没法对压缩文件进行加密因此当你需要对压缩文件加密时还需要调用可执行命令。
这两种实现方式就是我们今天要学习的重点了,接下来我们分别看一下这两种方式的具体操作方法。
## 使用zipfile实现无密码压缩
如果我想要把“C:\\data\\”文件夹压缩为“当前日期.zip”文件就可以使用**目录遍历、按日期自动生成压缩包的文件名、****把****文件夹写入压缩文件****这****三个步骤来实现。**
### 目录遍历
**我们先来学习****怎么实现****目录遍历功能**。我在第16讲已经为你讲解过它的技术细节了这里我就继续使用os库来实现目录的遍历。
由于目录遍历的功能与其他功能之间的调用关系耦合得比较宽松所以我就把目录遍历功能单独定义成一个getAllFiles()函数,并把要遍历的目录作为函数的参数,把该目录下的所有文件以及所在路径作为函数的返回值。
我把getAllFiles()函数的代码放在下方,供你参考。
```
import os
# 遍历目录,得到该目录下所有的子目录和文件
def getAllFiles(dir):
for root,dirs,files in os.walk(dir):
for file in files:
yield os.path.join(root, file)
```
细心的你一定发现了在函数getAllFiles()的返回语句中我使用yield语句代替了之前学习过的return语句返回文件路径和名称。为什么我要使用yield语句呢
原因就在于,****一个函数如果使用yield语句来返回的话这个函数则被称作生成器****。yield的返回数据类型以及对类型的访问方式都和return不同。我来为你解释一下yield和return的具体区别以及使用yield的好处。
首先从返回类型来看yield返回的数据类型叫做生成器类型这一类型的好处是调用getAllFiles()一次函数就会返回一个文件路径和文件名。而return返回的是一个列表类型需要一次性把要备份目录下的所有文件都访问一次一旦要备份的文件数量非常多就会导致计算机出现程序不响应的问题。
除了返回类型还有调用方式也和return不同。使用yield返回的对象被称作生成器对象该对象没法像列表一样一次性获得对象中的所有数据你必须使用for循环迭代访问才能依次获取数据。
此外当所有的数据访问完成还会引发一个“StopIteration”异常告知当前程序这个生成器对象的内容已经全部被取出来那么这个生成器将会在最后一次访问完成被计算机回收这样yield就能够知道对象是否已经全部被读取完。
从yield和return的行为对比可以说yield返回对象最大的好处是可以逐个处理而不是一次性处理大量的磁盘读写操作这样就有效减少了程序因等待磁盘IO而出现不响应的情况。这就意味着你不必在调用getAllFiles()函数时,因为需要备份的文件过多,而花费较长的时间等待它执行完成。
### 按日期自动生成压缩包的文件名
**接下来我们来学习一下按日期自动生成压缩包的函数**genZipfilename()。按日期生成文件名,在定时备份的场景中经常被用到,我们希望每天产生一个新的备份文件,及时保存计算机每天文件的变化。
这就要求今天的备份的文件名称不能和昨天的同名,避免覆盖上次备份的文件。
所以genZipfilename()函数就把程序执行的日期作为文件名来进行备份例如当前的日期是2021年4月12日那么备份文件会自动以“20210412.zip”作为文件名称。我把代码贴在下方供你参考。
```
import datetime
# 以年月日作为zip文件名
def genZipfilename():
today = datetime.date.today()
basename = today.strftime('%Y%m%d')
extname = "zip"
return f"{basename}.{extname}"
```
在这段代码中“datetime.date.today()”函数能够以元组格式取得今天的日期不过它的返回格式是元组且年、月、日默认采用了三个元素被存放在元组中这种格式是没法直接作为文件名来使用的。因此你还需要通过strftime()函数把元组里的年、月、日三个元素转换为一个字符串,再把字符串作为文件的名称来使用。
### **把文件夹写入压缩文件**
**最后准备工作都完成之后你就可以使用zipfile库把要备份的目录写入到zip文件了**。zipfile库是Python的标准库所以不需要安装软件包为了让这个完整脚本都不需要安装第三方软件包我在实现文件遍历的时候同样采用os库代替pathlib库。
除了不需要安装之外zipfile库在使用上也比较友好它创建和写入zip文件的方式就是模仿普通文件的操作流程使用with关键字打开zip文件并使用write()函数把要备份的文件写入zip文件。
所以通过学习一般文件的操作你会发现Python在对其他格式的文件操作上都遵循着相同的操作逻辑这也体现出Python语言相比其他语言更加优雅和简单。
那么我把使用zipfile库实现创建zip文件的功能写入zipWithoutPassword()函数中,你可以对照一般文件的写入逻辑来学习和理解这段代码,代码如下:
```
from zipfile import ZipFile
def zipWithoutPassword(files,backupFilename):
with ZipFile(backupFilename, 'w') as zf:
for f in files:
zf.write(f)
```
对比一般的文件写入操作zip文件的打开使用了“ZipFile()函数”而一般文件的打开使用了open函数。写入方法与一般文件相同都是调用“write()”函数实现写入。
这三个函数也就是函数getAllFiles()、genZipfilename()和zipWithoutPassword()就是把备份目录到zip文件的核心函数了。我们以备份“C:\\data”文件夹为“20210412.zip”压缩文件为例依次调用三个函数就能实现自动备份目录了我把调用的代码也写在下方供你参考。
```
if __name__ == '__main__':
# 要备份的目录
backupDir = r"C:\data"
# 要备份的文件
backupFiles = getAllFiles(backupDir)
# zip文件的名字“年月日.zip”
zipFilename = genZipfilename()
# 自动将要备份的目录制作成zip文件
zipWithoutPassword(backupFiles, zipFilename)
```
在执行这段代码后就会在代码运行的目录下产生“20210412.zip”文件你通过计算机上的winrar等压缩软件查看就会发现其中会有“C:\\data”文件夹下的所有文件。由于文件名称是以当前日期自动产生的所以每天执行一次备份脚本就能实现按天备份指定的文件夹为压缩包了。
不过在备份时,除了要保证数据的可用性,你还有考虑数据的安全性,最好的办法就是在备份时为压缩包指定密码。接下来我就带你使用命令行调用实现有密码的文件压缩。
## 使用可执行命令实现有密码压缩
在制作有密码的压缩包时我们必须使用命令代替zipfile来压缩文件因为zipfile默认是不支持密码压缩功能的。当你需要对压缩数据有保密性的要求时可以使用7zip、winrar这些知名压缩软件的命令行进加密压缩。
我在本讲中就以7zip压缩工具为例带你学习一下怎么使用Python通过命令行方式调用7zip实现文件的加密压缩。
### 执行方式和执行参数
要想使用7zip实现压缩并被Python直接调用你除了需要在Windows上安装7zip外还需要知道它的**执行方式和执行的参数。**
**我先来带你学习一下执行方式。**7zip软件Windows安装成功后它的命令行可执行程序叫做“7z.exe”。但是它想要在命令行运行的话需要指定程序的完整路径。例如“c:\\path\\to\\installed\\7zip\\7z.exe”。如果你希望在命令行直接输入“7z.exe”运行需要你把可执行程序放在命令搜索路径中。我在这里有必要为你解释一下**命令搜索路径**的概念,有助于你以后在各种操作系统上执行命令行工具。
一条命令要想运行,必须要使用**路径+可执行文件的名称**才可以。例如我Windows中需要把Python的可执行命令“python.exe”安装到“C:\\python3.8\\scripts\\python.exe”这一位置。
那么一般情况下当你需要运行Python解释器时必须输入很长的路径。这种做法在经常使用命令行参数时没法接受的一个是你需要记住大量命令的所在路径另一个是较长的路径也会降低你的执行效率。
因此在各种操作系统上都有“命令搜索路径”的概念。在Windows中命令搜索路径被保存在Path环境变量中Path变量的参数是由分号分隔开的文件夹当你在命令行输入“python.exe”并回车运行它时操作系统会遍历Path变量参数中的每个文件夹。如果找到了“python.exe”文件就可以直接运行它如果没有找到则会提示用户该命令不存在。这就避免你每次执行一条命令时都需要输入较长的路径。
再回到7zip的命令行执行文件“7z.exe”上我把它安装在“C:\\7zip\\”文件夹下如果你希望执行运行7z.exe且不输入路径那么根据上面的分析现在有两种解决办法。
1. 把7z.exe放到现有的命令搜索路径中例如“C:\\python3.8\\scripts\\”文件夹。
2. 把7z.exe所在的文件夹“C:\\7zip\\”加入到命令搜索路径Path变量的参数中。加入的方法是在Windows的搜索栏搜索关键字“环境变量然后在弹出的环境变量菜单把路径加入到Path变量参数即可。
设置完成环境变量后7z.exe就不必在命令行中输入路径直接运行即可。
在你掌握了执行方式后我再来带你学习一下它的参数要想使用支持密码加密方式的zip压缩包你需要使用四个参数它们分别是
1. a参数7z.exe能够把文件夹压缩为压缩包也能解压一个压缩包。a参数用来指定7z将要对一个目录进行的压缩操作。
2. \-t参数用来指定7z.exe制作压缩包的类型和名称。为了制作一个zip压缩包我将把该参数指定为-tzip并在该参数后指定zip压缩包的名称。
3. \-p参数用来指定制作的压缩包的密码。
4. “目录”参数:用来指定要把哪个目录制作为压缩包。
如果我希望把压缩包“20210412.zip”的密码制作为“password123”可以把这四个压缩包的参数组合在一起使用如下命令行
```
7z.exe a -tzip 20210412.zip -ppassword123 C:\data
```
### 扩展zipfile
由于命令的参数较多且记住它的顺序也比较复杂所以我们可以利用Python的popen()函数把“7z.exe”封装在Python代码中会更容易使用。
因此我在无密码压缩的代码中就可以再增加一个函数zipWithPassword()用来处理要压缩的目录、压缩文件名和密码参数并通过这个函数再去调用popen()函数封装命令行调用7z.exe的代码从而实现有密码的压缩功能。代码如下
```
import os
def zipWithPassword(dir, backupFilename, password=None):
cmd = f"7z.exe a -tzip {backupFilename} -p{password} {dir}"
status = os.popen(cmd)
return status
```
我来解释一下这段代码。在实现有密码压缩的函数中为了调用函数更加方便我把“压缩的文件夹、zip文件名称、密码”作为该函数的参数这样当在你调用zipWithPassword()函数时就能指定所有需要加密的文件和目录了。此外在执行命令时我还通过os.popen()函数产生了一个新的子进程如果你不记得这个概念可以参考第五讲用来执行7z.exe这样7z.exe会按照函数的参数把文件夹压缩成zip文件并增加密码。
通过zipWithPassword()函数你就能够实现zipfile的扩展实现有密码文件压缩功能了。
## 小结
最后我来为你总结一下今天这节课的主要内容。我通过zipfile库和7zip软件分别实现了无密码压缩文件和有密码压缩文件。
无密码压缩文件更加简单方便,而有密码压缩文件更加安全,配合自动根据当前日期改变压缩文件名称,可以作为你进行每日数据自动化备份的主要工具。
除了备份功能的学习外我还为你讲解了新的函数返回方式yield和return不同的是yield返回的是生成器对象需要使用for迭代方式访问它的全部数据。yield语句除了可以和zipfile库一起实现数据备份外还经常被应用于互联网上的图片批量下载压缩场景中。
以上内容就是怎么实现无密码和有密码压缩的全部内容了,我将完整代码贴在下方中,一起提供给你,你可以直接修改需要备份的目录,完成你自己文件夹的一键备份脚本。
```
from zipfile import ZipFile
import os
import datetime
# 以年月日作为zip文件名
def genZipfilename():
today = datetime.date.today()
basename = today.strftime('%Y%m%d')
extname = "zip"
return f"{basename}.{extname}"
# 遍历目录,得到该目录下所有的子目录和文件
def getAllFiles(dir):
for root,dirs,files in os.walk(dir):
for file in files:
yield os.path.join(root, file)
# 无密码生成压缩文件
def zipWithoutPassword(files,backupFilename):
with ZipFile(backupFilename, 'w') as zf:
for f in files:
zf.write(f)
def zipWithPassword(dir, backupFilename, password=None):
cmd = f"7z.exe a -tzip {backupFilename} -p{password} {dir}"
status = os.popen(cmd)
return status
if __name__ == '__main__':
# 要备份的目录
backupDir = "/data"
# 要备份的文件
backupFiles = getAllFiles(backupDir)
# zip文件的名字“年月日.zip”
zipFilename = genZipfilename()
# 自动将要备份的目录制作成zip文件
zipWithoutPassword(backupFiles, zipFilename)
# 使用密码进行备份
zipWithPassword(backupDir, zipFilename, "password123")
```
## 思考题
按照惯例,我来为你留一道思考题,如果需要备份的是两个甚至更多的目录,你会怎么改造脚本呢?
欢迎把你的想法和思考分享在留言区,我们一起交流讨论。也欢迎你把课程分享给你的同事、朋友,我们一起做职场中的效率人。我们下节课再见!