在我們?nèi)粘?xiě)程序的過(guò)程中,免不了與文件系統(tǒng)打交道,各種開(kāi)發(fā)語(yǔ)言都提供了與文件系統(tǒng)交互的指令函數(shù),比如Python中的os模塊,就提供了許多這樣的指令函數(shù),今天就以創(chuàng)建文件夾為例,來(lái)說(shuō)一下,如果用的不當(dāng),會(huì)帶來(lái)的線程安全問(wèn)題。
在Python的os模塊中,創(chuàng)建文件夾有兩種方式,一個(gè)是os.mkdir(),另外一個(gè)是os.makedirs(),這兩個(gè)的區(qū)別如下:
os.mkdir() 函數(shù):os.mkdir(path, mode=0o777)
使用模式mode創(chuàng)建名為path的目錄。如果目錄已經(jīng)存在,則拋出異常FileExistsError。
在一些系統(tǒng)中,可以忽略mode,在一些平臺(tái)可以通過(guò)調(diào)用chmod()來(lái)指定它們。mode就是所創(chuàng)建目錄對(duì)應(yīng)的文件及文件夾的讀寫(xiě)權(quán)限,可以參考linux命令下的chmod命令來(lái)理解。
os.makedirs()函數(shù):os.makedirs(path, mode=0o777, exist_ok=False)
遞歸目錄創(chuàng)建函數(shù),和mkdir()很像,但是所有中間級(jí)目錄都要包含葉目錄。
遞歸創(chuàng)建目錄,題中應(yīng)有之意即路徑中哪一層不存在,則自動(dòng)創(chuàng)建。
區(qū)別于os.mkdir(path, mode=0o777),只創(chuàng)建最后一層的目錄。
path:所要?jiǎng)?chuàng)建的目錄
參數(shù)mode傳遞給mkdir();
如果exist_ok是False(默認(rèn)),當(dāng)目標(biāo)目錄(即要?jiǎng)?chuàng)建的目錄)已經(jīng)存在,會(huì)拋出一個(gè)OSError。
具體使用實(shí)例如下:
在上圖/data/zzz文件夾下沒(méi)有任何文件夾,我們現(xiàn)在就來(lái)創(chuàng)建一個(gè)名為a的文件夾:
這時(shí)a文件夾已經(jīng)存在,我們?cè)賵?zhí)行一下上次的指令函數(shù):
這時(shí)就會(huì)報(bào)錯(cuò),如果程序中沒(méi)有處理異常,或者異常之后沒(méi)有做補(bǔ)償措施,那么會(huì)對(duì)業(yè)務(wù)系統(tǒng)中的某個(gè)方面會(huì)有一些影響,雖然影響不是很大,但是我們完全可以通過(guò)編程語(yǔ)言提供的其他函數(shù),進(jìn)行系統(tǒng)完善,下面就用os.makdirs(),來(lái)避免上面的錯(cuò)誤:
從上圖中可以看出,如果os.makedirs()使用參數(shù)exist_ok的默認(rèn)值,也就是exist_ok=False,還是會(huì)報(bào)[Errno 17] File exists的錯(cuò)誤,這時(shí)就要用exist_ok=True,這樣如果文件夾存在,會(huì)忽略創(chuàng)建文件夾。也有人會(huì)用os.mkdir()這樣寫(xiě)程序,就是加一個(gè)判斷,如果文件夾存在,才去用os.mkdir()去創(chuàng)建文件夾,如果這個(gè)方法沒(méi)有加鎖,那么就會(huì)帶來(lái)線程安全問(wèn)題,就是兩個(gè)或者多個(gè)線程同時(shí)進(jìn)入創(chuàng)建文件夾的只指令處,也會(huì)發(fā)生拋出異常的現(xiàn)象,如果加鎖了,也會(huì)對(duì)該方法性能方面有些影響,況且鎖使用不好,容易把自己都給鎖死,所以還是使用編程語(yǔ)言提供的函數(shù)去正確的編程。這兩個(gè)還有一個(gè)區(qū)別就是是否支持遞歸創(chuàng)建文件夾:
可見(jiàn)os.mkdir()不支持遞歸創(chuàng)建文件夾,只有父層文件夾存在時(shí),才能創(chuàng)建子層文件夾,而os.makedirs()是支持遞歸創(chuàng)建文件夾的。