使用shell巧妙高效的批量删除历史文件或目录
2013-09-25 17:55
316 查看
背景:有实时产生的数据按小时分文件保存,如“/data/2013/09/18/14.txt”。现需要保留30天的最新数据,而删除所有其它的历史数据。注意“保留30天的最新数据”,可能不是连续的30天,中间若某一天数据缺失,也必须把最新的30天的数据保留下来。
思路:获取所有数据路径列表,去除最新30天的数据路径,然后遍历删除。若是使用脚本语言来开发(如php,python),循环获取路径列表的代码就已经比较臃肿了,效率也不高,特别是文件目录特多的时候。使用shell应该更方便和高效
词命令还存在一个隐患,rm的参数字符长度可能超过1024的限制,修改如下
这个命令里使用了find,效率还是有些不满意,使劲的寻找更高效获取路径列表的方式。哈哈,黄天不负苦心人啊,终于找到一个
使用time命令测试运行速度,第二条命令和第三条命令其运行时间分别是0.007s和0.002s,后者快了3倍
但郁闷的是,我不知道“echo /data/*/*/*/“到底是怎么个原理,有什么限制等等,网上也还没有搜索到相关资料,有知道的朋友请告知一下啊
2013-10-17 编辑补充:
原命令还可以修改精简一下,通过指定awk使用的换行符来避免for循环
还有一个问题就是,你可能想在日志中记录下删除了那些文件。可以先把文件列表赋值给一个变量,记录日志后再删除
${files//\ /"\n"} 是把变量files里的所有空格都替换成\n换行符,这样xargs才能正确切分参数。不知道为什么会这样
2015-4-16 继续优化命令
说明:
ls : -1参数表示一个文件或目录单独占一行显示, -r 表示按路径降序排列(默认是升序排列)
xargs: -d 参数设置分隔符
思路:获取所有数据路径列表,去除最新30天的数据路径,然后遍历删除。若是使用脚本语言来开发(如php,python),循环获取路径列表的代码就已经比较臃肿了,效率也不高,特别是文件目录特多的时候。使用shell应该更方便和高效
rm -rf `find /data/*/*/*/ -type d|awk '{a[NR]=$0}END{n=asort(a,sa)-30;for(i=1;i<=n;i++){print sa[i]}}'`
词命令还存在一个隐患,rm的参数字符长度可能超过1024的限制,修改如下
find /data/*/*/*/ -type d|awk '{a[NR]=$0}END{n=asort(a,sa)-30;for(i=1;i<=n;i++){print sa[i]}}'|xargs -I{} rm -rf {}
这个命令里使用了find,效率还是有些不满意,使劲的寻找更高效获取路径列表的方式。哈哈,黄天不负苦心人啊,终于找到一个
echo /data/*/*/*/|awk '{for(i=1;i<=NF;i++){a[i ]=$i}}END{n=asort(a,sa)-30;for(i=1;i<=n;i++){print sa[i ]}}'|xargs -I{} rm -rf {}
使用time命令测试运行速度,第二条命令和第三条命令其运行时间分别是0.007s和0.002s,后者快了3倍
但郁闷的是,我不知道“echo /data/*/*/*/“到底是怎么个原理,有什么限制等等,网上也还没有搜索到相关资料,有知道的朋友请告知一下啊
2013-10-17 编辑补充:
原命令还可以修改精简一下,通过指定awk使用的换行符来避免for循环
echo /data/*/*/*/|awk 'BEGIN{ORS=RS=" "}{a[NR]=$0}END{n=asort(a,sa)-30;for(i=1;i<=n;i++){print sa[i ]}}'|xargs -I{} rm -rf {}
还有一个问题就是,你可能想在日志中记录下删除了那些文件。可以先把文件列表赋值给一个变量,记录日志后再删除
files=`echo /data/*/*/*/|awk 'BEGIN{ORS=RS=" "}{a[NR]=$0}END{n=asort(a,sa)-30;for(i=1;i<=n;i++){print sa[i ]}}'` echo $files; #输出日志 echo -e ${files//\ /"\n"}|xargs -I{} rm -rf {}关键是最后一行, echo 的-e参数是为了可以输出\n换行,如无此参数则不会识别。
${files//\ /"\n"} 是把变量files里的所有空格都替换成\n换行符,这样xargs才能正确切分参数。不知道为什么会这样
2015-4-16 继续优化命令
files=`ls -1 -r /data/*/*/*/ | awk 'NR>=30{print $0}'` echo $files; #输出日志 echo -e $files | xargs -I{} -d " " rm -rf {}
说明:
ls : -1参数表示一个文件或目录单独占一行显示, -r 表示按路径降序排列(默认是升序排列)
xargs: -d 参数设置分隔符
相关文章推荐
- 使用CMD模式批量删除指定目录(支持目录中多层文件夹嵌套)下指定格式文件
- 使用shell删除目录下几天前生成文件方法
- Linux监控系统磁盘使用比例,当使用率达到指定比例,删除指定目录下的所有文件的shell脚本
- Linux下使用Shell脚本删除一个目录下的所有子目录和文件
- 使用shell脚本删除当前目录下.c文件对应的可执行文件
- Linux下使用Shell根据txt文件批量创建目录
- linux下使用shell编写的删除历史文件的脚本 find | xargs tar 有文件个数限制哦
- 使用shell脚本删除当前目录下.c文件对应的可执行文件
- 使用递归删除目录下的空文档和文件
- shell脚本删除指定目录下超过指定时间的文件
- 批量删除maven下载失败文件lastUpdated的目录
- 【git】删除某个文件的所有历史记录,批量删除远程分支
- 使用ShellExcecute打开指定目录并选中指定文件的方法
- PHP中使用glob函数实现一句话删除某个目录下的所有文件
- Asp.Net 文件操作基类(读取,删除,批量拷贝,删除,写入,获取文件夹大小,文件属性,遍历目录)(二)
- 批处理 批量删除当前目录及子目录下多个指定的扩展名文件
- [Linux使用] Linux下批量删除.svn目录
- Asp.Net 文件操作基类(读取,删除,批量拷贝,删除,写入,获取文件夹大小,文件属性,遍历目录)
- shell练习-批量目录/文件del
- 使用rm删除一个目录下的除了ttt以外的所有文件