草庐IT

Python基础(十三)——文件操作(open函数、close函数)

Mood Crows 2023-06-22 原文

本文以Python3以上为学习基础。

目录

1、 使用文件操作第一原则

2、open函数

2.1、文件打开模式

2.1.1、只读模式打开文件——只读(r)

2.1.2、读写模式打开文件——读写模式(r+)

2.1.3、写模式打开文件——写模式(w)

2.1.4、读写模式打开文件——读写模式(w+)

2.1.5、写模式打开文件(追加内容)——写模式(a)

2.1.6、读写模式打开文件(追加内容)——读写模式(a)

 2.2、二进制模式打开文件

3、close()函数


在Linux中所有的东西都存放在文件中。

下面讲解Python中文件的相关知识。

在某个文件夹下面建立了一个文件:zxc.txt。

并且在里面输入了如下内容: 

 我们现在打开文件,并且遍历文件的内容:

可以看到我们把文件的内容遍历出来了。

但是每行代码的内容什么含义呢?

我们一行一行的看。

第一行:

f = open("C:\\Users\\lenovo\\Desktop\\Python_File_test\\zxc.txt")

 第一行就是打开文件,采用的方法是open函数。但是在使用文件操作的时候我们要明确一个原则。

1、 使用文件操作第一原则

使用open函数一定要同时把文件关闭函数close()写上。

为什么呢?我们做个验证。

 这个就是我创建好的文件zxc.txt。

当前的文件内容是空的。

下面我们执行下面的代码。

注意:我只是打开了文件,没有关闭文件。

 为了方便查看。加了一个代码执行结束的提示。

下面我们查看zxc.txt文件的内容。

我们成功输出结果:1 2$

我们下面试着手动修改以下文件。

看到没有,我们的文件被占用。但是是什么占用了呢?

我们利用Windows的资源监视器看看。

 很明显,python占用这个文件。原因就是我们使用open函数带开了这个文件。没有关闭,这个文件一直被占用。

因此我们在使用open函数的时候一定要close文件。否则你一直开着它,就算你不懂也应该能想到一定有问题。具体情况后续介绍。

我们现在知道要close到文件了,但是怎么做呢?我们试试看。

 利用前面的文件句柄 f 进行close。

这时候我们使用Windows的资源监视器看看这个文件的相关进程。

 结果很明显。文件相关进程结束了。

所以我们只要写了open函数,一定要紧跟着写上close()函数。然后在open函数和close函数之间进行添加代码处理,就像下面这样。

这里文件关闭函数也就说了,格式就是:文件对象.close(),例如上例中的f.close()。

2、open函数

继续说open函数,open函数是Python的一个内置函数。用于创建一个file对象(文件对象)。

这里文件对象就是等号前面的 f。这样变量f跟对象文件 zxc.txt 用线连起来了(就是对象引用)。

关于对象的引用和放风筝一样。放风筝的转轮就是f,风筝就是zxc.txt,连接转轮和风筝的线就是Python。这里涉及到Python的深浅拷贝问题,具体可以参考:Python深浅拷贝。

我们现来看open函数的语法格式:

