Linux中unlink函數(shù)和刪除文件的操作方法

小編給大家分享一下Linux中unlink函數(shù)和刪除文件的操作方法,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!

成都創(chuàng)新互聯(lián)公司專業(yè)為企業(yè)提供云巖網(wǎng)站建設(shè)、云巖做網(wǎng)站、云巖網(wǎng)站設(shè)計、云巖網(wǎng)站制作等企業(yè)網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計與制作、云巖企業(yè)網(wǎng)站模板建站服務(wù),10年云巖做網(wǎng)站經(jīng)驗,不只是建網(wǎng)站,更提供有價值的思路和整體網(wǎng)絡(luò)服務(wù)。

什么是Linux系統(tǒng)

Linux是一種免費使用和自由傳播的類UNIX操作系統(tǒng),是一個基于POSIX的多用戶、多任務(wù)、支持多線程和多CPU的操作系統(tǒng),使用Linux能運行主要的Unix工具軟件、應(yīng)用程序和網(wǎng)絡(luò)協(xié)議。

1. unlink函數(shù)

??對于硬鏈接來說,unlink 用來刪除目錄項,并把 inode 引用計數(shù)減 1,這兩步也是一個原子過程。直到 inode 引用計數(shù)為 0,才會真正刪除文件。

??對于軟鏈接來說,unlink 直接刪除軟鏈接,而不影響軟鏈接指向的文件。

函數(shù)原型:

int unlink(const char *pathname);  

參數(shù)說明:

??pathname:指定要移除的鏈接文件

返回值說明:

??成功返回0;失敗則返回-1,同時設(shè)置errno為相應(yīng)值

2. 實驗代碼—myunlink

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
int main(int argc, char *argv[]){ 
 //為一個已經(jīng)存在的文件創(chuàng)建目錄項(硬鏈接)
 if(link(argv[1], argv[2]) == -1){
 perror("link error");
 exit(1);
 }
 //刪除之前的文件目錄項
 if(unlink(argv[1]) == -1){
 perror("unlink error");
 exit(1);
 }
 return 0;
}

??當(dāng)我們執(zhí)行./myunlink hellotest命令完后,會刪除 hellotest,同時 inode 引用計數(shù)減 1。

3. 刪除文件

??不用說,相信大家都用過rm -rf命令吧。

??現(xiàn)在我們再來思考一下,以前我們通過rm命令刪除文件時你有沒有質(zhì)疑過,文件真的被刪除掉了嗎?

??如果真的刪除的了話,那么操作系統(tǒng)又是怎么把文件刪除掉的?

??操作系統(tǒng)在設(shè)計的時候是通過把文件的inode索引號與磁盤中的block塊建立了關(guān)聯(lián),這樣我們通過文件找到block塊的位置,也就看到了文件的數(shù)據(jù)了。

??在刪除文件時,是由系統(tǒng)的2個變量來控制的一個是i_link,表示文件的硬鏈接數(shù)量,另一個是i_count,表示文件的引用計數(shù),文件刪除的必需條件就是i_link = 0和i_count = 0。

??在磁盤中的文件只要把i_link = 0(硬鏈接數(shù)干掉)就可以把文件刪除了,如果這個文件在程序中被打開,我們還需要把運行的程序干掉 i_count = 0,這樣才能達到刪除文件的目的。

4. linux下刪除文件的大概過程

??linux下文件刪除過程大概如圖: 

Linux中unlink函數(shù)和刪除文件的操作方法 

圖1-linux下文件刪除的大概過程

??當(dāng)前磁盤中的/test/file目錄下有一個test文件(i_link = 1),還有一個硬鏈接文件hard_link指向test文件(i_link = 1),且./test進程又打開了test文件(i_count = 1),如果要刪除test.txt文件,必須把./test進程干掉(i_count = 0),然后刪除hard_link硬鏈接文件和/test/file目錄下的test.txt文件(也就是讓i_link = 0)。

