1. 程式人生 > 實用技巧 >Python 檔案I/O 檔案讀寫模式r,r+,w,w+,a,a+的區別

Python 檔案I/O 檔案讀寫模式r,r+,w,w+,a,a+的區別

  • http://kuanghy.github.io/python-os/
  • http://python.usyiyi.cn/python_278/library/os.html

Python File(檔案) 方法

open() 方法

Python open() 方法用於開啟一個檔案,並返回檔案物件,在對檔案進行處理過程都需要使用到這個函式,如果該檔案無法被開啟,會丟擲 OSError。

注意:使用 open() 方法一定要保證關閉檔案物件,即呼叫 close() 方法。

open() 函式常用形式是接收兩個引數:檔名(file)和模式(mode)。

open(file, mode='r')

完整的語法格式為:

open(file, mode='r', buffering=-1, encoding=None, errors=None, newline=None, closefd=True, opener=None)

引數說明:

  • file: 必需,檔案路徑(相對或者絕對路徑)。
  • mode: 可選,檔案開啟模式
  • buffering: 設定緩衝
  • encoding: 一般使用utf8
  • errors: 報錯級別
  • newline: 區分換行符
  • closefd: 傳入的file引數型別
  • opener: 設定自定義開啟器,開啟器的返回值必須是一個開啟的檔案描述符。

mode 引數有:

模式描述
t 文字模式 (預設)。
x 寫模式,新建一個檔案,如果該檔案已存在則會報錯。
b 二進位制模式。
+ 開啟一個檔案進行更新(可讀可寫)。
U 通用換行模式(不推薦)。
r 以只讀方式開啟檔案。檔案的指標將會放在檔案的開頭。這是預設模式。
rb 以二進位制格式開啟一個檔案用於只讀。檔案指標將會放在檔案的開頭。這是預設模式。一般用於非文字檔案如圖片等。
r+ 開啟一個檔案用於讀寫。檔案指標將會放在檔案的開頭。
rb+ 以二進位制格式開啟一個檔案用於讀寫。檔案指標將會放在檔案的開頭。一般用於非文字檔案如圖片等。
w 開啟一個檔案只用於寫入。如果該檔案已存在則開啟檔案,並從開頭開始編輯,即原有內容會被刪除。如果該檔案不存在,建立新檔案。
wb 以二進位制格式開啟一個檔案只用於寫入。如果該檔案已存在則開啟檔案,並從開頭開始編輯,即原有內容會被刪除。如果該檔案不存在,建立新檔案。一般用於非文字檔案如圖片等。
w+ 開啟一個檔案用於讀寫。如果該檔案已存在則開啟檔案,並從開頭開始編輯,即原有內容會被刪除。如果該檔案不存在,建立新檔案。
wb+ 以二進位制格式開啟一個檔案用於讀寫。如果該檔案已存在則開啟檔案,並從開頭開始編輯,即原有內容會被刪除。如果該檔案不存在,建立新檔案。一般用於非文字檔案如圖片等。
a 開啟一個檔案用於追加。如果該檔案已存在,檔案指標將會放在檔案的結尾。也就是說,新的內容將會被寫入到已有內容之後。如果該檔案不存在,建立新檔案進行寫入。
ab 以二進位制格式開啟一個檔案用於追加。如果該檔案已存在,檔案指標將會放在檔案的結尾。也就是說,新的內容將會被寫入到已有內容之後。如果該檔案不存在,建立新檔案進行寫入。
a+ 開啟一個檔案用於讀寫。如果該檔案已存在,檔案指標將會放在檔案的結尾。檔案開啟時會是追加模式。如果該檔案不存在,建立新檔案用於讀寫。
ab+ 以二進位制格式開啟一個檔案用於追加。如果該檔案已存在,檔案指標將會放在檔案的結尾。如果該檔案不存在,建立新檔案用於讀寫。

預設為文字模式,如果要以二進位制模式開啟,加上b。

file 物件

file 物件使用 open 函式來建立,下表列出了 file 物件常用的函式:

序號方法及描述
1

file.close()

關閉檔案。關閉後文件不能再進行讀寫操作。

2

file.flush()

重新整理檔案內部緩衝,直接把內部緩衝區的資料立刻寫入檔案, 而不是被動的等待輸出緩衝區寫入。

3

file.fileno()

返回一個整型的檔案描述符(file descriptor FD 整型), 可以用在如os模組的read方法等一些底層操作上。

4

file.isatty()

如果檔案連線到一個終端裝置返回 True,否則返回 False。

5

file.next()

返回檔案下一行。

6

file.read([size])

從檔案讀取指定的位元組數,如果未給定或為負則讀取所有。

7

file.readline([size])

讀取整行,包括 "\n" 字元。

8

file.readlines([sizeint])

讀取所有行並返回列表,若給定sizeint>0,則是設定一次讀多少位元組,這是為了減輕讀取壓力。

9

file.seek(offset[, whence])

設定檔案當前位置

10

file.tell()

返回檔案當前位置。

11

file.truncate([size])

擷取檔案,擷取的位元組通過size指定,預設為當前檔案位置。

12

file.write(str)

將字串寫入檔案,返回的是寫入的字元長度。

13

file.writelines(sequence)

向檔案寫入一個序列字串列表,如果需要換行則要自己加入每行的換行符。

Python OS 檔案/目錄方法

os模組提供了非常豐富的方法用來處理檔案和目錄。常用的方法如下表所示:

序號方法及描述
1

os.access(path, mode)


檢驗許可權模式
2

os.chdir(path)


改變當前工作目錄
3

os.chflags(path, flags)


設定路徑的標記為數字標記。
4

os.chmod(path, mode)


更改許可權
5

os.chown(path, uid, gid)


更改檔案所有者
6

os.chroot(path)


改變當前程序的根目錄
7

os.close(fd)


關閉檔案描述符 fd
8

os.closerange(fd_low, fd_high)


關閉所有檔案描述符,從 fd_low (包含) 到 fd_high (不包含), 錯誤會忽略
9

os.dup(fd)


複製檔案描述符 fd
10

os.dup2(fd, fd2)


將一個檔案描述符 fd 複製到另一個 fd2
11

os.fchdir(fd)


通過檔案描述符改變當前工作目錄
12

os.fchmod(fd, mode)


改變一個檔案的訪問許可權,該檔案由引數fd指定,引數mode是Unix下的檔案訪問許可權。
13

os.fchown(fd, uid, gid)


修改一個檔案的所有權,這個函式修改一個檔案的使用者ID和使用者組ID,該檔案由檔案描述符fd指定。
14

os.fdatasync(fd)


強制將檔案寫入磁碟,該檔案由檔案描述符fd指定,但是不強制更新檔案的狀態資訊。
15

os.fdopen(fd[, mode[, bufsize]])


通過檔案描述符 fd 建立一個檔案物件,並返回這個檔案物件
16

os.fpathconf(fd, name)


返回一個開啟的檔案的系統配置資訊。name為檢索的系統配置的值,它也許是一個定義系統值的字串,這些名字在很多標準中指定(POSIX.1, Unix 95, Unix 98, 和其它)。
17

os.fstat(fd)


返回檔案描述符fd的狀態,像stat()。
18

os.fstatvfs(fd)


返回包含檔案描述符fd的檔案的檔案系統的資訊,像 statvfs()
19

os.fsync(fd)


強制將檔案描述符為fd的檔案寫入硬碟。
20

os.ftruncate(fd, length)


裁剪檔案描述符fd對應的檔案, 所以它最大不能超過檔案大小。
21

os.getcwd()


返回當前工作目錄
22

os.getcwdu()


返回一個當前工作目錄的Unicode物件
23

os.isatty(fd)


如果檔案描述符fd是開啟的,同時與tty(-like)裝置相連,則返回true, 否則False。
24

os.lchflags(path, flags)


設定路徑的標記為數字標記,類似 chflags(),但是沒有軟連結
25