file object = open(file_name [, access_mode][, buffering]
  • file_name:就是你要访问的文件的路径。

在本例中就是zxc.txt的文件路径。

关于这个文件路径的写法,有两种:

(1)open("zxc.txt")

意味着这个文件zxc.txt是在当前文件夹内的。就是你Python程序的所在文件夹。

(2)open("C:\\Users\\lenovo\\Desktop\\Python_File_test\\zxc.txt")

这就是打开其它文件夹内的文件,用相对路径或者绝对路径来表示,从而让python能够找到那个文件。里面的 \ 用了 \\ 表示,是为了防止转义。


  • access_mode:access_mode决定了打开文件的模式:只读,写入,追加等。这个参数是非强制的,默认文件访问模式为只读(r)。

可以省略,默认的打开模式是只读(r)。本例中就是只读打开。

其他的文件打开模式参考下表:(表格来源于:菜鸟教程

模式描述
t文本模式 (默认)。
x写模式,新建一个文件,如果该文件已存在则会报错。
b二进制模式。
+打开一个文件进行更新(可读可写)。
U通用换行模式(不推荐)。
r以只读方式打开文件。文件的指针将会放在文件的开头。这是默认模式。
rb以二进制格式打开一个文件用于只读。文件指针将会放在文件的开头。这是默认模式。一般用于非文本文件如图片等。
r+打开一个文件用于读写。文件指针将会放在文件的开头。
rb+以二进制格式打开一个文件用于读写。文件指针将会放在文件的开头。一般用于非文本文件如图片等。
w打开一个文件只用于写入。如果该文件已存在则打开文件,并从开头开始编辑,即原有内容会被删除。如果该文件不存在,创建新文件。
wb以二进制格式打开一个文件只用于写入。如果该文件已存在则打开文件,并从开头开始编辑,即原有内容会被删除。如果该文件不存在,创建新文件。一般用于非文本文件如图片等。
w+打开一个文件用于读写。如果该文件已存在则打开文件,并从开头开始编辑,即原有内容会被删除。如果该文件不存在,创建新文件。
wb+以二进制格式打开一个文件用于读写。如果该文件已存在则打开文件,并从开头开始编辑,即原有内容会被删除。如果该文件不存在,创建新文件。一般用于非文本文件如图片等。
a打开一个文件用于追加。如果该文件已存在,文件指针将会放在文件的结尾。也就是说,新的内容将会被写入到已有内容之后。如果该文件不存在,创建新文件进行写入。
ab以二进制格式打开一个文件用于追加。如果该文件已存在,文件指针将会放在文件的结尾。也就是说,新的内容将会被写入到已有内容之后。如果该文件不存在,创建新文件进行写入。
a+打开一个文件用于读写。如果该文件已存在,文件指针将会放在文件的结尾。文件打开时会是追加模式。如果该文件不存在,创建新文件用于读写。
ab+以二进制格式打开一个文件用于追加。如果该文件已存在,文件指针将会放在文件的结尾。如果该文件不存在,创建新文件用于读写。

 文件打开模式有很多,我们总结一下主要用到的是下面几种。

模式可做操作若文件不存在是否覆盖文件原来内容
r只读报错——
r+可读、可写报错
w只写创建
w+ 可读、可写创建
 
a  只写创建否,追加写
a+可读、可写创建否,追加写

下面我们用例子来看,怎么写。(下面所有的例子,为了方便查看,我把文件创建在和Python程序所在的文件夹里面,这样可以直接写文件名称)


2.1、文件打开模式

2.1.1、只读模式打开文件——只读(r)

文件存在的场合:

 这里我尝试文件中写 hello world。报错意思没有写权限。

文件不存在的场合:

 很明显,出现了文件不存在的错误。

2.1.2、读写模式打开文件——读写模式(r+)

文件存在的场合,zxc.txt文件的内容:

下面我们用读写模式打开文件,并向里面写入hello world。

 这时候文件内容,注意看:光标在h前面。原来的内容被删除。

  

文件不存在的场合:

  很明显,出现了文件不存在的错误。

2.1.3、写模式打开文件——写模式(w)

文件存在的场合,zxc.txt的文件内容:

 下面我们用写模式打开文件,并向里面写入Python。

 文件的内容如下,仔细看:输入的光标位置在最开始。P前面有个光标。

很明显,删除原来的文件内容,重新写入Python,并且把光标放在文件最开始。

 文件不存在的场合:

可以看到我的文件夹里面没有zxc.txt。

  下面我们用写模式打开文件,并向里面写入Python。

 这时候我的文件夹:

 可以看到,自己创建了zxc.txt文件。

文件内容如下,仔细看:输入的光标位置在最开始。P前面有个光标。

 很明显,自己创建了文件,写入Python,并且把光标放在文件最开始。

既然w是写模式,我们试着读取文件内容看看。这时候的文件内容是Python。

很明显,出现文件不能读的错误。 

2.1.4、读写模式打开文件——读写模式(w+)

文件存在的场合,文件内容:

 既然是读写模式(w+),所以我们先读文件内容,再写入Python,然后再读取文件内容。

可以看到我们打印文件内容的时候没有报错,但是没有打印出任何内容?

那我们写入成功了吗?

很明显写入成功了,但是为什么没有打印出来内容?

其实是由于读写机制的问题,当一个文件被读或者写一次后,文件中文件指针指向的是整个文件的末尾,再次读时从末尾开始读的话肯定是什么都到。

文件不存在的场合:可以看到我的文件夹里面没有zxc.txt。

这时候我们验证读写模式(w+)。

可以看到创建了文件zxc.txt。 

 文件内容:


其实文件写模式(w)、读写模式(w+)还可以用作删除文件内容。

因为他们整个工作原理就是把文件原来的内容删除,然后写入新的内容。

如果我们写入的内容为空,那么不就是删除文件内容。

就如同下面这个例子。

文件内容: 

 可以看到,文件zxc.txt里面有无数个Python。

下面我们利用读写模式,但是不写入任何内容。

 这时候我们查看文件内容:

 文件内容是空,原来的内容删掉了。

同理,写模式(w)是一样的。


2.1.5、写模式打开文件(追加内容)——写模式(a)

文件存在的场合,文件内容如下:

 下面我们写入内容 hello world

 文件内容:

 可以看到,Python后面追加了hello world。

文件不存在的场合,可以看到,我的文件夹里没有文件zxc.txt:

 下面我们连续写入内容 hello world 和 Python。

可以看到zxc.txt被创建 

 文件内容:

 既然是写模式,我们验证一下zxc.txt不能读。

出现错误不能读。

2.1.6、读写模式打开文件(追加内容)——读写模式(a)

文件存在的场合,文件内容如下:

 下面我们写入内容 hello world

 文件内容:

 可以看到,Python后面追加了hello world。

文件不存在的场合,可以看到,我的文件夹里没有文件zxc.txt:

 下面我们连续写入内容 hello world 和 Python。

可以看到zxc.txt被创建 

 文件内容:

 既然是读写模式,我们验证一下zxc.txt可读。

 没有出现错误,但是没有打印出内容。

原因其实是由于读写机制的问题,当一个文件被读或者写一次后,文件中文件指针指向的是整个文件的末尾,再次读时从末尾开始读的话肯定是什么都到。


 所有模式我们都进行了说明,我们再看一下回顾一下我们整理的文件打开模式表:

模式可做操作若文件不存在是否覆盖文件原来内容
r只读报错——
r+可读、可写报错
w只写创建
w+ 可读、可写创建
 
a  只写创建否,追加写
a+可读、可写创建否,追加写

对于第三个参数buffering。

buffering的可取值有0,1, >1三个

  • 0代表buffer关闭(只适用于二进制模式,也就是我们文件打开模式中带b的,例如rb【二进制只读模式】)
  • 1代表line buffer(只适用于文本模式,也就是我们文件打开模式中的t)
  • >1表示初始化的buffer大小 

这个参数我没用到过,所以这里我无法进行举例解释了,所以如果需要,还请自行查询。

 2.2、二进制模式打开文件

 我们看到了在文件打开模式中有以下模式:rb、wb……有这种带 b 的。

什么意思呢?

就是用二进制的方式打开文件。

有什么用?

我们来验证一下:
创建文件zxc.txt,内容如下:

下面读取文件:

很明显,我们出现了以下错误:

UnicodeDecodeError: 'gbk' codec can't decode byte 0x82 in position 0: incomplete multibyte sequence

主要原因是因为编码的问题,可能是因为0x82这个字节在gbk编码中没有这个字符,可能原字符是两个字节,在gbk里被解析成了一个字节,导致字符不存在。

解决方法有两个:一个是二进制读取,一个是改编编码方式。

二进制读取就是利用rb、wb……

这就是我们文件打开方式有二进制的原因。

下面我们进行验证。

可以看到,二进制就把文件内容读出来了。并且看到原来是中文的句号在gbk里面没有。

前面的r、w、r+……就是上面讲的读模式、写模式、读写模式等等。配合二进制方式,就可以读取文件了。

3、close()函数

上面文件打开模式例子都有close函数的用法:f.close()。

有关Python基础(十三)——文件操作(open函数、close函数)的更多相关文章

  1. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

    我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

  2. ruby - 其他文件中的 Rake 任务 - 2

    我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时

  3. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

    我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

  4. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  5. ruby-on-rails - Rails 3 中的多个路由文件 - 2

    Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

  6. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  7. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  8. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  9. Ruby 写入和读取对象到文件 - 2

    好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信

  10. ruby - 如何使用 Ruby aws/s3 Gem 生成安全 URL 以从 s3 下载文件 - 2

    我正在编写一个小脚本来定位aws存储桶中的特定文件,并创建一个临时验证的url以发送给同事。(理想情况下,这将创建类似于在控制台上右键单击存储桶中的文件并复制链接地址的结果)。我研究过回形针,它似乎不符合这个标准,但我可能只是不知道它的全部功能。我尝试了以下方法:defauthenticated_url(file_name,bucket)AWS::S3::S3Object.url_for(file_name,bucket,:secure=>true,:expires=>20*60)end产生这种类型的结果:...-1.amazonaws.com/file_path/file.zip.A

随机推荐