面试官给我挖坑:rm删除文件以后,空间就被释放了吗?

在Linux,你是否是曾经天真的觉得,使用rm删除一个文件,占用的空间就释放了?事情可能不是经常如人意。node

产生一个指定大小的随机内容文件

咱们先看一下当前各个挂载目录的空间大小:bash

$ 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下:ui

$ df -h/dev/sda11      454M  312M  115M  74% /boot复制代码

这里你不用关心到底多了多少,你只须要关注,/boot下的文件增多了。atom

测试程序: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空间:code

$ df -hdev/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字样。

既然咱们都说了,这样的状况下文件是没有被删除的,那么还能不能恢复呢?实际上仍是能够读取的。

总结

实际上对于这种文件被删除了,经常出现于程序的日志文件中,可能你有一个定时任务去清理程序产生的日志文件,可是若是程序自己忘记关闭句柄,就会致使磁盘空间得不到释放,最终就是你认为文件都被删除了,可是磁盘却依然被占着。因此,养成好习惯,打开文件后,不用时,记得关闭文件描述符。

若是发现明明已经删除了大量文件,可是空间却并无恢复正常,那么不妨看看是否是还有程序打开了这些文件。

相关文章
相关标签/搜索