面試題:rm 刪除文件以後,空間就被釋放了嗎?你知道答案嗎?

  • 產生一個指定大小的隨機內容文件node

  • 一個文件什麼狀況下才會被刪除?程序員

  • 如何釋放已經被刪除文件佔用的空間?面試

  • 總結設計模式

在Linux,你是否是曾經天真的覺得,使用rm刪除一個文件,佔用的空間就釋放了?事情可能不是經常如人意。網絡

產生一個指定大小的隨機內容文件

咱們先看一下當前各個掛載目錄的空間大小:多線程

$ df -h
/dev/sda11      454M  280M  147M  66% /boot
 

我這裏挑選了其中一個結果展現(你能夠選擇任一掛載目錄),接下來準備在/boot下生成一個文件。dom

首先咱們產生一個50M大小的文件:測試

$ dd if=/dev/urandom of=/boot/test.txt bs=50M count=1
 

至此,咱們產生了一個50M大小的文件,再看boot下:atom

$ df -h
/dev/sda11      454M  312M  115M  74% /boot
 

這裏你不用關心到底多了多少,你只須要關注,/boot下的文件增多了。spa

測試程序:

#include<stdio.h>
#include<unistd.h>
int main(void)
{
    FILE *fp = NULL;
    fp = fopen("/boot/test.txt", "rw+");
    if(NULL == fp)
    {
       perror("open file failed");
       return -1;
    }
    while(1)
    {
       //do nothing
       sleep(1);
    }
    fclose(fp);
    return 0;
}
 

至於程序自己,也沒幹啥實際的事情,就是打開一個文件,而後一直循環。編譯並運行:

$ gcc -o openFile openFile.c
$ ./openFile
 

打開另一個窗口,刪掉test.txt:

$ rm /boot/test.txt
 

再看一下boot空間:

$ df -h
dev/sda11      454M  312M  115M  74% /boot
 

咦?空間大小怎麼一點都沒變!!明明使用rm把它刪除了啊?

咱們把openFile程序停掉,再看看:

$$ df -h
/dev/sda11      454M  280M  147M  66% /boot
 

乖乖,空間立刻就釋放掉了,也就是按照預期,咱們的文件被刪除了。

一個文件什麼狀況下才會被刪除?

實際上,只有當一個文件的引用計數爲0(包括硬連接數)的時候,纔可能調用unlink刪除,只要它不是0,那麼就不會被刪除。所謂的刪除,也不過是文件名到 inode 的連接刪除,只要不被從新寫入新的數據,磁盤上的block數據塊不會被刪除,所以,你會看到,即使刪庫跑路了,某些數據仍是能夠恢復的。換句話說,當一個程序打開一個文件的時候(獲取到文件描述符),它的引用計數會被+1,rm雖然看似刪除了文件,實際上只是會將引用計數減1,但因爲引用計數不爲0,所以文件不會被刪除。

struct inode {
struct hlist_node   i_hash; /* hash鏈表的指針 */
struct list_head    i_list; /* backing dev IO list */
struct list_head    i_sb_list; /* 超級塊的inode鏈表 */
struct list_head    i_dentry; /* 引用inode的目錄項對象鏈表頭 */
unsigned long    i_ino; /* 索引節點號 */
atomic_t         i_count; /* 引用計數 */
unsigned int     i_nlink; /* 硬連接數目 */
 

關於裏面的細節,還有不少內容(如硬連接數量也會影響文件是否被刪除),這裏不一一展開。

如何釋放已經被刪除文件佔用的空間?

關於釋放,前面已經說了,重啓打開該文件的進程便可。可是有沒有方法找到哪些文件被刪除了,但仍是被某些進程打開了呢?

天然是有方法的:

$ lsof |grep deleted

 

其中被標記爲deleted的文件,就是這樣的一些文件。

其實在前面的例子中,咱們也能夠很容易觀察到(openFile程序運行,test.txt文件被刪除):

$ ls -al /proc/`pidof openFile`/fd
total 0
lrwx------ 1 root root 64 5月   4 09:27 0 -> /dev/pts/25
lrwx------ 1 root root 64 5月   4 09:27 1 -> /dev/pts/25
lrwx------ 1 root root 64 5月   4 09:27 2 -> /dev/pts/25
lrwx------ 1 root root 64 5月   4 09:27 3 -> /boot/test.txt (deleted)

 

看見沒有,test.txt後面還有deleted字樣。

既然咱們都說了,這樣的狀況下文件是沒有被刪除的,那麼還能不能恢復呢?實際上仍是能夠讀取的。

總結

實際上對於這種文件被刪除了,經常出現於程序的日誌文件中,可能你有一個定時任務去清理程序產生的日誌文件,可是若是程序自己忘記關閉句柄,就會致使磁盤空間得不到釋放,最終就是你認爲文件都被刪除了,可是磁盤卻依然被佔着。因此,養成好習慣,打開文件後,不用時,記得關閉文件描述符。

若是發現明明已經刪除了大量文件,可是空間卻並無恢復正常,那麼不妨看看是否是還有程序打開了這些文件。

小編總結了2020面試題,這份面試題的包含的模塊分爲19個模塊,分別是: Java 基礎、容器、多線程、反射、對象拷貝、Java Web 、異常、網絡、設計模式、Spring/Spring MVC、Spring Boot/Spring Cloud、Hibernate、MyBatis、RabbitMQ、Kafka、Zookeeper、MySQL、Redis、JVM 。 關注公衆號:程序員白楠楠

相關文章
相關標籤/搜索