Linux檔案尋找命令find,xargs詳述

來源:互聯網
上載者:User

總結:zhy2111314 來自:LinuxSir.Org 整理:北南南北 摘要: 本文是find 命令的詳細說明,可貴的是針對參數舉了很多的執行個體,大量的例證,讓初學者更為容易理解;本文是zhyfly兄貼在論壇中;我對本文進行了再次整理,為方便大家閱讀;
目錄
著作權聲明 前言:關於find命令
一、find 命令格式
      1、find命令的一般形式為;       2、find命令的參數;       3、find命令選項;       4、使用exec或ok來執行shell命令;
二、find命令的例子;
      1、尋找目前使用者主目錄下的所有檔案;       2、為了在目前的目錄中檔案屬主具有讀、寫入權限,並且檔案所屬組的使用者和其他使用者具有讀許可權的檔案;       3、為了尋找系統中所有檔案長度為0的普通檔案,並列出它們的完整路徑;       4、尋找/var/logs目錄中更改時間在7日以前的普通檔案,並在刪除之前詢問它們;       5、為了尋找系統中所有屬於root組的檔案;       6、find命令將刪除當目錄中訪問時間在7日以來、含有數字尾碼的admin.log檔案       7、為了尋找當前檔案系統中的所有目錄並排序;       8、為了尋找系統中所有的rmt磁帶裝置;
三、xargs 四、find 命令的參數;
      1、使用name選項       2、用perm選項       3、忽略某個目錄       4、使用find尋找檔案的時候怎麼避開某個檔案目錄       5、使用user和nouser選項       6、使用group和nogroup選項       7、按照更改時間或訪問時間等尋找檔案       8、尋找比某個檔案新或舊的檔案       9、使用type選項       10、使用size選項       11、使用depth選項       12、使用mount選項
五、關於本文 六、相關文檔
+++++++++++++++++++++++++++++++++++++++++++++++++ 本文 +++++++++++++++++++++++++++++++++++++++++++++++++
著作權聲明
本文是zhyfly兄貼在LinuxSir.Org 的一個文章而整理出來的,如果您對著作權有疑問,請在本帖後面跟帖。謝謝;本文的HTML版本由北南南北整理;修改了整篇文檔的全形及解說文字中的單詞中每個字母空格的問題;為標題加了編號,方便大家閱讀;
前言:關於find命令
由於find具有強大的功能,所以它的選項也很多,其中大部分選項都值得我們花時間來瞭解一下。即使系統中含有網路檔案系統( NFS),find命令在該檔案系統中**,只你具有相應的許可權。
在運行一個非常消耗資源的find命令時,很多人都傾向於把它放在後台執行,因為遍曆一個大的檔案系統可能會花費很長的時間(這裡是指30G位元組以上的檔案系統)。
一、find 命令格式
1、find命令的一般形式為;
find pathname -options [-print -exec -ok ...]
2、find命令的參數;
pathname: find命令所尋找的目錄路徑。例如用.來表示目前的目錄,用/來表示系統根目錄。 -print: find命令將匹配的檔案輸出到標準輸出。 -exec: find命令對匹配的檔案執行該參數所給出的shell命令。相應命令的形式為'command' {  } \;,注意{   }和\;之間的空格。 -ok: 和-exec的作用相同,只不過以一種更為安全的模式來執行該參數所給出的shell命令,在執行每一個命令之前,都會給出提示,讓使用者來確定是否執行。
3、find命令選項
-name
按照檔案名稱尋找檔案。
-perm 按照檔案許可權來尋找檔案。
-prune 使用這一選項可以使find命令不在當前指定的目錄中尋找,如果同時使用-depth選項,那麼-prune將被find命令忽略。
-user 按照檔案屬主來尋找檔案。
-group 按照檔案所屬的組來尋找檔案。
-mtime -n +n 按照檔案的更改時間來尋找檔案, - n表示檔案更改時間距現在n天以內,+ n表示檔案更改時間距現在n天以前。find命令還有-atime和-ctime 選項,但它們都和-m time選項。
-nogroup 尋找無有效所屬組的檔案,即該檔案所屬的組在/etc/groups中不存在。
-nouser 尋找無有效屬主的檔案,即該檔案的屬主在/etc/passwd中不存在。 -newer file1 ! file2
尋找更改時間比檔案file1新但比檔案file2舊的檔案。 -type
尋找某一類型的檔案,諸如:
b - 塊裝置檔案。 d - 目錄。 c - 字元裝置檔案。 p - 管道檔案。 l - 符號連結檔案。 f - 普通檔案。
-size n:[c] 尋找檔案長度為n塊的檔案,帶有c時表示檔案長度以位元組計。 -depth:在尋找檔案時,首先尋找目前的目錄中的檔案,然後再在其子目錄中尋找。 -fstype:尋找位於某一類型檔案系統中的檔案,這些檔案系統類型通常可以在設定檔/etc/fstab中找到,該設定檔中包含了本系統中有關檔案系統的資訊。
-mount:在尋找檔案時不跨越檔案系統mount點。 -follow:如果find命令遇到符號連結檔案,就跟蹤至連結所指向的檔案。 -cpio:對匹配的檔案使用cpio命令,將這些檔案備份到磁帶裝置中。
另外,下面三個的區別:    -amin n   尋找系統中最後N分鐘訪問的檔案
  -atime n   尋找系統中最後n*24小時訪問的檔案
  -cmin n   尋找系統中最後N分鐘被改變檔案狀態的檔案
  -ctime n   尋找系統中最後n*24小時被改變檔案狀態的檔案
    -mmin n   尋找系統中最後N分鐘被改變檔案資料的檔案
  -mtime n   尋找系統中最後n*24小時被改變檔案資料的檔案
