查找命令:find,locate,xargs,exec


9.1.locate

功能:在数据库中查找,速度快,缺点:不精确,会忽略临时目录/tmp、/var/tmp

常用选项:

-i:忽略大小写

-n:打印查找结果的前n行

示例:

[root@localhost ~]# locate -i /etc/passwd   #报错,没有locate命令
-bash: locate: command not found
[root@localhost ~]# yum install -y mlocate   #安装locate命令
[root@localhost ~]# updatedb  #刷新数据库
[root@localhost ~]# locate -i /etc/passwd  #不区分大小写
/etc/passwd
/etc/passwd-
 [root@localhost ~]# locate -n 1 /etc/passwd
/etc/passwd

9.2.find(重点掌握)

功能:在目录结构中搜索文件,并执行指定的操作。此命令提供了相当多的查找条件,功能很强大。

特点:精确查找,磁盘搜索,IO读写,cpu开销相对较大

语法: find [-H] [-L] [-P] [-D debugopts] [-Olevel] [path...] [expression]

常用选项:

-name: 按照文件名查找,支持*号和[]号。

-iname:忽略大小写按照文件名查找

-perm:按照文件权限来查找,支持完全指定和-号、+号部分符合。

-prune:使用这一选项可以使find命令不在当前指定的目录中查找,如果同时使用-depth选项,那么-prune将被find命令忽略。

-user:按照文件属主来查找

-group:按照文件所属的组来查找

-mtime :-n +n按照文件的更改时间来查找

-amin(atime): -n +n按照文件的访问时间来查找

-cmin (ctime):-n +n按照文件状态的更改时间来查找

stat +文件名 可以查看amc文件时间

- n代表n天以内,+n代表那天以前,n代表当天

如下面的例子:

25 26 27 28 29 30 31 (当天为28)

n=3 表示找出28号的文件

+3 表示找出25、26、27号的文件

-3 表示找出29、30、31号的文件

-nogroup:查找无有效所属组的文件,即该文件所属的组在/etc/groups中不存在。

-nouser:查找无有效属主的文件,即该文件的属主在/etc/passwd中不存在。

-newer file1 ! -newer file2查找更改时间比文件file1新但比文件file2旧的文件。

-type:按照文件类型查找

  b - 块设备文件。

  d - 目录。

  c - 字符设备文件。

  p - 管道文件。

  l - 符号链接文件。

 f - 普通文件。

 s socket文件

-size n:[c] 查找文件长度为n块的文件,带有c时表示文件长度以字节计。

-depth:在查找文件时,首先查找当前目录中的文件,然后再在其子目录中查找。

-fstype:查找位于某一类型文件系统中的文件,这些文件系统类型通常可以在配置文件/etc/fstab中找到,该配置文件中包含了本系统中有关文件系统的信息。

-mount:在查找文件时不跨越文件系统mount点。

-follow:如果find命令遇到符号链接文件,就跟踪至链接所指向的文件。

-regex pattern:对搜索结果的 整个路径 按正规表达式进行过滤,必须对全路径考虑,例如结果是./test,那正规表达式应该用"./t.*",而不能用"t.*"

-cpio:对匹配的文件使用cpio命令。

find命令还支持使用逻辑运算符:

-a 类似&& ,连接两个不同的条件(两个条件必须同时满足)

-o 类似|| ,连接两个不同条件 (两个条件满足其一即可)

!,也是非、否

-exec:直接执行后面所跟的命令,不提示

-ok:交互式执行后面所跟的命令

-not:对条件取反

示例:

#列出当前目录及子目录的所有文件
[root@localhost home]# find
.
./file5
./old03
./old03/.gnome2
./old03/.bash_logout
./old03/.bash_profile
说明:什么参数都不加就是列出当前目录及子目录的所有文件,也可以这样写find .  ,find . -print

#查找特殊目录文件或路径
[root@localhost scripts]# find ./test -name "*.txt"
./test/test.txt
./test/town.txt
./test/number.txt
./test/name.txt
[root@localhost scripts]# find ./test -iname "*.txt"   #i忽略大小写
./test/test.txt
./test/town.txt
./test/number.txt
./test/AV.txt
./test/name.txt
#限制目录查找的深度
需要用到选项maxdepth
[root@localhost ~]# find . -maxdepth 2 -name "*.vim"  
./.vim_old/.vim
./.vim
[root@localhost ~]# find . -maxdepth 3 -name "*.vim"
说明:maxdepth后接的数字代表层级,在当前目录下,vim_old为第一层,.vim到第二层就结束了。跟这个选项还有一个相对的mindepth,这个是从子目录往上找,maxdepth是从父目录往下找。