??也就是說linux下是通過link的數(shù)量來控制文件刪除的,當(dāng)一個文件的link = 0時,這個文件才會被刪除。一般一個文件有2個link計數(shù)器,一個是i_link和i_count。

??i_count是當(dāng)前進程打開文件的引用計數(shù),i_link是文件鏈接的數(shù)量,可以把i_count理解為內(nèi)存中文件的計數(shù)器,而i_link是磁盤中的計數(shù)器。對于rm命令來說實際就是設(shè)置磁盤中文件的i_link計數(shù)為0。如果一個文件被進程所使用,而用戶又執(zhí)行了rm命令把文件刪除掉了,此時程序還能正常執(zhí)行,依舊能從文件中讀取正確的數(shù)據(jù),這是因為rm命令只是把i_link設(shè)置為 0(是將文件到inode的關(guān)聯(lián)斷開,并沒有刪除掉inode與磁盤中的block數(shù)據(jù)塊,此時停止進程,被刪除的數(shù)據(jù)可以找回來,如果進程正在寫入數(shù)據(jù),那么磁盤的block塊的數(shù)據(jù)會被進程寫入的數(shù)據(jù)覆蓋掉,原先的數(shù)據(jù)就恢復(fù)不了了)。

??而進程仍然在引用該文件i_count = 1,執(zhí)行rm命令系統(tǒng)并不會真正的刪除該文件,如果要刪除該文件必須讓進程解除對該文件的引用計數(shù),也就是把進程干掉,這樣文件才會被真正的刪除掉。

??即便如此,文件真的被刪除了嗎?前面我們說過文件的數(shù)據(jù)是存儲在磁盤上block塊中,當(dāng)我們要查找文件當(dāng)中的數(shù)據(jù)時并不是直接找到磁盤上的block塊,因為磁盤上的block塊實在是太多了,你怎么就知道你的數(shù)據(jù)存儲在哪個block塊中?

??假設(shè)你一不小心把非常重要的數(shù)據(jù)刪除掉了,這將意味著你的數(shù)據(jù)就永遠也找不回來了,從而造成無法挽回的損失了,由此可見數(shù)據(jù)的重要性,因此操作系統(tǒng)不會輕易把數(shù)據(jù)從磁盤中真正的刪除掉。

??看到這里,相信你已經(jīng)明白了,實際上你所謂的右鍵刪除操作只是把文件的inode索引號與磁盤中的block的關(guān)聯(lián)斷開了而已,但文件的數(shù)據(jù)并沒有真正的被刪除掉。如果你想真的刪除數(shù)據(jù)的話,要么把磁盤格式化,要么把原先的數(shù)據(jù)刪除掉,然后寫入新的數(shù)據(jù)覆蓋掉,當(dāng)然,你也可以選擇格式化和數(shù)據(jù)覆蓋雙重保險,這個時候你的數(shù)據(jù)想要恢復(fù)基本上是非常困難的,即便可以頂多只能恢復(fù)一部分?jǐn)?shù)據(jù)了吧。

??如果你真的一不小心刪除了很重要的數(shù)據(jù)的話,這個時候趕緊恢復(fù)數(shù)據(jù),其他的任何多余的操作盡量不要做,這樣在數(shù)據(jù)恢復(fù)過程中才能盡量減少數(shù)據(jù)丟失。

5. myunlink2.c程序

#include <unistd.h>
#include <fcntl.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
 /*
 unlink函數(shù)是刪除一個dentry
 */
