python文件打开方式详解——a、a+、r+、w+区别

第一步 排除文件打开方式错误:

r只读,r+读写,不创建

w新建只写,w+新建读写,二者都会将文件内容清零

(以w方式打开,不能读出。w+可读写)

**w+与r+区别:

r+:可读可写,若文件不存在,报错;w+: 可读可写,若文件不存在,创建

r+与a+区别:

[python]fd = open("1.txt",'w+')  

fd.write('123')  
fd = open("1.txt",'r+')  
fd.write('456')  
fd = open("1.txt",'a+')  
fd.write('789')  

结果:456789

说明r+进行了覆盖写。

以a,a+的方式打开文件,附加方式打开

a附加写方式打开,不可读;a+: 附加读写方式打开)

以 'U' 标志打开文件, 所有的行分割符通过 Python 的输入方法(例#如 read*() ),返回时都会被替换为换行符\n. ('rU' 模式也支持 'rb' 选项) . 

r和U要求文件必须存在

不可读的打开方式w和a

若不存在会创建新文件的打开方式:a,a+,w,w+

>>> fd=open(r'f:\mypython\test.py','w')    #只读方式打开,读取报错 
>>> fd.read()  
Traceback (most recent call last):  
  File "<stdin>", line 1, in <module>  
IOError: File not open for reading  
>>> fd=open(r'f:\mypython\test.py','a')#附加写方式打开,读取报错 
>>> fd.read()  
Traceback (most recent call last):  
  File "<stdin>", line 1, in <module>  
IOError: File not open for reading  
>>></span></span></span>  

2.正确读写方式打开,出现乱码

>>> fd=open(r'f:\mypython\test.py','a+')  
>>> fd.write('123')  
>>> fd.read()  
>>> fd.close()  

close之前,手动打开文件,什么都没写入;close后,手动打开文件,乱码:123嚅?     

原因分析:指针问题。open()以a+模式开启了一个附加读写模式的文件,由于是a,所以指针在文件末尾。此时如果做read(),则python发现指针位置就是EOF,读取到空字符串。

在写入123之后,指针的位置是4,仍然是文件尾,文件在内存中是123[EOF]。

但看起来read()的时候,Python仍然去试图在磁盘的文件上,将指针从文件头向后跳3,再去读取到EOF为止。

也就是说,你实际上是跳过了该文件真正的EOF,为硬盘底层的数据做了一个dump,一直dump到了一个从前存盘文件的[EOF]为止。所以最后得到了一些根本不期待的随机乱字符,而不是编码问题造成的乱码。

解决方案:读取之前将指针重置为文件头(如果读取之后重置再读,无效)

>>> fd=open(r'f:\mypython\test.py','a+')  
>>> fd.seek(0)  
>>> fd.read()  
'123'<span style="white-space:pre">           </span>#顺利读出</span></span> 

3.文件里有内容,却读出空字符

>>> fd=open(r'f:\mypython\test.py','w+') #清空内容,重新写入 
>>> fd.write('456')  
>>> fd.flush()<span style="white-space:pre">     </span>#确定写入,此时文件内容为“456” 
>>> fd.read()  
'' #读出空 

原因:同样是指针问题,写入后指针指向末尾[EOF],因此读出空

解决方案一、调用close后重新打开,指针位于开头。(r,r+,a+,U都可以,注意不要用w,w+,a打开)

>>> fd.close()  
>>> fd=open(r'f:\mypython\test.py','a+')  
>>> fd.read()  
'456' 
>>> fd.close()  
>>> fd=open(r'f:\mypython\test.py','r+')  
>>> fd.read()  
'456'<pre name="code" class="python">>>> fd.close()  
>>> fd=open(r'f:\mypython\test.py','r')  
>>> fd.read()  
'456' 
>>> fd.close()  
>>> fd=open(r'f:\mypython\test.py','U')  
>>> fd.read()  
'456' 

解决方案二、调用seek指向开头

>>> fd=open(r'f:\mypython\test.py','w+')  
>>> fd.write('456')  
>>> fd.seek(0)  
>>> fd.read()  
'456' 

seek函数     seek(offset[, whence]) ,offset是相对于某个位置的偏移量。位置由whence决定,默认whence=0,从开头起;whence=1,从当前位置算起;whence=2相对于文件末尾移动,通常offset取负值。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏MasiMaro 的技术博文

C/C++中define定义的常量与const常量

常量是在程序中不能更改的量,在C/C++中有两种方式定义常量,一种是利用define宏定义的方式,一种是C++中新提出来的const型常变量,下面主要讨论它们之...

1071
来自专栏前端布道

MongoDB初识

什么是MongoDB MongoDB 是由C++语言编写的,是一个基于分布式文件存储的开源数据库系统。 在高负载的情况下,添加更多的节点,可以保证服务器性能。 ...

3838
来自专栏北京马哥教育

Linux Shell 通配符、元字符、转义符最全使用攻略

来自:程默 - 博客园,作者:程默 www.cnblogs.com/chengmo/archive/2010/10/17/1853344.html 说到she...

7306
来自专栏JMCui

Git 中 .gitignore 的配置语法

    在日常的开发中,当我们需要将一个项目提交到 Git 时,并不是所有的文件都需要提交,比如一些自动生成的文件,类似于 .idea 文件、class 文件等...

2223
来自专栏企鹅号快讯

Python的进程

进程 说明:本文是基于Py2.X环境, Python实现多进程的方式主要有两种:一种方法是使用os模块中的fork方法; 另一种是使用multiprocessi...

24210
来自专栏思考的代码世界

Python编程从入门到实践之文件和异常处理|第12天

要使用文本文件中的信息,首先需要将信息读取到内存中。为此,我们可以一次性读取文件的 全部内容,也可以以每次一行的方式逐步读取。

4407
来自专栏IMWeb前端团队

hapi vs. express [1] —— 路由

hapi vs. express [1] —— 路由 前言 express 与 hapi 是两个基于 nodejs 的 web server 开发框架,它们由于...

2255
来自专栏抠抠空间

django之urls系统

Django的urls系统简介 Django 1.11版本 URLConf官方文档:https://docs.djangoproject.com/en/1.11...

2937
来自专栏思考的代码世界

Python基础学习09天

1686
来自专栏小二的折腾日记

day5(面向对象2)

wait notify notifyAll 都使用在同步中,因为要对持有监视器(锁)的线程操作。所以要使用在同步中,以为只有同步才具有锁。 为什么这些操作线程...

561

扫码关注云+社区

领取腾讯云代金券