#反向查找
[root@localhost ~]# find . -not -name "*.vim"
[root@localhost ~]# find .  ! -name "*.vim"
[root@localhost ~]#find . \( ! -name '*log*' -a ! -name '*cfg*' \) #找到不包含log和cfg的文件
说明:! -not都是代表不包含.vim后缀的文件
#以文件类型及执行其他命令查找
[root@localhost ~]# find . -type f -mtime -1 -print  #在当前文件夹下搜索最近1小时被修改的文件并打印
[root@localhost ~]# find . -type d -name ".svn"|xargs rm -rf  #删除.svn目录
[root@localhost ~]# find . -type d -name ".svn" -exec rm -rf {} \;  #删除.svn目录
[root@localhost ~]#find . -type f -name "*.php" -delete   #搜索并删除文件
[root@localhost ~]#find ./ -type f -name "*.sh"
f[root@localhost ~]#find /data/bbb -type d \( -name 'city' -o -name 'equipment' -o -name 'soldier'  \) #多目录查找
[root@localhost ~]#find . -type d -name "*p*"   #递归找出带有p字符的目录
[root@localhost ~]#find . -type f -exec ls -l {} \; >yourfile #重定向操作
[root@localhost ~]#find . -type d| sort  #列出所有目录并排序
[root@localhost ~]# find /etc -type f -empty  #查找空文件
[root@localhost ~]#find /etc -type d -empty  #查找空目录

#以文件修改时间查找
[root@localhost ~]# find . -type f -atime  -7 -print  # #在当前文件夹下搜索最近7天内被访问过的文件并打印
   1 >date
   2. Mon Aug 23 19:25:44 CST 2010
   3. >find ./ -mtime +22 -a -mtime -54   #列出mtime为7月的文件,+22表示22天以前就在7月份了,但是前面的日期没有限定,22天以前包含1-7月份啊,再加个54天以内的,那就是54-22,刚好是7月份的文件。
[root@docker-node5 ~]#find logs -type f -mtime +5 -exec rm {} \;  #在/ l o g s目录中查找更改时间在5日以前的文件并删除
[root@docker-node5 ~]#find ./ -mtime +7 -ok rm -f {} \;    #3保留7天以内的文件(7天以前删掉)
< rm ... ./file8 > ? y
< rm ... ./file7 > ? y
< rm ... ./file6 > ? y
[root@localhost mnt]# find /var/log/ -mtime +3 -type f -print  #找出3天以前被修改过的文档
[root@localhost mnt]# find /var/log/ -mtime -3 -type f -print  #找出3天内被修改过的文档
[root@localhost mnt]# find /var/log/ -mtime 3 -type f -print  #找出第三天被修改过的文档
[root@localhost mnt]# find /var/log/ -mtime +2 -mtime -4 -type f -print #找出第三天被修改过的文档
#多条件查找
[root@localhost mnt]# find /etc -name "*.sh" -o -name "*.txt" #在etc目录下查找sh或txt结尾的文件,满足其一即可
[root@docker-node5 ~]#find /etc -name "passwd*" -exec grep "sam" {} \; #查找passwd文件,并查看有没有sam这个用户。
[root@localhost mnt]# pwd
/mnt
[root@localhost mnt]# find /mnt/ -name abc
/mnt/test/abc
/mnt/abc
[root@localhost mnt]# find /mnt/ -path /mnt/test -prune -o -name abc -print  #查找abc,除过test下的abc
/mnt/abc
#以文件权限查找
[root@localhost ~]#find . -type f -perm 644 -print  #列出具有特定权限的文件
[root@localhost ~]#find . -group root -exec ls -l {} \;  #搜索属于root组的文件
[root@localhost ~]# find / -perm 2644  #查找有644及s属性
[root@localhost ~]# find / -maxdepth 2 -perm /u=s 2>/dev/null
/bin/umount
/bin/su
/bin/mount
/bin/ping
/bin/ping6
/sbin/unix_chkpwd
/sbin/pam_timestamp_check
[root@localhost ~]# ll /bin/umount 
-rwsr-xr-x. 1 root root 53472 Oct 15  2014 /bin/umount
[root@localhost ~]# find / -maxdepth 1 -perm /u=r
/
/lib64
/boot
/bin
/home
/usr
[root@localhost ~]# find . -user root  
#以文件大小查找
[root@localhost ~]# find . -type f -size +2k  #搜索文件大于2k的文件
#支持正则表达式查找
-regextype 指定所使用的正则表达式类型,可选的有emacs(默认),posix-awk,posix-basic,posix-egrep,posix-extended,喜欢grep -E,就用posix-egrep
用find查找目录下以1-3位数字命名的文件
$find ./ -regextype posix-egrep -regex '.*/[0-9]{1,3}'
[root@localhost mnt]# find /etc -regex ".*\.\(txt\|sh\)"  #在etc目录下查找txt及sh结尾的文件
/etc/kde/env/imsettings-kde.sh
/etc/pki/nssdb/pkcs11.txt
/etc/X11/xinit/xinitrc.d/50-xinput.sh
/etc/X11/xinit/xinitrc.d/localuser.sh
/etc/X11/xinit/xinitrc.d/00-start-message-bus.sh
/etc/bash_completion.d/gdbus-bash-completion.sh
#查找隐藏文件
[root@localhost ~]# find ~ -type f -name ".*"
说明:查找家目录下的隐藏文件
-exec会把find查找到的结果一次性全部交给后面的命令来进行处理,有时候系统对能够传递给exec的命令长度有限制,这样在find命令运行几分钟之后,就会出现 溢出错误。错误信息通常是“参数列太长”或“参数列溢出”。这就是xargs命令的用处所在 ,特别是与find命令一起使用。有些时候匹配到的文件发起的一个进程,并非将匹配到的文件全部作为参数一次执行,会导致进程过多,系统性能下降的问题,因此效率不高。

