linux下find和cp來實現文件的查找和複製

目的:
想查找webui_jsp工程的下面的的postgres驅動程序文件並複製到一個指定的目錄。
這樣做可以省去了,我查找到文件並且輸入很長文件目錄的麻煩。

/xxx/$ find webui_jsp/ -name *post*.jar -exec cp {} /tmp/ \;

如果分開來做的話,可以有下面的步驟

/workspace$ find webui_jsp/ -name *post*.jar
webui_jsp/WebRoot/WEB-INF/lib/postgresql-8.3-603.jdbc2.jar
/workspace$ cp webui_jsp/WebRoot/WEB-INF/lib/postgresql-8.3-603.jdbc2.jar /tmp


這兩個步驟就需要輸入很長的目錄,比較麻煩。

$$$$$$$$$$$$$$$$$$$$$$$$$$$$$

由於find具有強大的功能,所以它的選項也很多,其中大部分選項都值得我們花時間來了解一下。即使系統 中含有網絡文件系統(NFS),find命令在該文件系統中同樣有效,只你具有相應的權限。
在運行一個非常消耗資源的find命令時,很多人都傾向於把它放在後臺執行,因爲遍歷一個大的文件系統可能會花費很長的時間(這裏是指30G字節以上的文 件系統)。

Find命令的一般形式爲:
find pathname -options [-print -exec -ok]

