1. 程式人生 > >【Python】python檔案開啟方式詳解——a、a+、r+、w+區別

【Python】python檔案開啟方式詳解——a、a+、r+、w+區別

第一步 排除檔案開啟方式錯誤:

r只讀,r+讀寫,不建立

w新建只寫,w+新建讀寫,二者都會將檔案內容清零

(以w方式開啟,不能讀出。w+可讀寫)

w+與r+區別:

r+:可讀可寫,若檔案不存在,報錯;w+: 可讀可寫,若檔案不存在,建立

r+與a+區別:

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取負值。

4. 記得close()關閉

當我們寫檔案時,作業系統往往不會立刻把資料寫入磁碟,而是放到記憶體快取起來,空閒的時候再慢慢寫入。只有呼叫close()方法時,作業系統才保證把沒有寫入的資料全部寫入磁碟。忘記呼叫close()的後果是資料可能只寫了一部分到磁碟,剩下的丟失了。所以,還是用with語句來得保險:

with open('/Users/michael/test.txt', 'w') as f:
    f.write('Hello, world!')

----------

看評論:對於'r+'來說,如果先讀取了內容,再寫入的話就變成了追加的模式,如果直接寫入內容,就是覆蓋了