理解Linux文件系统挂载参数noatime nodiratime
很多线上服务器为了提供文件系统IO性能,会在挂载文件系统的时候指定“noatime,nodiratime”参数,意味着当访问一个文件和目录的时候,access time都不会更新。但是如果未指定上面的参数,atime则会更新。那么具体差异在哪里?
未指定 noatime,nodiratime
$ touch test ; stat test ;
...
Access: -- ::23.507135507 +
Modify: -- ::23.507135507 +
Change: -- ::23.507135507 + $ echo hello >> test ; stat test;
...
Access: -- ::23.507135507 +
Modify: -- ::38.018430637 +
Change: -- ::38.018430637 + $ cat test ;stat test
...
Access: -- ::02.916135510 +
Modify: -- ::38.018430637 +
Change: -- ::38.018430637 +
可以看出未指定"noatime,nodiratime"的情况下
- read文件的时候会导致atime更新,不会导致mtime和ctime更新
- write文件只会导致mtime和ctime更新,不会导致atime更新。
指定 noatime,nodiratime
$touch test ; stat test ;
...
Access: -- ::28.680135484 +
Modify: -- ::28.680135484 +
Change: -- ::28.680135484 + $ sleep ; echo hello >> test ; stat test;
...
Access: -- ::28.680135484 +
Modify: -- ::38.682727983 +
Change: -- ::38.682727983 + $ cat test ;stat test
...
Access: -- ::28.680135484 +
Modify: -- ::38.682727983 +
Change: -- ::38.682727983 +
可以看出指定"noatime,nodiratime"的情况下
- read文件的时候不会导致atime、mtime、ctime改变
- write文件只会导致mtime和ctime更新,不会导致atime更新。
实际应用场景
在平日里经常有删除文件的需求,大概如下
删除过去N天内都未访问过的文件或者目录(删除N天前访问过的文件) $ #注意这条命令很危险!
$ find /home/fire/ -atime +N -exec rm -rf {} \;
假设 /home/fire 目录是一周之前创建的,那么对于这条命令有两个执行结果
$ #注意这条命令很危险!
$ find /home/fire/ -atime + -exec rm -rf {} \;
- 指定"noatime":find的时候发现 /home/fire 是7天之前创建的,立马就会删除整个目录。而且还会报错"find: /home/fire: No such file or directory",原因就是第一个rm -rf /home/fire 之后 find失败了。这种是很危险的!原因是会误删除文件。
- 未指定"noatime":那就得看情况,如果/home/fire过去7天没有被访问过,那么就和情况一一样,直接删除。如果过去7天内,该目录有人访问过,atime肯定是7天之内,那么就会遍历下面的目录,依次按照之前逻辑。但是遍历过程会更改目录的atime。
看了上面的例子会发现find去删除目录的时候变得好复杂,而且一定要小心。所以find删除更适用于删除文件,不要删除目录。
删除N天内未被访问过的文件
$ find /home/fire/ -atime +N -type f -exec rm -f {} \;
注:
上面是转载的原文,基本没有问题,但关于禁止文件和目录的访问时间,只需指定noatime即可,因为在linux内核中,当指定了noatime,关于nodiratime的判断也就自动跳过了。
下面是最新稳定版linux 4.13.3内核 中关于这段的代码说明:
bool __atime_needs_update(const struct path *path, struct inode *inode,
bool rcu)
{
struct vfsmount *mnt = path->mnt;
struct timespec now; if (inode->i_flags & S_NOATIME)
return false; /* Atime updates will likely cause i_uid and i_gid to be written
* back improprely if their true value is unknown to the vfs.
*/
if (HAS_UNMAPPED_ID(inode))
return false; if (IS_NOATIME(inode))
return false;
if ((inode->i_sb->s_flags & MS_NODIRATIME) && S_ISDIR(inode->i_mode))
return false; if (mnt->mnt_flags & MNT_NOATIME)
return false;
if ((mnt->mnt_flags & MNT_NODIRATIME) && S_ISDIR(inode->i_mode))
return false; now = current_time(inode); if (!relatime_need_update(path, inode, now, rcu))
return false; if (timespec_equal(&inode->i_atime, &now))
return false; return true;
}
生产环境中的Linux内核多没有这么新,一般是2.6.32左右的,下面是2.6.32的linux内核关于这一点的说明:
void touch_atime(struct vfsmount *mnt, struct dentry *dentry)
{
struct inode *inode = dentry->d_inode;
struct timespec now; if (inode->i_flags & S_NOATIME)
return;
if (IS_NOATIME(inode))
return;
if ((inode->i_sb->s_flags & MS_NODIRATIME) && S_ISDIR(inode->i_mode))
return; if (mnt->mnt_flags & MNT_NOATIME)
return;
if ((mnt->mnt_flags & MNT_NODIRATIME) && S_ISDIR(inode->i_mode))
return; now = current_fs_time(inode->i_sb); if (!relatime_need_update(mnt, inode, now))
return; if (timespec_equal(&inode->i_atime, &now))
return; if (mnt_want_write(mnt))
return; inode->i_atime = now;
mark_inode_dirty_sync(inode);
mnt_drop_write(mnt);
}
EXPORT_SYMBOL(touch_atime);
通过查看mount(2)可以看到几个参数说明:
MS_NOATIME
Do not update access times for (all types of) files on this file system MS_NODIRATIME
Do not update access times for directories on this file system. This flag provides a subset of the functionality
provided by MS_NOATIME; that is, MS_NOATIME implies MS_NODIRATIME. MS_RELATIME (Since Linux 2.6.)
When a file on this file system is accessed, only update the file’s last access time (atime) if the current value
of atime is less than or equal to the file’s last modification time (mtime) or last status change time (ctime).
This option is useful for programs, such as mutt(), that need to know when a file has been read since it was last modified.
Since Linux 2.6., the kernel defaults to the behavior provided by this flag (unless MS_NOATIME was specified),
and the MS_STRICTATIME flag is required to obtain traditional semantics.In addition, since Linux 2.6., the file’s last
access time is always updated if it is more than day old. MS_STRICTATIME (Since Linux 2.6.)
Always update the last access time (atime) when files on this file system are accessed.
(This was the default behavior before Linux 2.6..) Specifying this flag overrides the effect
of setting the MS_NOATIME and MS_RELATIME flags.
通过判定MS_NODIRATIME,如果为真,就不再更新目录的访问时间。
上面参数中提到了MS_RELATIME,对于某些用户某些应用,atime 信息可能是必要的,比如邮件程序和备份工具。如果完全禁止 atime 更新会使这些程序无法正常工作。针对这种情况,Linux 在内核 2.6.20 中添加了一个新的 mount 选项 relatime(relative atime 的意思)。relatime 的意思是访问文件时,仅在 atime 早于文件的更改时间时对 atime 进行更新。在内核 2.6.24 中,又对 relatime 进行了扩展,在访问文件时,当 atime 已经超过某个时间(例如一天)没有更新,就对 atime 进行更新。这个扩展的意思就是调整 atime 的更新粒度。
下面是mount(8)中的参数说明:
relatime
Update inode access times relative to modify or change time. Access time is only updated if the previous access
time was earlier than the current modify or change time. (Similar to noatime, but doesn’t break mutt or
other applications that need to know if a file has been read since the last time it was modified.) Since Linux 2.6., the kernel defaults to the behavior provided by this option (unless noatime was specified),
and the strictatime option is required to obtain traditional semantics. In addition, since Linux 2.6., the
file’s last access time is always updated if it is more than day old. strictatime
Allows to explicitly requesting full atime updates. This makes it possible for kernel to defaults to
relatime or noatime but still allow userspace to override it. For more details about the default system mount
options see /proc/mounts.
使用 relatime(relative atime)挂载,可在前一个 atime 更新比 mtime 或者 ctime 更新旧时更新 atime。
在这个示例中,GFS2 文件系统位于 /dev/vg01/lvol0,并挂载到目录 /mygfs2。只有在前一个 atime 更新比 mtime 或者 ctime 更新旧时才会进行 atime 更新。
mount /dev/vg01/lvol0 /mygfs2 -o relatime
参考资料:
http://www.firefoxbug.com/index.php/archives/2801/
http://www.cnblogs.com/allegro/archive/2011/04/18/2019598.html
https://mp.weixin.qq.com/s?__biz=MzA3MzYwNjQ3NA==&mid=400742435&idx=1&sn=d5fa0776ebce30be1de897cd7d1eb295&scene=1&srcid=1123K3Ka8EtwjcNXhNfUGJV7#rd
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux-stable.git/tree/fs/inode.c?h=v4.13.3
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux-stable.git/tree/fs/inode.c?h=linux-2.6.32.y
理解Linux文件系统挂载参数noatime nodiratime的更多相关文章
- Linux 文件系统挂载
文件系统挂载简介 磁盘分区和格式化完成后,磁盘分区要想能够使用,就需要挂载,在挂载某个分区前需要先建立一个挂载点 挂载:将新的文件系统关联至当前根文件系统 卸载:将某文件系统与当前根文件系统的关 ...
- Linux入门基础 #5:Linux文件系统挂载管理
本文出自 http://blog.csdn.net/shuangde800 ------------------------------------------------------------ ...
- Linux文件系统挂载管理
http://itercast.com/lecture/19 文件系统创建好之后需要挂载到系统中方可使用,windows.Mac系统会自动挂载文件系统,而Linux下一般需要手工挂载或配置系统进行自动 ...
- 理解Linux文件系统之inode
很少转发别人的文章,但是这篇写的太好了. 理解inode 作者: 阮一峰 inode是一个重要概念,是理解Unix/Linux文件系统和硬盘储存的基础. 我觉得,理解inode,不仅有助于提高系统 ...
- [转]理解Linux文件系统之inode
很少转发别人的文章,但是这篇写的太好了. 理解inode 作者: 阮一峰 inode是一个重要概念,是理解Unix/Linux文件系统和硬盘储存的基础. 我觉得,理解inode,不仅有助于提高系统 ...
- Linux 入门记录:九、Linux 文件系统挂载管理
一.挂载操作 磁盘或分区创建好文件系统后,需要挂载到一个目录才能够使用. Windows 或 Mac 系统会进行自动挂载,一旦创建好文件系统后会自动挂载到系统上,Windows 上称之为 C 盘.D ...
- linux文件系统挂载
接上一篇博客 创建好了文件系统的磁盘仍然是不能用的,必须挂载到某个目录下才能使用 [root@gechong mnt]# ll /dev/sdb* brw-rw----. root disk , 11 ...
- 理解Linux文件系统之 inode
一.inode是什么? 理解inode,要从文件储存说起. 文件储存在硬盘上,硬盘的最小存储单位叫做”扇区”(Sector).每个扇区储存512字节(相当于0.5KB). 操作系统读取硬盘的时候,不会 ...
- 深入理解Linux文件系统与日志分析
一.inode和bolck概述 二.链接文件 三.inode节点耗尽故障处理 四.EXT类型文件恢复 五.xfs文件备份和恢复 六.日志文件 一.inode和bolck概述 1.定义 文件数据 文件数 ...
随机推荐
- TW实习日记:第31-32天
不知不觉的,实习的净工作天数,已经都超过一个月了.因为对工作内容不是很满意,所以打算月底离职,也不知道是公司太缺人还是我真的能干活,领导竟然三番两次找我让我再考虑...明天又要找我了,哎...随机应变 ...
- 统计单词数:string函数使用
题目描述 一般的文本编辑器都有查找单词的功能,该功能可以快速定位特定单词在文章中的位置,有的还能统计出特定单词在文章中出现的次数. 现在,请你编程实现这一功能,具体要求是:给定一个单词,请你输出它在给 ...
- LeetCode 169. Majority Element - majority vote algorithm (Java)
1. 题目描述Description Link: https://leetcode.com/problems/majority-element/description/ Given an array ...
- js如何处理字符串中带有↵字符
js或vue中如何处理字符串中带有↵字符 split('\n') 使用split('\n')将字符串分割成数组就行 如果我们在vue中,只需要在页面中绑定变量时操作split('\n')就可以了: & ...
- wwnjld团队第二轮迭代成员分数
2014-01-05 第二轮迭代团队内成员分数如下(依据分数分配规则以及团队会议协商所得结果): 吴渊渊 23 汪仁贵 21.5 高小洲 19.5 聂建 22.5 吕家辉 23.5 程志 10
- EasyJSWebView原理分析
概述 在iOS6之前,native只能调用webiew里的js代码,官方没有提供js调用native方法的接口.到了iOS7,官方提供了JSContext用来与js交互,native和js可以双向调用 ...
- C语言特殊符号
-> ->在C语言中称为间接引用运算符,是二目运算符,优先级同成员运算符“.”.用法:p->a,其中p是指向一个结构体的指针,a是这个结构体类型的一个成员.表达式p->a引用了 ...
- C#通过SC命令和静态公共类来操作Windows服务
调用的Windows服务应用程序网址:http://www.cnblogs.com/pingming/p/5115304.html 一.引用 二.公共静态类:可以单独放到类库里 using Syste ...
- 第三部分shell编程3(shell脚本2)
7. if 判断一些特殊用法 if [ -z $a ] 这个表示当变量a的值为空时会怎么样if grep -q '123' 1.txt; then 表示如果1.txt中含有'123'的行时会怎么样if ...
- Oracle查询字段中有空格的数据
一.问题说明 最近在给某个用户下的表批量添加注释时,在程序中将注释名用trim()过滤一遍就可以了,但是在程序执行成功后怎么检测添加的注释名是否有空格存在呢? 二.解决方法 1.SELECT * FR ...