4、使用exec或ok來執行shell命令
使用find時,只要把想要的操作寫在一個檔案裡,就可以用exec來配合find尋找,很方便的
在有些作業系統中只允許-exec選項執行諸如l s或ls -l這樣的命令。大多數使用者使用這一選項是為了尋找舊檔案並刪除它們。建議在真正執行rm命令刪除檔案之前,最好先用ls命令看一下,確認它們是所要刪除的檔案。
exec選項後面跟隨著所要執行的命令或指令碼,然後是一對兒{ },一個空格和一個\,最後是一個分號。為了使用exec選項,必須要同時使用print選項。如果驗證一下find命令,會發現該命令只輸出從當前路徑起的相對路徑及檔案名稱。
例如:為了用ls -l命令列出所匹配到的檔案,可以把ls -l命令放在find命令的-exec選項中
# find . -type f -exec ls -l {  } \; -rw-r--r--    1 root     root        34928 2003-02-25  ./conf/httpd.conf -rw-r--r--    1 root     root        12959 2003-02-25  ./conf/magic -rw-r--r--    1 root     root          180 2003-02-25 begin_of_the_skype_highlighting              180 2003-02-25      end_of_the_skype_highlighting  ./conf.d/README
上面的例子中,find命令匹配到了目前的目錄下的所有普通檔案,並在-exec選項中使用ls -l命令將它們列出。 在/logs目錄中尋找更改時間在5日以前的檔案並刪除它們:
$ find logs -type f -mtime +5 -exec rm {  } \;
記住:在shell中用任何方式刪除檔案之前,應當先查看相應的檔案,一定要小心!當使用諸如mv或rm命令時,可以使用-exec選項的安全模式。它將在對每個匹配到的檔案進行操作之前提示你。
在下面的例子中, find命令在目前的目錄中尋找所有檔案名稱以.LOG結尾、更改時間在5日以上的檔案,並刪除它們,只不過在刪除之前先給出提示。
$ find . -name "*.conf"  -mtime +5 -ok rm {  } \; < rm ... ./conf/httpd.conf > ? n
按y鍵刪除檔案,按n鍵不刪除。
任何形式的命令都可以在-exec選項中使用。
在下面的例子中我們使用grep命令。find命令首先匹配所有檔案名稱為“ passwd*”的檔案,例如passwd、passwd.old、passwd.bak,然後執行grep命令看看在這些檔案中是否存在一個sam使用者。
# find /etc -name "passwd*" -exec grep "sam" {  } \; sam:501:501::/usr/sam:/bin/bash
二、find命令的例子;
1、尋找目前使用者主目錄下的所有檔案:
下面兩種方法都可以使用
$ find $HOME -print $ find ~ -print
2、讓目前的目錄中檔案屬主具有讀、寫入權限,並且檔案所屬組的使用者和其他使用者具有讀許可權的檔案;
$ find . -type f -perm 644 -exec ls -l {  } \;
3、為了尋找系統中所有檔案長度為0的普通檔案,並列出它們的完整路徑;
$ find / -type f -size 0 -exec ls -l {  } \;
4、尋找/var/logs目錄中更改時間在7日以前的普通檔案,並在刪除之前詢問它們;
$ find /var/logs -type f -mtime +7 -ok rm {  } \;
5、為了尋找系統中所有屬於root組的檔案;
$find . -group root -exec ls -l {  } \; -rw-r--r--    1 root     root          595 10月 31 01:09 ./fie1
6、find命令將刪除當目錄中訪問時間在7日以來、含有數字尾碼的admin.log檔案。
該命令只檢查三位元字,所以相應檔案的尾碼不要超過999。先建幾個admin.log*的檔案 ,才能使用下面這個命令
$ find . -name "admin.log[0-9][0-9][0-9]" -atime -7  -ok rm {  } \; < rm ... ./admin.log001 > ? n < rm ... ./admin.log002 > ? n < rm ... ./admin.log042 > ? n < rm ... ./admin.log942 > ? n
7、為了尋找當前檔案系統中的所有目錄並排序;
$ find . -type d | sort
8、為了尋找系統中所有的rmt磁帶裝置;
$ find /dev/rmt -print
三、xargs
xargs - build and execute command lines from standard input
在使用find命令的-exec選項處理匹配到的檔案時, find命令將所有匹配到的檔案一起傳遞給exec執行。但有些系統對能夠傳遞給exec的命令長度有限制,這樣在find命令運行幾分鐘之後,就會出現溢出錯誤。錯誤資訊通常是“參數列太長”或“參數列溢出”。這就是xargs命令的用處所在,特別是與find命令一起使用。
find命令把匹配到的檔案傳遞給xargs命令,而xargs命令每次只擷取一部分檔案而不是全部,不像-exec選項那樣。這樣它可以先處理最先擷取的一部分檔案,然後是下一批,並如此繼續下去。
在有些系統中,使用-exec選項會為處理每一個匹配到的檔案而發起一個相應的進程,並非將匹配到的檔案全部作為參數一次執行;這樣在有些情況下就會出現進程過多,系統效能下降的問題,因而效率不高;
而使用xargs命令則只有一個進程。另外,在使用xargs命令時,究竟是一次擷取所有的參數,還是分批取得參數,以及每一次擷取參數的數目都會根據該命令的選項及系統核心中相應的可調參數來確定。
來看看xargs命令是如何同find命令一起使用的,並給出一些例子。
下面的例子尋找系統中的每一個普通檔案,然後使用xargs命令來測試它們分別屬於哪類檔案
#find . -type f -print | xargs file ./.kde/Autostart/Autorun.desktop: UTF-8 Unicode English text ./.kde/Autostart/.directory:      ISO-8859 text\ ......
在整個系統中尋找記憶體資訊轉儲檔案(core dump) ,然後把結果儲存到/tmp/core.log 檔案中:
$ find / -name "core" -print | xargs echo "" >/tmp/core.log
上面這個執行太慢,我改成在目前的目錄下尋找
#find . -name "file*" -print | xargs echo "" > /temp/core.log # cat /temp/core.log ./file6
在目前的目錄下尋找所有使用者具有讀、寫和執行許可權的檔案,並收回相應的寫入權限:
# ls -l drwxrwxrwx    2 sam      adm          4096 10月 30 20:14 file6 -rwxrwxrwx    2 sam      adm             0 10月 31 01:01 http3.conf -rwxrwxrwx    2 sam      adm             0 10月 31 01:01 httpd.conf
# find . -perm -7 -print | xargs chmod o-w # ls -l drwxrwxr-x    2 sam      adm          4096 10月 30 20:14 file6 -rwxrwxr-x    2 sam      adm             0 10月 31 01:01 http3.conf -rwxrwxr-x    2 sam      adm             0 10月 31 01:01 httpd.conf
用grep命令在所有的普通檔案中搜尋hostname這個詞:
# find . -type f -print | xargs grep "hostname" ./httpd1.conf:#     different IP addresses or hostnames and have them handled by the ./httpd1.conf:# VirtualHost: If you want to maintain multiple domains/hostnames on your
用grep命令在目前的目錄下的所有普通檔案中搜尋hostnames這個詞:
# find . -name \* -type f -print | xargs grep "hostnames" ./httpd1.conf:#     different IP addresses or hostnames and have them handled by the ./httpd1.conf:# VirtualHost: If you want to maintain multiple domains/hostnames on your
注意,在上面的例子中, \用來取消find命令中的*在shell中的特殊含義。
find命令配合使用exec和xargs可以使使用者對所匹配到的檔案執行幾乎所有的命令。
四、find 命令的參數
下面是find一些常用參數的例子,有用到的時候查查就行了,像上面前幾個貼子,都用到了其中的的一些參數,也可以用man或查看論壇裡其它貼子有find的命令手冊
1、使用name選項
檔案名稱選項是find命令最常用的選項,要麼單獨使用該選項,要麼和其他選項一起使用。
可以使用某種檔案名稱模式來匹配檔案,記住要用引號將檔案名稱模式引起來。
不管當前路徑是什麼,如果想要在自己的根目錄$HOME中尋找檔案名稱符合*.txt的檔案,使用~作為 'pathname'參數,波浪號~代表了你的$HOME目錄。
$ find ~ -name "*.txt" -print
想要在目前的目錄及子目錄中尋找所有的‘ *.txt’檔案,可以用:
$ find . -name "*.txt" -print
想要的目前的目錄及子目錄中尋找檔案名稱以一個大寫字母開頭的檔案,可以用:
$ find . -name "[A-Z]*" -print
想要在/etc目錄中尋找檔案名稱以host開頭的檔案,可以用:
$ find /etc -name "host*" -print
想要尋找$HOME目錄中的檔案,可以用:
$ find ~ -name "*" -print 或find . -print
要想讓系統高負荷運行,就從根目錄開始尋找所有的檔案。
$ find / -name "*" -print
如果想在目前的目錄尋找檔案名稱以兩個小寫字母開頭,跟著是兩個數字,最後是.txt的檔案,下面的命令就能夠返回名為ax37.txt的檔案:
$find . -name "[a-z][a-z][0--9][0--9].txt" -print
2、用perm選項
按照檔案許可權模式用-perm選項,按檔案許可權模式來尋找檔案的話。最好使用八進位的許可權標記法。
如在目前的目錄下尋找檔案許可權位為755的檔案,即檔案屬主可以讀、寫、執行,其他使用者可以讀、執行的檔案,可以用:
$ find . -perm 755 -print
還有一種表達方法:在八位元字前面要加一個橫杠-,表示都匹配,如-007就相當於777,-006相當於666
# ls -l -rwxrwxr-x    2 sam      adm             0 10月 31 01:01 http3.conf -rw-rw-rw-    1 sam      adm         34890 10月 31 00:57 httpd1.conf -rwxrwxr-x    2 sam      adm             0 10月 31 01:01 httpd.conf drw-rw-rw-    2 gem      group        4096 10月 26 19:48 sam -rw-rw-rw-    1 root     root         2792 10月 31 20:19 temp
# find . -perm 006 # find . -perm -006 ./sam ./httpd1.conf ./temp
-perm mode:檔案許可正好符合mode
-perm +mode:檔案許可部分符合mode
-perm -mode: 檔案許可完全符合mode
3、忽略某個目錄
如果在尋找檔案時希望忽略某個目錄,因為你知道那個目錄中沒有你所要尋找的檔案,那麼可以使用-prune選項來指出需要忽略的目錄。在使用-prune選項時要當心,因為如果你同時使用了-depth選項,那麼-prune選項就會被find命令忽略。
如果希望在/apps目錄下尋找檔案,但不希望在/apps/bin目錄下尋找,可以用:
$ find /apps -path "/apps/bin" -prune -o -print
4、使用find尋找檔案的時候怎麼避開某個檔案目錄
比如要在/usr/sam目錄下尋找不在dir1子目錄之內的所有檔案
find /usr/sam -path "/usr/sam/dir1" -prune -o -print
find [-path ..] [expression] 在路徑列表的後面的是運算式
-path "/usr/sam" -prune -o -print 是 -path "/usr/sam" -a -prune -o -print 的簡寫運算式按順序求值, -a 和 -o 都是短路求值,與 shell 的 && 和 || 類似如果 -path "/usr/sam" 為真,則求值 -prune , -prune 返回真,與邏輯運算式為真;否則不求值 -prune,與邏輯運算式為假。如果 -path "/usr/sam" -a -prune 為假,則求值 -print ,-print返回真,或邏輯運算式為真;否則不求值 -print,或邏輯運算式為真。
這個運算式組合特例可以用偽碼寫為
if -path "/usr/sam"  then           -prune else           -print
避開多個檔案夾
find /usr/sam \( -path /usr/sam/dir1 -o -path /usr/sam/file1 \) -prune -o -print
圓括弧表示運算式的結合。 \ 表示引用,即指示 shell 不對後面的字元作特殊解釋,而留給 find 命令去解釋其意義。
尋找某一確定檔案,-name等選項加在-o 之後
#find /usr/sam  \(-path /usr/sam/dir1 -o -path /usr/sam/file1 \) -prune -o -name "temp" -print
5、使用user和nouser選項
按檔案屬主尋找檔案,如在$HOME目錄中尋找檔案屬主為sam的檔案,可以用:
$ find ~ -user sam -print
在/etc目錄下尋找檔案屬主為uucp的檔案:
$ find /etc -user uucp -print
為了尋找屬主帳戶已經被刪除的檔案,可以使用-nouser選項。這樣就能夠找到那些屬主在/etc/passwd檔案中沒有有效帳戶的檔案。在使用-nouser選項時,不必給出使用者名稱; find命令能夠為你完成相應的工作。
例如,希望在/home目錄下尋找所有的這類檔案,可以用: $ find /home -nouser -print
6、使用group和nogroup選項
就像user和nouser選項一樣,針對檔案所屬於的使用者組, find命令也具有同樣的選項,為了在/apps目錄下尋找屬於gem使用者組的檔案,可以用:
$ find /apps -group gem -print
要尋找沒有有效所屬使用者組的所有檔案,可以使用nogroup選項。下面的find命令從檔案系統的根目錄處尋找這樣的檔案
$ find / -nogroup-print
7、按照更改時間或訪問時間等尋找檔案
如果希望按照更改時間來尋找檔案,可以使用mtime,atime或ctime選項。如果系統突然沒有可用空間了,很有可能某一個檔案的長度在此期間增長迅速,這時就可以用mtime選項來尋找這樣的檔案。
用減號-來限定更改時間在距今n日以內的檔案,而用加號+來限定更改時間在距今n日以前的檔案。
希望在系統根目錄下尋找更改時間在5日以內的檔案,可以用:
$ find / -mtime -5 -print
為了在/var/adm目錄下尋找更改時間在3日以前的檔案,可以用:
$ find /var/adm -mtime +3 -print
8、尋找比某個檔案新或舊的檔案
如果希望尋找更改時間比某個檔案新但比另一個檔案舊的所有檔案,可以使用-newer選項。它的一般形式為:
newest_file_name ! oldest_file_name
其中,!是邏輯非符號。
尋找更改時間比檔案sam新但比檔案temp舊的檔案:
例:有兩個檔案
-rw-r--r--    1 sam      adm             0 10月 31 01:07 fiel -rw-rw-rw-    1 sam      adm         34890 10月 31 00:57 httpd1.conf -rwxrwxr-x    2 sam      adm             0 10月 31 01:01 httpd.conf drw-rw-rw-    2 gem      group        4096 10月 26 19:48 sam -rw-rw-rw-    1 root     root         2792 10月 31 20:19 temp
# find -newer httpd1.conf  ! -newer temp -ls 1077669    0 -rwxrwxr-x   2 sam      adm             0 10月 31 01:01 ./httpd.conf 1077671    4 -rw-rw-rw-   1 root     root         2792 10月 31 20:19 ./temp 1077673    0 -rw-r--r--   1 sam      adm             0 10月 31 01:07 ./fiel
尋找更改時間在比temp檔案新的檔案:
$ find . -newer temp -print
9、使用type選項
在/etc目錄下尋找所有的目錄,可以用:
$ find /etc -type d -print
在目前的目錄下尋找除目錄以外的所有類型的檔案,可以用: $ find . ! -type d -print
在/etc目錄下尋找所有的符號連結檔案,可以用 $ find /etc -type l -print
10、使用size選項
可以按照檔案長度來尋找檔案,這裡所指的檔案長度既可以用塊(block)來計量,也可以用位元組來計量。以位元組計量檔案長度的表達形式為N c;以塊計量檔案長度只用數字表示即可。
在按照檔案長度尋找檔案時,一般使用這種以位元組表示的檔案長度,在查看檔案系統的大小,因為這時使用塊來計量更容易轉換。 在目前的目錄下尋找檔案長度大於1 M位元組的檔案: $ find . -size +1000000c -print
在/home/apache目錄下尋找檔案長度恰好為100位元組的檔案:
$ find /home/apache -size 100c -print
在目前的目錄下尋找長度超過10塊的檔案(一塊等於512位元組):
$ find . -size +10 -print
11、使用depth選項
在使用find命令時,可能希望先匹配所有的檔案,再在子目錄中尋找。使用depth選項就可以使find命令這樣做。這樣做的一個原因就是,當在使用find命令向磁帶上備份檔案系統時,希望首先備份所有的檔案,其次再備份子目錄中的檔案。
在下面的例子中, find命令從檔案系統的根目錄開始,尋找一個名為CON.FILE的檔案。
它將首先匹配所有的檔案然後再進入子目錄中尋找。
$ find / -name "CON.FILE" -depth -print
12、使用mount選項
在當前的檔案系統中尋找檔案(不進入其他檔案系統),可以使用find命令的mount選項。
從目前的目錄開始尋找位於本檔案系統中檔案名稱以XC結尾的檔案:
$ find . -name "*.XC" -mount -print