os.lchmod(path, mode)


修改連線檔案許可權
26

os.lchown(path, uid, gid)


更改檔案所有者,類似 chown,但是不追蹤連結。
27

os.link(src, dst)


建立硬連結,名為引數 dst,指向引數 src
28

os.listdir(path)


返回path指定的資料夾包含的檔案或資料夾的名字的列表。
29

os.lseek(fd, pos, how)


設定檔案描述符 fd當前位置為pos, how方式修改: SEEK_SET 或者 0 設定從檔案開始的計算的pos; SEEK_CUR或者 1 則從當前位置計算; os.SEEK_END或者2則從檔案尾部開始. 在unix,Windows中有效
30

os.lstat(path)


像stat(),但是沒有軟連結
31

os.major(device)


從原始的裝置號中提取裝置major號碼 (使用stat中的st_dev或者st_rdev field)。
32

os.makedev(major, minor)


以major和minor裝置號組成一個原始裝置號
33

os.makedirs(path[, mode])


遞迴資料夾建立函式。像mkdir(), 但建立的所有intermediate-level資料夾需要包含子資料夾。
34

os.minor(device)


從原始的裝置號中提取裝置minor號碼 (使用stat中的st_dev或者st_rdev field )。
35

os.mkdir(path[, mode])


以數字mode的mode建立一個名為path的資料夾.預設的 mode 是 0777 (八進位制)。
36

os.mkfifo(path[, mode])


建立命名管道,mode 為數字,預設為 0666 (八進位制)
37

os.mknod(filename[, mode=0600, device])
建立一個名為filename檔案系統節點(檔案,裝置特別檔案或者命名pipe)。

38

os.open(file, flags[, mode])


開啟一個檔案,並且設定需要的開啟選項,mode引數是可選的
39

os.openpty()


開啟一個新的偽終端對。返回 pty 和 tty的檔案描述符。
40

os.pathconf(path, name)


返回相關檔案的系統配置資訊。
41

os.pipe()


建立一個管道. 返回一對檔案描述符(r, w) 分別為讀和寫
42

os.popen(command[, mode[, bufsize]])


從一個 command 開啟一個管道
43

os.read(fd, n)


從檔案描述符 fd 中讀取最多 n 個位元組,返回包含讀取位元組的字串,檔案描述符 fd對應檔案已達到結尾, 返回一個空字串。
44

os.readlink(path)


返回軟連結所指向的檔案
45

os.remove(path)


刪除路徑為path的檔案。如果path 是一個資料夾,將丟擲OSError; 檢視下面的rmdir()刪除一個 directory。
46

os.removedirs(path)


遞迴刪除目錄。
47

os.rename(src, dst)


重新命名檔案或目錄,從 src 到 dst
48

os.renames(old, new)


遞迴地對目錄進行更名,也可以對檔案進行更名。
49

os.rmdir(path)


刪除path指定的空目錄,如果目錄非空,則丟擲一個OSError異常。
50

os.stat(path)


獲取path指定的路徑的資訊,功能等同於C API中的stat()系統呼叫。
51

os.stat_float_times([newvalue])
決定stat_result是否以float物件顯示時間戳

52

os.statvfs(path)


獲取指定路徑的檔案系統統計資訊
53

os.symlink(src, dst)


建立一個軟連結
54

os.tcgetpgrp(fd)


返回與終端fd(一個由os.open()返回的開啟的檔案描述符)關聯的程序組
55

os.tcsetpgrp(fd, pg)


設定與終端fd(一個由os.open()返回的開啟的檔案描述符)關聯的程序組為pg。
56

os.tempnam([dir[, prefix]])


返回唯一的路徑名用於建立臨時檔案。
57

os.tmpfile()


返回一個開啟的模式為(w+b)的檔案物件 .這檔案物件沒有資料夾入口,沒有檔案描述符,將會自動刪除。
58

os.tmpnam()


為建立一個臨時檔案返回一個唯一的路徑
59

os.ttyname(fd)