int main(void){
 int fd;
 char *p = "test of unlink\n";
 char *p2 = "after write something.\n";
 //當(dāng)進程打開temp.txt文件時,引用計數(shù)會+1
 fd = open("temp.txt", O_RDWR|O_CREAT|O_TRUNC, 0644);
 if(fd < 0){
 perror("open temp error");
 exit(1);
 }
 //具備了被釋放的條件
 int ret = unlink("temp.txt"); 
 if(ret < 0){
 perror("unlink error");
 exit(1);
 }
 //向temp.txt文件寫入第一個字符串,通過返回值判斷寫操作是否成功
 ret = write(fd, p, strlen(p));
 if (ret == -1) {
 perror("-----write error");
 }
 printf("hi! I'm printf\n");
 //向temp.txt文件寫入第二個字符串,通過返回值判斷寫操作是否成功
 ret = write(fd, p2, strlen(p2));
 if (ret == -1) {
 perror("-----write error");
 }
 printf("Enter anykey continue\n");
getchar();
 //當(dāng)close關(guān)閉fd的時候,進程對文件的引用計數(shù)就會-1,斷開進程與文件的關(guān)聯(lián)關(guān)系
 close(fd);
 return 0;
}

程序運行結(jié)果:

Linux中unlink函數(shù)和刪除文件的操作方法

??程序的運行結(jié)果和我們所預(yù)料的一樣,當(dāng)程序運行的時候,調(diào)用open函數(shù)創(chuàng)建并打開了temp.txt文件,此時進程對temp文件的引用計數(shù)i_count會加1,同時temp文件本身也會有一個i_link鏈接計數(shù)也會加1。

??當(dāng)調(diào)用了unlink函數(shù)刪除temp文件時,只是把i_link鏈接計數(shù)減1,而進程的i_count計數(shù)還是1,并沒有斷開與temp文件的關(guān)聯(lián)關(guān)系,因此進程可以調(diào)用write函數(shù)往temp文件里面寫數(shù)據(jù),自然也就能成功了。當(dāng)程序運行結(jié)束后,調(diào)用close關(guān)閉對temp文件的引用,此時temp文件就會被操作系統(tǒng)刪除掉。

6. 總結(jié)

??在不了解文件系統(tǒng)原理的情況下,通常我們會認為數(shù)據(jù)已經(jīng)刪除掉,其實不然,磁盤上的文件數(shù)據(jù)還在,只是把dentry目錄和磁盤上的數(shù)據(jù)的聯(lián)系斷開,我們找不到數(shù)據(jù)肯定會認為刪掉了,但是只要我們想辦法讓數(shù)據(jù)和dentry目錄之間重新建立連接,就可以讓刪除的數(shù)據(jù)恢復(fù)。

??因此我們刪除文件,從某種意義上說,只是讓文件具備了被釋放的條件,至于什么時候釋放這取決于操作系統(tǒng)。

??對于unlink函數(shù)來說,清除文件時,如果文件的硬鏈接數(shù)到0了,沒有dentry對應(yīng),但該文件仍不會馬上被釋放。要等到所有打開該文件的進程關(guān)閉該文件,系統(tǒng)才會挑時間將該文件釋放掉。

7. 不要隨便使用rm命令

??相信看到這里,你應(yīng)該知道了,數(shù)據(jù)對于計算機的重要性了,因為一旦某些至關(guān)重要的數(shù)據(jù)刪除了,那就真的永遠沒了,這也是操作系統(tǒng)為什么不直接將數(shù)據(jù)從磁盤中刪除的原因。但也不要因為這樣,你就可以肆無忌憚的使用rm命令了,因為有時候數(shù)據(jù)刪除了,并不能百分百的恢復(fù)回來。

看完了這篇文章,相信你對“Linux中unlink函數(shù)和刪除文件的操作方法”有了一定的了解,如果想了解更多相關(guān)知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!

文章題目:Linux中unlink函數(shù)和刪除文件的操作方法
網(wǎng)址分享:http://muchs.cn/article24/pppgce.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供、App設(shè)計、品牌網(wǎng)站建設(shè)用戶體驗、云服務(wù)器關(guān)鍵詞優(yōu)化

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

h5響應(yīng)式網(wǎng)站建設(shè)