五、關於本文
本文是find 命令的詳細說明,可貴的是針對參數舉了很多的執行個體,大量的例證,讓初學者更為容易理解;本文是zhy2111314兄貼在論壇中;我對本文進行了再次整理,為方便大家閱讀; ── 北南南北
六、相關文檔
    * 由 北南南北 在 2005/12/20 - 10:24 發表     * Linux     * 基礎知識     * 命令/SHELL/PERL     * 要發表評論,請先登入 或 註冊
-exec 和 管道符
用 -exec來執行shell命令,和通過管道符|來執行有什麼不同?
    * 由 li_qinshan 在 2008/10/15 - 18:05 發表     * 要發表評論,請先登入 或 註冊
可不可以忽略 Permission denied的
-bash-3.1$ find /home/f/a -type f -name enen -exec ls -l {} \; find: /home/f/a/liuyunge: Permission denied find: /home/f/a/ll1012: Permission denied find: /home/f/a/hebaidu: Permission denied find: /home/f/a/ET_SUN: Permission denied find: /home/f/a/Rory.Lu: Permission denied -rw-r--r-- 1 lufeng 27108 60 Nov 1 14:24 /home/f/a/lufeng/enen find: /home/f/a/shenglee2007: Permission denied find: /home/f/a/zlh750620: Permission denied find: /home/f/a/mlh720goodboy: Permission denied
    * 由 LinuxSir 在 2007/11/03 - 10:20 發表     * 要發表評論,請先登入 或 註冊
-mtime +n
-mtime +n -n
"用減號-來限定更改時間在距今n日以內的檔案,而用加號+來限定更改時間在距今n日以前的檔案"
疑問:用加號+來限定更改時間在距今 (n+1) 日以前的檔案
    * 由 LinuxSir 在 2007/03/06 - 15:33 發表     * 要發表評論,請先登入 或 註冊

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.