返回一個字串,它表示與檔案描述符fd 關聯的終端裝置。如果fd 沒有與終端裝置關聯,則引發一個異常。
60

os.unlink(path)


刪除檔案路徑
61

os.utime(path, times)


返回指定的path檔案的訪問和修改的時間。
62

os.walk(top[, topdown=True[, onerror=None[, followlinks=False]]])


輸出在資料夾中的檔名通過在樹中游走,向上或者向下。
63

os.write(fd, str)


寫入字串到檔案描述符 fd中. 返回實際寫入的字串長度
64

os.path 模組


獲取檔案的屬性資訊。

參考地址:

    • http://kuanghy.github.io/python-os/
    • http://python.usyiyi.cn/python_278/library/os.html

Python 檔案I/O

本章只講述所有基本的 I/O 函式,更多函式請參考Python標準文件。

列印到螢幕

最簡單的輸出方法是用print語句,你可以給它傳遞零個或多個用逗號隔開的表示式。此函式把你傳遞的表示式轉換成一個字串表示式,並將結果寫到標準輸出如下:

#!/usr/bin/python
# -*- coding: UTF-8 -*- 

print "Python 是一個非常棒的語言,不是嗎?"

你的標準螢幕上會產生以下結果:

Python 是一個非常棒的語言,不是嗎?

讀取鍵盤輸入

Python提供了兩個內建函式從標準輸入讀入一行文字,預設的標準輸入是鍵盤。如下:

  • raw_input
  • input

raw_input函式

raw_input([prompt]) 函式從標準輸入讀取一個行,並返回一個字串(去掉結尾的換行符):

#!/usr/bin/python
# -*- coding: UTF-8 -*- 
 
str = raw_input("請輸入:")
print "你輸入的內容是: ", str

這將提示你輸入任意字串,然後在螢幕上顯示相同的字串。當我輸入"Hello Python!",它的輸出如下:

請輸入:Hello Python
你輸入的內容是:  Hello Python

input函式

input([prompt])函式和raw_input([prompt])函式基本類似,但是 input 可以接收一個Python表示式作為輸入,並將運算結果返回。

#!/usr/bin/python
# -*- coding: UTF-8 -*- 
 
str = input("請輸入:")
print "你輸入的內容是: ", str

這會產生如下的對應著輸入的結果:

請輸入:[x*5 for x in range(2,10,2)]
你輸入的內容是:  [10, 20, 30, 40]

開啟和關閉檔案

現在,您已經可以向標準輸入和輸出進行讀寫。現在,來看看怎麼讀寫實際的資料檔案。

Python 提供了必要的函式和方法進行預設情況下的檔案基本操作。你可以用file物件做大部分的檔案操作。

open 函式

你必須先用Python內建的open()函式開啟一個檔案,建立一個file物件,相關的方法才可以呼叫它進行讀寫。

語法:

file object = open(file_name [, access_mode][, buffering])

各個引數的細節如下:

  • file_name:file_name變數是一個包含了你要訪問的檔名稱的字串值。
  • access_mode:access_mode決定了開啟檔案的模式:只讀,寫入,追加等。所有可取值見如下的完全列表。這個引數是非強制的,預設檔案訪問模式為只讀(r)。
  • buffering:如果buffering的值被設為0,就不會有寄存。如果buffering的值取1,訪問檔案時會寄存行。如果將buffering的值設為大於1的整數,表明了這就是的寄存區的緩衝大小。如果取負值,寄存區的緩衝大小則為系統預設。

不同模式開啟檔案的完全列表:

