面試官跟我說rm刪除文件之後,空間就被釋放了嗎?我該怎麼答

来源:https://www.cnblogs.com/MonsterJ/archive/2020/06/05/13050944.html
-Advertisement-
Play Games

在Linux,你是不是曾經天真的以為,使用rm刪除一個文件,占用的空間就釋放了?事情可能不是常常如人意。 產生一個指定大小的隨機內容文件 我們先看一下當前各個掛載目錄的空間大小: $ df -h /dev/sda11 454M 280M 147M 66% /boot 我這裡挑選了其中一個結果展示(你 ...


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

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

我們先看一下當前各個掛載目錄的空間大小:

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

我這裡挑選了其中一個結果展示(你可以選擇任一掛載目錄),接下來準備在/boot下生成一個文件。

首先我們產生一個50M大小的文件:

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

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

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

這裡你不用關心到底多了多少,你只需要關註,/boot下的文件增多了。

測試程式:

#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字樣。

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

面試官跟我說rm刪除文件之後,空間就被釋放了嗎?我該怎麼答
在這裡插入圖片描述

總結

實際上對於這種文件被刪除了,常常出現於程式的日誌文件中,可能你有一個定時任務去清理程式產生的日誌文件,但是如果程式本身忘記關閉句柄,就會導致磁碟空間得不到釋放,最終就是你認為文件都被刪除了,但是磁碟卻依然被占著。所以,養成好習慣,打開文件後,不用時,記得關閉文件描述符。

如果發現明明已經刪除了大量文件,但是空間卻並沒有恢復正常,那麼不妨看看是不是還有程式打開了這些文件。

作者:編程珠璣


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • from docx import Document word=Document(',,,,,,.docx') paragraphs=word.paragraphs #返回一個列表,有多少段就有多少個值 for i in paragraphs: print(i.text) #列印每段內容 a=para ...
  • from docx import Document word =Document(r'表格名稱.docx') #讀取表格 tables=word.tables for i in tables[:]: for j,row in enumerate(i.rows[:]): row_content=[] ...
  • Lambda表達式 函數式編程: 函數: 函數就是有輸入量、輸出量的一套計算方案,也就是“拿數據做操作”面向對象思想強調“必須通過對象的形式來做事情” 函數式思想: 儘量忽略面向對象的複雜語法:“強調做什麼,而不是以什麼形式去做”,Lambda表達式就是函數思想的體現 例1: 需求:啟動一個線程,在 ...
  • 在知乎上發現一個有趣的話題“一個程式員的水平能差到什麼程度?”其中有一個高贊回答,引發了程式員的共鳴。 ...
  • 1.基本思路 我現階段的分頁查詢的實現是基於sql語句的。 select * from user where id limit a, b 構造出相應的a和b就可以查詢出想要的數據,在顯示在頁面上。重點是要構造出當前的頁數,就要封裝一個javaBean,存儲有關分頁的基本屬性。 這樣只需在servic ...
  • 方式一: this.getClass().getClassLoader().getResourceAsStream(String name) 首先調用對象的getClass()方法是獲得對象當前的類類型,這部分數據存在方法區,而後在類類型上調用getClassLoader()方法是得到當前類型的類加 ...
  • 商城購物 products = [["iPhone Pro",12888],["MacBook Pro",22999],["HuaWei P40 Pro",6999],["Mi 10 Pro",5999],["Mi Game Book",8999]] shop_car = [] # 用戶購物車 sh ...
  • 手勢拖動 UIPanGestureRecognizer * panGestureRecognizer = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(doMoveAction:)]; 核心部分 拖到到前面還 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...