xargs会把find查找到的结果逐一交给后面的命令进行来处理,而xargs命令每次只获取一部分文件而不是全部,不像-exec选项那样。这样它可以先处理最先获取的一部分文件,然后是下一批,继续下去,直到结束。
-ok和-exec的作用相同,只不过以一种更为安全的模式来执行该参数所给出的shell命令,在执行每一个命令之前,都会给出提示,让用户来确定是否执行

9.3.xargs

功能:从标准输入来执行命令

常用选项:

-a file   从指定文件读取数据作为标准输入

-0      处理包含空格的文件名,print0

-d  delimiter  分隔符,默认是空格分隔显示

-i      标准输入的结果以{}代替

-I      标准输入的结果以指定的名字代替

-t      显示执行命令

-p      交互式提示是否执行命令

-n      最大命令行参数

--show-limits  查看系统命令行长度限制

示例:

上面find的例子已经很多了

介绍两个常用的:

[root@localhost scripts]# cat number.txt   
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
[root@localhost scripts]# cat number.txt |xargs -n1  #列变行,-n1就是最大1列打印
1
2
3
4
5
6
7
8
[root@localhost scripts]# cat number.txt |xargs -n2  #这里就是最大两列打印
1 2
3 4
5 6
7 8
9 10
11 12
13 14
15 16
17 18
19 20
[root@localhost scripts]# cat number.txt |xargs -n3   #3列显示
1 2 3
4 5 6
7 8 9
10 11 12
13 14 15
16 17 18
19 20


9.4.练习

1、找出根目录下的所有块设备文件,并且将标准输出及标准错误重定向到/tmp/find.test文件中

[root@localhost mnt]# find / -type b > /tmp/find.test 2>&1

2、找出/root目录下小于2M的文件并长列出,同时将其追加到/tmp/find.test文件中

1M:等于1M

+1M:大于1M

-1M:小于1M


[root@localhost mnt]# find /root/ -size -2M -exec ls -l {} \; >>/tmp/find.test

3、在/home/test目录中创建10个文件,并且修改file1~file5的时间为系统时间的5天前,file6的时间为5月28,file7的为5月29,file8的为5月30

[root@localhost home]# pwd
/home
[root@localhost home]# touch file{1..10}
[root@localhost home]# ls
file1   file2  file4  file6  file8  old01  old03  old05  old07  old09  yyl
file10  file3  file5  file7  file9  old02  old04  old06  old08  old10  yyl01
[root@localhost home]# touch -d $(date +%Y%m%d) --date="6 days ago" file{1..5}
[root@localhost home]# ll
total 48
-rw-r--r-- 1 root  root     0 May 25 15:37 file1
-rw-r--r-- 1 root  root     0 May 31 15:35 file10
-rw-r--r-- 1 root  root     0 May 25 15:37 file2
-rw-r--r-- 1 root  root     0 May 25 15:37 file3
-rw-r--r-- 1 root  root     0 May 25 15:37 file4
-rw-r--r-- 1 root  root     0 May 25 15:37 file5
-rw-r--r-- 1 root  root     0 May 31 15:35 file6
-rw-r--r-- 1 root  root     0 May 31 15:35 file7
-rw-r--r-- 1 root  root     0 May 31 15:35 file8
-rw-r--r-- 1 root  root     0 May 31 15:35 file9
[root@localhost home]# touch -d 20170528 file6
[root@localhost home]# touch -d 20170529 file7
[root@localhost home]# touch -d 20170530 file8
[root@localhost home]# ll
total 48
-rw-r--r-- 1 root  root     0 May 25 15:37 file1
-rw-r--r-- 1 root  root     0 May 31 15:35 file10
-rw-r--r-- 1 root  root     0 May 25 15:37 file2
-rw-r--r-- 1 root  root     0 May 25 15:37 file3
-rw-r--r-- 1 root  root     0 May 25 15:37 file4
-rw-r--r-- 1 root  root     0 May 25 15:37 file5
-rw-r--r-- 1 root  root     0 May 28 00:00 file6
-rw-r--r-- 1 root  root     0 May 29 00:00 file7
-rw-r--r-- 1 root  root     0 May 30 00:00 file8
-rw-r--r-- 1 root  root     0 May 31 15:35 file9

要求:

1)找出5天以前的文件并将其拷贝到/backup目录

[root@localhost home]# find ./ -mtime +5 -ok cp {} /backup/ \;

2)找出5天以内的文件并将其拷贝到192.168.5.1上的/home/test目录里,并且重命名成自己的名字。redhat用户密码为123

[root@localhost home]# find /home/redhat -mtime -5 -exec scp {} [email protected]:/home/test/zhangsan \;