模式描述
t 文字模式 (預設)。
x 寫模式,新建一個檔案,如果該檔案已存在則會報錯。
b 二進位制模式。
+ 開啟一個檔案進行更新(可讀可寫)。
U 通用換行模式(不推薦)。
r 以只讀方式開啟檔案。檔案的指標將會放在檔案的開頭。這是預設模式。
rb 以二進位制格式開啟一個檔案用於只讀。檔案指標將會放在檔案的開頭。這是預設模式。一般用於非文字檔案如圖片等。
r+ 開啟一個檔案用於讀寫。檔案指標將會放在檔案的開頭。
rb+ 以二進位制格式開啟一個檔案用於讀寫。檔案指標將會放在檔案的開頭。一般用於非文字檔案如圖片等。
w 開啟一個檔案只用於寫入。如果該檔案已存在則開啟檔案,並從開頭開始編輯,即原有內容會被刪除。如果該檔案不存在,建立新檔案。
wb 以二進位制格式開啟一個檔案只用於寫入。如果該檔案已存在則開啟檔案,並從開頭開始編輯,即原有內容會被刪除。如果該檔案不存在,建立新檔案。一般用於非文字檔案如圖片等。
w+ 開啟一個檔案用於讀寫。如果該檔案已存在則開啟檔案,並從開頭開始編輯,即原有內容會被刪除。如果該檔案不存在,建立新檔案。
wb+ 以二進位制格式開啟一個檔案用於讀寫。如果該檔案已存在則開啟檔案,並從開頭開始編輯,即原有內容會被刪除。如果該檔案不存在,建立新檔案。一般用於非文字檔案如圖片等。
a 開啟一個檔案用於追加。如果該檔案已存在,檔案指標將會放在檔案的結尾。也就是說,新的內容將會被寫入到已有內容之後。如果該檔案不存在,建立新檔案進行寫入。
ab 以二進位制格式開啟一個檔案用於追加。如果該檔案已存在,檔案指標將會放在檔案的結尾。也就是說,新的內容將會被寫入到已有內容之後。如果該檔案不存在,建立新檔案進行寫入。
a+ 開啟一個檔案用於讀寫。如果該檔案已存在,檔案指標將會放在檔案的結尾。檔案開啟時會是追加模式。如果該檔案不存在,建立新檔案用於讀寫。
ab+ 以二進位制格式開啟一個檔案用於追加。如果該檔案已存在,檔案指標將會放在檔案的結尾。如果該檔案不存在,建立新檔案用於讀寫。

下圖很好的總結了這幾種模式:

模式rr+ww+aa+
+ + + +
+ + + + +
建立 + + + +
覆蓋 + +
指標在開始 + + + +
指標在結尾 + +

File物件的屬性

一個檔案被開啟後,你有一個file物件,你可以得到有關該檔案的各種資訊。

以下是和file物件相關的所有屬性的列表:

屬性描述
file.closed 返回true如果檔案已被關閉,否則返回false。
file.mode 返回被開啟檔案的訪問模式。
file.name 返回檔案的名稱。
file.softspace 如果用print輸出後,必須跟一個空格符,則返回false。否則返回true。

如下例項:

#!/usr/bin/python
# -*- coding: UTF-8 -*-
 
# 開啟一個檔案
fo = open("foo.txt", "w")
print "檔名: ", fo.name
print "是否已關閉 : ", fo.closed
print "訪問模式 : ", fo.mode
print "末尾是否強制加空格 : ", fo.softspace

以上例項輸出結果:

檔名:  foo.txt
是否已關閉 :  False
訪問模式 :  w
末尾是否強制加空格 :  0

close()方法

File 物件的 close()方法重新整理緩衝區裡任何還沒寫入的資訊,並關閉該檔案,這之後便不能再進行寫入。

當一個檔案物件的引用被重新指定給另一個檔案時,Python 會關閉之前的檔案。用 close()方法關閉檔案是一個很好的習慣。

語法:

fileObject.close()

例子:

#!/usr/bin/python
# -*- coding: UTF-8 -*-
 
# 開啟一個檔案
fo = open("foo.txt", "w")
print "檔名: ", fo.name
 
# 關閉開啟的檔案
fo.close()

以上例項輸出結果:

檔名:  foo.txt

讀寫檔案:

file物件提供了一系列方法,能讓我們的檔案訪問更輕鬆。來看看如何使用read()和write()方法來讀取和寫入檔案。

write()方法

write()方法可將任何字串寫入一個開啟的檔案。需要重點注意的是,Python字串可以是二進位制資料,而不是僅僅是文字。