讓我們來看看該命令的參數:
pathname:   find命令所查找的目錄路徑。例如用.來表示當前目錄,用/來表示系統根目錄。
-print:    find命令將匹配的文件輸出到標準輸出。
-exec:     find命令對匹配的文件執行該參數所給出的shell命令。相應命令的形式爲'command' {} \;,注意{}和\;之間的空格。
-ok:    和-exec的作用相同,只不過以一種更爲安全的模式來執行該參數所給出的shell命令,在執行每一個命令之前,都會給出提示,讓用戶來確定是否執行。
find命令選項:
-name:按照文件名查找文件。
-inum: 按照inode號查找文件
-perm:按照文件權限來查找文件。
-prune:使用這一選項可以使f i n d命令不在當前指定的目錄中查找,如果同時使用-depth選項,那麼-prune將被find命令忽略。
-user: 按照文件屬主來查找文件。
-group:按照文件所屬的組來查找文件。
-mtime -n +n:按照文件的更改時間來查找文件,-n表示文件更改時間距現在n天以內,+n表示文件更改時 間距現在n天以前。Find命令還有-atime和-ctime選項,但它們都和-mtime選項。
-nogroup:查找無有效所屬組的文件,即該文件所屬的組在/etc/groups中不存在。
-nouser:查找無有效屬主的文件,即該文件的屬主在/etc/passwd中不存在。
-newer file1 ! -newer 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命令,將這些文件備份到磁帶設備中。
\(expr\): 如果expr爲真,那麼表達式爲真;用來組合用"與"或"或"組合的條件
!expr: 如果expr爲假,那麼表達式爲真
可用-a或一個空格表示兩個條件的邏輯與,用-o表示兩個條件的邏輯或.注意在括號的前後和-o的前後至 少需要一個空格,複雜表達式可用圓括括號括起來,如\(and\).
使用find時,只要把想要的操作寫在一個文件裏,就可以用exec來配合find查找,很方便的
(在有些操作系統中只允許-exec選項執行諸如ls或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   ./conf.d/README
在/logs目錄中查找更改時間在5日以前的文件並刪除它們:
$ find logs -type f -mtime +5 -exec rm {} \;
記住,在shell中用任何方式刪除文件之前,應當先查看相應的文件,一定要小心!當使用諸如mv或rm 命令時,可以使用-exec選項的安全模式。它將在對每個匹配到的文件進行操作之前提示你。
在下面的例子中, find命令在當前目錄中查找所有文件名以.conf結尾、更改時間在5日以上的文件,並刪除它們,只不過在刪除之前先給出提示。
find . -name "*.conf" -mtime +5 -ok rm {} \;
< rm ... ./conf/httpd.conf > ? n
按y鍵刪除文件,按n鍵不刪除。
遞歸搜索目錄/usr和.目錄,並找出所有inode號爲258072的文件,並打印這些文件的絕對路徑 名:
find /usr . -inum 258072 -print
搜索名爲core的文件或擴展名爲.ps或.o的文件,並顯示它們的絕對路徑名,然後將它們從文件結構中 刪除
find . \( -name core -o -name '*.ps' -o -name '*.o' \) -print -exec rm {} \;
任何形式的命令都可以在-exec選項中使用。
在下面的例子中我們使用grep命令。find命令首先匹配所有文件名爲“passwd*”的文件,例如passwd、passwd.old、 passwd.bak,然後執行grep命令看看在這些文件中是否存在一個sam用戶。
find /etc -name "passwd*" -exec grep "sam" {} \;
sam:x:501:501::/usr/sam:/bin/bash
查找當前用戶主目錄下的所有文件,下面兩種方法都可以使用:
$ find $HOME -print
$ find ~ -print
爲了在當前目錄中文件屬主具有讀、寫權限,並且文件所屬組的用戶和其他用戶具有讀權限的文件,可以用:
$ find . -type f -perm 644 -exec ls -l {} \;
爲了查找系統中所有文件長度爲0的普通文件,並列出它們的完整路徑,可以用:
$ find / -type f -size 0 -exec ls -l {} \;
查找/var/logs目錄中更改時間在7日以前的普通文件,並在刪除之前詢問它們:
$ find /var/logs -type f -mtime +7 -ok rm {} \;
爲了查找系統中所有屬於root組的文件,可以用:
$find . -group root -exec ls -l {} \;
-rw-r--r-- 1 root     root       595 10月 31 01:09 ./fie1
下面的find命令將刪除當目錄中訪問時間在7日以來、含有數字後綴的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
爲了查找當前文件系統中的所有目錄並排序,可以用:
$ find . -type d   |sort
爲了查找系統中所有的r m t磁帶設備,可以用:
$ find /dev/rmt -print
-------------------------------------------------------------------------
xargs

     在使用find命令的-exec選項處理匹配到的文件時, find命令將所有匹配到的文件一起傳遞給exec執行。但有些系統對能夠傳遞給exec的命令長度有限制,這樣在find命令運行幾分鐘之後,就會出現 溢出錯誤。錯誤信息通常是“參數列太長”或“參數列溢出”。這就是xargs命令的用處所在,特別是與find命令一起使用。
      find命令把匹配到的文件傳遞給xargs命令,而xargs命令每次只獲取一部分文件而不是全部,不像-exec選項那樣。這樣它可以先處理最先獲取 的一部分文件,然後是下一批,並如此繼續下去。
      在有些系統中,使用-exec選項會爲處理每一個匹配到的文件而發起一個相應的進程,並非將匹配到的文件全部作爲參數一次執行;這樣在有些情況下就會出現 進程過多,系統性能下降的問題,因而效率不高;
      而使用xargs命令則只有一個進程。另外,在使用xargs命令時,究竟是一次獲取所有的參數,還是分批取得參數,以及每一次獲取參數的數目都會根據該 命令的選項及系統內核中相應的可調參數來確定。
      來看看xargs命令是如何同find命令一起使用的,並給出一些例子。
下面的例子查找系統中的每一個普通文件,然後使用xargs命令來測試它們分別屬於哪類文件
#find . -type f -print | xargs file
在整個系統中查找內存信息轉儲文件(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
如在當前目錄下查找文件權限位爲7 5 5的文件,即文件屬主可以讀、寫、執行,其他用戶可以讀、執行的文件,可以用:
$find . -perm 755 -print
還有一種表達方法:在八進制數字前面要加一個橫槓-,表示都匹配,如-007就相當於777,-006相當於666
# find . -perm -006
如果在查找文件時希望忽略某個目錄,因爲你知道那個目錄中沒有你所要查找的文件,那麼可以使用 -prune選項來指出需要忽略的目錄。在使用-prune選項時要當心,因爲如果你同時使用了-depth選項,那麼-prune選項就會被find命 令忽略。
$ find /apps -path "/apps/bin" -prune -o -print
補一個:使用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

爲了查找屬主帳戶已經被刪除的文件,可以使用- n o u s e r選項。這樣就能夠找到那些屬主在/etc/passwd文件中沒有有效帳戶的文件。在使用-nouser選項時,不必給出用戶名; f i n d命令能夠爲你完成相應的工作。
例如,希望在/home目錄下查找所有的這類文件,可以用:

$ find /home -nouser -print

如果希望按照更改時間來查找文件,可以使用mtime,atime或ctime選項。如果系統突然沒有可用空間了,很有可能某一個文件的長度在此期 間增長迅速,這時就可以用mtime選項來查找這樣的文件。
用減號-來限定更改時間在距今n日以內的文件,而用加號+來限定更改時間在距今n日以前的文件。
希望在系統根目錄下查找更改時間在5日以內的文件,可以用:

$ find / -mtime -5 -print

查找更改時間比文件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

可以按照文件長度來查找文件,這裏所指的文件長度既可以用塊(block)來計量,也可以用字節來計量。以字節計量文件長度的表達形式爲N c;以塊計量文件長度只用數字表示即可。
在按照文件長度查找文件時,一般使用這種以字節表示的文件長度,在查看文件系統的大小,因爲這時使用塊來計量更容易轉換。

在當前目錄下查找文件長度大於1M字節的文件:

$ find . -size +1000000c -print

在當前目錄下查找長度超過10塊的文件(一塊等於512字節):

$ find . -size +10 -print

在使用find命令時,可能希望先匹配所有的文件,再在子目錄中查找。使用depth選項就可以使find命令這樣做。這樣做的一個原因就是,當在 使用find命令向磁帶上備份文件系統時,希望首先備份所有的文件,其次再備份子目錄中的文件。

在下面的例子中,find命令從文件系統的根目錄開始,查找一個名爲CON.FILE的文件。
它將首先匹配所有的文件然後再進入子目錄中查找。

$ find / -name "CON.FILE" -depth -print

在當前的文件系統中查找文件(不進入其他文件系統),可以使用find命令的mount選項。
從當前目錄開始查找位於本文件系統中文件名以XC結尾的文件:

$ find . -name "*.XC" -mount -print

cpio命令可以用來向磁帶設備備份文件或從中恢復文件。可以使用find命令在整個文件系統中(更多的情況下是在部分文件系統中)查找文件,然後用 cpio命令將其備份到磁帶上。
如果希望使用cpio命令備份/etc,/home和/apps目錄中的文件,可以使用下面所給出的命令,不過要記住你是在文件系統的根目錄下:

#find etc home apps -depthe -print | cpio -ivcdC65535 -o

在上面的例子中,應當注意到路徑中缺少/。這叫作相對路徑。之所以使用相對路徑,是因爲在從磁帶中恢復這些文件的時候,可以選擇恢復文件的路徑。例 如,可以將這些文件先恢復到另外一個目錄中,對它們進行某些操作後,再恢復到原始目錄中。如果在備份時使用了絕對路徑,例如/etc,那麼在恢復時,就只 能恢復到/etc目錄中去,別無其他選擇。在上面的例子中,我告訴f i n d命令首先進入/etc目錄,然後是/home和/apps目錄,先匹配這些目錄下
的文件,然後再匹配其子目錄中的文件,所有這些結果將通過管道傳遞給cpio命令進行備份。
順便說一下,在上面的例子中cpio命令使用了C65536選項,本可以使用B選項,不過這樣每塊的大小隻有512字節,而使用了C65536選項後,塊 的大小變成了64K 字節(65536/1024)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章