write()方法不會在字串的結尾新增換行符('\n'):

語法:

fileObject.write(string)

在這裡,被傳遞的引數是要寫入到已開啟檔案的內容。

例子:

#!/usr/bin/python
# -*- coding: UTF-8 -*-
 
# 開啟一個檔案
fo = open("foo.txt", "w")
fo.write( "www.runoob.com!\nVery good site!\n")
 
# 關閉開啟的檔案
fo.close()

上述方法會建立foo.txt檔案,並將收到的內容寫入該檔案,並最終關閉檔案。如果你開啟這個檔案,將看到以下內容:

$ cat foo.txt 
www.runoob.com!
Very good site!

read()方法

read()方法從一個開啟的檔案中讀取一個字串。需要重點注意的是,Python字串可以是二進位制資料,而不是僅僅是文字。

語法:

fileObject.read([count])

在這裡,被傳遞的引數是要從已開啟檔案中讀取的位元組計數。該方法從檔案的開頭開始讀入,如果沒有傳入count,它會嘗試儘可能多地讀取更多的內容,很可能是直到檔案的末尾。

例子:

這裡我們用到以上建立的 foo.txt 檔案。

#!/usr/bin/python
# -*- coding: UTF-8 -*-
 
# 開啟一個檔案
fo = open("foo.txt", "r+")
str = fo.read(10)
print "讀取的字串是 : ", str
# 關閉開啟的檔案
fo.close()

以上例項輸出結果:

讀取的字串是 :  www.runoob

檔案位置:


檔案定位

tell()方法告訴你檔案內的當前位置, 換句話說,下一次的讀寫會發生在檔案開頭這麼多位元組之後。

seek(offset [,from])方法改變當前檔案的位置。Offset變量表示要移動的位元組數。From變數指定開始移動位元組的參考位置。

如果from被設為0,這意味著將檔案的開頭作為移動位元組的參考位置。如果設為1,則使用當前的位置作為參考位置。如果它被設為2,那麼該檔案的末尾將作為參考位置。

例子:

就用我們上面建立的檔案foo.txt。

#!/usr/bin/python
# -*- coding: UTF-8 -*-
 
# 開啟一個檔案
fo = open("foo.txt", "r+")
str = fo.read(10)
print "讀取的字串是 : ", str
 
# 查詢當前位置
position = fo.tell()
print "當前檔案位置 : ", position
 
# 把指標再次重新定位到檔案開頭
position = fo.seek(0, 0)
str = fo.read(10)
print "重新讀取字串 : ", str
# 關閉開啟的檔案
fo.close()

以上例項輸出結果:

讀取的字串是 :  www.runoob
當前檔案位置 :  10
重新讀取字串 :  www.runoob

重新命名和刪除檔案

Python的os模組提供了幫你執行檔案處理操作的方法,比如重新命名和刪除檔案。

要使用這個模組,你必須先匯入它,然後才可以呼叫相關的各種功能。

rename()方法:

rename()方法需要兩個引數,當前的檔名和新檔名。

語法:

os.rename(current_file_name, new_file_name)

例子:

下例將重新命名一個已經存在的檔案test1.txt。

#!/usr/bin/python
# -*- coding: UTF-8 -*-

import os
 
# 重新命名檔案test1.txt到test2.txt。
os.rename( "test1.txt", "test2.txt" )

remove()方法

你可以用remove()方法刪除檔案,需要提供要刪除的檔名作為引數。

語法:

os.remove(file_name)

例子:

下例將刪除一個已經存在的檔案test2.txt。

#!/usr/bin/python
# -*- coding: UTF-8 -*-

import os
 
# 刪除一個已經存在的檔案test2.txt
os.remove("test2.txt")

Python裡的目錄:

所有檔案都包含在各個不同的目錄下,不過Python也能輕鬆處理。os模組有許多方法能幫你建立,刪除和更改目錄。

mkdir()方法

可以使用os模組的mkdir()方法在當前目錄下建立新的目錄們。你需要提供一個包含了要建立的目錄名稱的引數。

語法:

os.mkdir("newdir")

例子:

下例將在當前目錄下建立一個新目錄test。

#!/usr/bin/python
# -*- coding: UTF-8 -*-

import os
 
# 建立目錄test
os.mkdir("test")

chdir()方法

可以用chdir()方法來改變當前的目錄。chdir()方法需要的一個引數是你想設成當前目錄的目錄名稱。

語法:

os.chdir("newdir")

例子:

下例將進入"/home/newdir"目錄。

#!/usr/bin/python
# -*- coding: UTF-8 -*-

import os
 
# 將當前目錄改為"/home/newdir"
os.chdir("/home/newdir")

getcwd()方法:

getcwd()方法顯示當前的工作目錄。

語法:

os.getcwd()

例子:

下例給出當前目錄:

#!/usr/bin/python
# -*- coding: UTF-8 -*-

import os
 
# 給出當前的目錄
print os.getcwd()

rmdir()方法

rmdir()方法刪除目錄,目錄名稱以引數傳遞。

在刪除這個目錄之前,它的所有內容應該先被清除。

語法:

os.rmdir('dirname')

例子:

以下是刪除" /tmp/test"目錄的例子。目錄的完全合規的名稱必須被給出,否則會在當前目錄下搜尋該目錄。

#!/usr/bin/python
# -*- coding: UTF-8 -*-

import os
 
# 刪除”/tmp/test”目錄
os.rmdir( "/tmp/test"  )

檔案、目錄相關的方法

File 物件和 OS 物件提供了很多檔案與目錄的操作方法,可以通過點選下面連結檢視詳情:

python 檔案讀寫模式r,r+,w,w+,a,a+的區別

如下表

模式 可做操作 若檔案不存在 是否覆蓋
r 只能讀 報錯 -
r+ 可讀可寫 報錯
w 只能寫 建立
w+  可讀可寫 建立
a   只能寫 建立 否,追加寫
a+ 可讀可寫 建立 否,追加寫

1.只讀模式(r)一個存在的檔案:

deffile_operation(): with open('/wzd/test.txt', mode='r') as f:#f.write('abc')r =f.readlines()

printr

print'---done---'file_operation()

正常輸出:

2.只讀模式(r)一個不存在的檔案:

deffile_operation(): with open('/wzd/test001.txt', mode='r') as f:#f.write('abc')r =f.readlines()

printr

print'---done---'file_operation()

注意上面的檔名字變了,輸出如下:

3.只讀模式去寫檔案:

deffile_operation(): with open('/wzd/test.txt', mode='r') as f: f.write('abc') r=f.readlines()

printr

print'---done---'file_operation()

w,r,wt,rt都是python裡面檔案操作的模式。
w是寫模式,r是讀模式。
t是windows平臺特有的所謂text mode(文字模式),區別在於會自動識別windows平臺的換行符。
類Unix平臺的換行符是\n,而windows平臺用的是\r\n兩個ASCII字元來表示換行,python內部採用的是\n來表示換行符。
rt模式下,python在讀取文字時會自動把\r\n轉換成\n.
wt模式下,Python寫檔案時會用\r\n來表示換行。



access_mode:開啟方式,r讀,w寫,a追加,r+ w+ a+ 都是以讀寫方式開啟,rb二進位制讀,wb二進位制寫,rb+ wb+ ab+二進位制讀寫
buffering:預設值
二、對檔案進行操作
將檔案中的內容讀入到一個字串變數/列表中
函式:read(),readline(),readlines(),write(),writelines()
1、read() 讀取整個檔案到字串變數中
2、readline() 讀取檔案中的一行,然後返回整行,包括行結束符到字串變數中
3、readlines() 讀取整個檔案,返回一個字串列表,列表中的每個元素都是一個字串,代表一行

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

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

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

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

w+與r+區別:

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

r+與a+區別:

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

  2. fd.write('123')

  3. fd = open("1.txt",'r+')

  4. fd.write('456')

  5. fd = open("1.txt",'a+')

  6. fd.write('789')

結果:456789

說明r+進行了覆蓋寫。

以a,a+的方式開啟檔案,附加方式開啟

(a:附加寫方式開啟,不可讀;a+:附加讀寫方式開啟)

以 'U' 標誌開啟檔案, 所有的行分割符通過 Python 的輸入方法(例#如 read*() ),返回時都會被替換為換行符\n. ('rU' 模式也支援 'rb' 選項) .

r和U要求檔案必須存在

不可讀的開啟方式:w和a

若不存在會建立新檔案的開啟方式:a,a+,w,w+

  1. >>> fd=open(r'f:\mypython\test.py','w') #只讀方式開啟,讀取報錯

  2. >>> fd.read()

  3. Traceback (most recent call last):

  4. File "<stdin>", line 1, in <module>

  5. IOError: File not open for reading

  6. >>> fd=open(r'f:\mypython\test.py','a')#附加寫方式開啟,讀取報錯

  7. >>> fd.read()

  8. Traceback (most recent call last):

  9. File "<stdin>", line 1, in <module>

  10. IOError: File not open for reading

  11. >>></span></span></span>

2.正確讀寫方式開啟,出現亂碼

  1. >>> fd=open(r'f:\mypython\test.py','a+')

  2. >>> fd.write('123')

  3. >>> fd.read()

  4. >>> fd.close()

close之前,手動開啟檔案,什麼都沒寫入;close後,手動開啟檔案,亂碼:123嚅?

原因分析:指標問題。open()以a+模式開啟了一個附加讀寫模式的檔案,由於是a,所以指標在檔案末尾。此時如果做read(),則Python發現指標位置就是EOF,讀取到空字串。

在寫入123之後,指標的位置是4,仍然是檔案尾,檔案在記憶體中是123[EOF]。

但看起來read()的時候,Python仍然去試圖在磁碟的檔案上,將指標從檔案頭向後跳3,再去讀取到EOF為止。

也就是說,你實際上是跳過了該檔案真正的EOF,為硬碟底層的資料做了一個dump,一直dump到了一個從前存檔檔案的[EOF]為止。所以最後得到了一些根本不期待的隨機亂字元,而不是編碼問題造成的亂碼。

解決方案:讀取之前將指標重置為檔案頭(如果讀取之後重置再讀,無效)

  1. >>> fd=open(r'f:\mypython\test.py','a+')

  2. >>> fd.seek(0)

  3. >>> fd.read()

  4. '123'<span style="white-space:pre"> </span>#順利讀出</span></span>

3.檔案裡有內容,卻讀出空字元

  1. >>> fd=open(r'f:\mypython\test.py','w+') #清空內容,重新寫入

  2. >>> fd.write('456')

  3. >>> fd.flush()<span style="white-space:pre"> </span>#確定寫入,此時檔案內容為“456”

  4. >>> fd.read()

  5. '' #讀出空

原因:同樣是指標問題,寫入後指標指向末尾[EOF],因此讀出空

解決方案一、呼叫close後重新開啟,指標位於開頭。(r,r+,a+,U都可以,注意不要用w,w+,a開啟)

  1. >>> fd.close()

  2. >>> fd=open(r'f:\mypython\test.py','a+')

  3. >>> fd.read()

  4. '456'

  5. >>> fd.close()

  6. >>> fd=open(r'f:\mypython\test.py','r+')

  7. >>> fd.read()

  8. '456'<pre name="code" class="python">>>> fd.close()

  9. >>> fd=open(r'f:\mypython\test.py','r')

  10. >>> fd.read()

  11. '456'

  12. >>> fd.close()

  13. >>> fd=open(r'f:\mypython\test.py','U')

  14. >>> fd.read()

  15. '456'

解決方案二、呼叫seek指向開頭

  1. >>> fd=open(r'f:\mypython\test.py','w+')

  2. >>> fd.write('456')

  3. >>> fd.seek(0)

  4. >>> fd.read()

  5. '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!')