1. 程式人生 > >在Linux使用exec執行命令時報的哪些錯

在Linux使用exec執行命令時報的哪些錯

在linux使用exec執行命令時報的哪些錯

問題1:find: paths must precede expression

[[email protected] data]# find /oracle/backup/exp/data -name exp_table01_db01_*.dmp.gz -atime +2 exec rm -rf {}\;
find: paths must precede expression: exp_table01_db01_20170928235039.dmp.gz

描述:在執行find命令時,在查找的內容必須要把‘‘號給括起來,要不然會報錯

問題2:find: missing argument to `-exec‘

[[email protected] data]# find /oracle/backup/exp/data -name ‘exp_table01_db01_*.dmp.gz‘ -atime +2 -exec rm -rf {}\;

find: missing argument to `-exec‘
描述:在執行find命令時,在後面執行exec時,最後的\斜杠前面有有空格,要不然會報錯


知識補充

Linux中exec命令相關:

exec和source都屬於bash內部命令(builtins commands),在bash下輸入man exec或man source可以查看所有的內部命令信息。
  bash shell的命令分為兩類:外部命令和內部命令。外部命令是通過系統調用或獨立的程序實現的,如sed、awk等等。內部

命令是由特殊的文件格式(.def)所實現,如cd、history、exec等等。

  在說明exe和source的區別之前,先說明一下fork的概念。

  fork是linux的系統調用,用來創建子進程(child process)。子進程是父進程(parent process)的一個副本,從父進程那裏

獲得一定的資源分配以及繼承父進程的環境。子進程與父進程唯一不同的地方在於pid(process id)。

  環境變量(傳給子進程的變量,遺傳性是本地變量和環境變量的根本區別)只能單向從父進程傳給子進程。不管子進程的環境

變量如何變化,都不會影響父進程的環境變量。

  shell script:

  有兩種方法執行shell scripts,一種是新產生一個shell,然後執行相應的shell scripts;一種是在當前shell下執行,不

再啟用其他shell。

  新產生一個shell然後再執行scripts的方法是在scripts文件開頭加入以下語句

  #!/bin/sh

  一般的script文件(.sh)即是這種用法。這種方法先啟用新的sub-shell(新的子進程),然後在其下執行命令。

  另外一種方法就是上面說過的source命令,不再產生新的shell,而在當前shell下執行一切命令。

  source:

  source命令即點(.)命令。

  在bash下輸入man source,找到source命令解釋處,可以看到解釋”Read and execute commands from filename in the

current shell environment and …”。從中可以知道,source命令是在當前進程中執行參數文件中的各個命令,而不是另起子

進程(或sub-shell)。

  exec:

  在bash下輸入man exec,找到exec命令解釋處,可以看到有”No new process is created.”這樣的解釋,這就是說exec命

令不產生新的子進程。那麽exec與source的區別是什麽呢?

  exec命令在執行時會把當前的shell process關閉,然後換到後面的命令繼續執行。

1. 系統調用exec是以新的進程去代替原來的進程,但進程的PID保持不變。因此,可以這樣認為,exec系統調用並沒有創建新的

進程,只是替換了原來進程上下文的內容。原進程的代碼段,數據段,堆棧段被新的進程所代替。

  一個進程主要包括以下幾個方面的內容:

  (1)一個可以執行的程序

  (2) 與進程相關聯的全部數據(包括變量,內存,緩沖區)

  (3)程序上下文(程序計數器PC,保存程序執行的位置)

  2. exec是一個函數簇,由6個函數組成,分別是以excl和execv打頭的。

  執行exec系統調用,一般都是這樣,用fork()函數新建立一個進程,然後讓進程去執行exec調用。我們知道,在fork()建立

新進程之後,父進各與子進程共享代碼段,但數據空間是分開的,但父進程會把自己數據空間的內容copy到子進程中去,還有上

下文也會copy到子進程中去。而為了提高效率,采用一種寫時copy的策略,即創建子進程的時候,並不copy父進程的地址空間,

父子進程擁有共同的地址空間,只有當子進程需要寫入數據時(如向緩沖區寫入數據),這時候會復制地址空間,復制緩沖區到子

進程中去。從而父子進程擁有獨立的地址空間。而對於fork()之後執行exec後,這種策略能夠很好的提高效率,如果一開始就

copy,那麽exec之後,子進程的數據會被放棄,被新的進程所代替。

  3. exec與system的區別

  (1) exec是直接用新的進程去代替原來的程序運行,運行完畢之後不回到原先的程序中去。

  (2) system是調用shell執行你的命令,system=fork+exec+waitpid,執行完畢之後,回到原先的程序中去。繼續執行下面的

部分。

  總之,如果你用exec調用,首先應該fork一個新的進程,然後exec. 而system不需要你fork新進程,已經封裝好了。

  exec I/O重定向詳解及應用實例

  1、 基本概念(這是理解後面的知識的前提,請務必理解)

  a、 I/O重定向通常與 FD有關,shell的FD通常為10個,即 0~9;

  b、 常用FD有3個,為0(stdin,標準輸入)、1(stdout,標準輸出)、2(stderr,標準錯誤輸出),默認與keyboard、monitor

、monitor有關;

  c、 用 來改變送出的數據信道(stdout, stderr),使之輸出到指定的檔案;

  e、 0 是 與 1> 是一樣的;

  f、 在IO重定向 中,stdout 與 stderr 的管道會先準備好,才會從 stdin 讀進資料;

  g、 管道“|”(pipe line):上一個命令的 stdout 接到下一個命令的 stdin;

  h、 tee 命令是在不影響原本 I/O 的情況下,將 stdout 復制一份到檔案去;

  i、 bash(ksh)執行命令的過程:分析命令-變量求值-命令替代(``和$( ))-重定向-通配符展開-確定路徑-執行命令;

  j、 ( ) 將 command group 置於 sub-shell 去執行,也稱 nested sub-shell,它有一點非常重要的特性是:繼承父shell

的Standard input, output, and error plus any other open file descriptors。

  k、 exec 命令:常用來替代當前 shell 並重新啟動一個 shell,換句話說,並沒有啟動子 shell。使用這一命令時任何現

有環境都將會被清除。exec 在對文件描述符進行操作的時候,也只有在這時,exec 不會覆蓋你當前的 shell 環境。

  2、cmd &n 使用系統調用 dup (2) 復制文件描述符 n 並把結果用作標準輸出

  &- 關閉標準輸出

  n&- 表示將 n 號輸出關閉

  上述所有形式都可以前導一個數字,此時建立的文件描述符由這個數字指定而不是缺省的 0 或 1。如:

  ... 2>file 運行一個命令並把錯誤輸出(文件描述符 2)定向到 file。

  ... 2>&1 運行一個命令並把它的標準輸出和輸出合並。(嚴格的說是通過復制文件描述符 1 來建立文件描述符 2 ,但效果

通常是合並了兩個流。)

  我們對 2>&1詳細說明一下 :2>&1 也就是 FD2=FD1 ,這裏並不是說FD2 的值 等於FD1的值,因為 > 是改變送出的數據信

道,也就是說把 FD2 的 “數據輸出通道” 改為 FD1 的 “數據輸出通道”。如果僅僅這樣,這個改變好像沒有什麽作用,因

為 FD2 的默認輸出和 FD1的默認輸出本來都是 monitor,一樣的!

  但是,當 FD1 是其他文件,甚至是其他 FD 時,這個就具有特殊的用途了。請大家務必理解這一點。

  3、 如果 stdin, stdout, stderr 進行了重定向或關閉, 但沒有保存原來的 FD, 可以將其恢復到 default 狀態嗎?

  *** 如果關閉了stdin,因為會導致退出,那肯定不能恢復。

  *** 如果重定向或關閉 stdout和stderr其中之一,可以恢復,因為他們默認均是送往monitor(但不知會否有其他影響)。如

恢復重定向或關閉的 stdout: exec 1>&2 ,恢復重定向或關閉的stderr:exec 2>&1。

  *** 如果stdout和stderr全部都關閉了,又沒有保存原來的FD,可以用:exec 1>/dev/tty 恢復。

  4、 cmd >a 2>a 和 cmd >a 2>&1 為什麽不同?

  cmd >a 2>a :stdout和stderr都直接送往文件 a ,a文件會被打開兩遍,由此導致stdout和stderr互相覆蓋。

  cmd >a 2>&1 :stdout直接送往文件a ,stderr是繼承了FD1的管道之後,再被送往文件a 。a文件只被打開一遍,就是FD1

將其打開。

  我想:他們的不同點在於:

  cmd >a 2>a 相當於使用了兩個互相競爭使用文件a的管道;

  而cmd >a 2>&1 只使用了一個管道,但在其源頭已經包括了stdout和stderr。

  從IO效率上來講,cmd >a 2>&1的效率應該更高!

  exec 0exec 1>outfilename # 打開文件outfilename作為stdout

  exec 2>errfilename # 打開文件 errfilename作為 stderr

  exec 0&- # 關閉 FD1

  exec 5>&- # 關閉 FD5

[[email protected] shell]$ cat 1

  11 22 33 44 55

  66 22 33 11 33

  324 25 63 634 745

  [[email protected] shell]$ cat 2

  > 1.txt

  exec 4<&1

  exec 1> 1.txt

  while read line

  do

  echo $line

  done < 1

  exec 1<&4

  exec 4>&-

  [[email protected] shell]$ sh ./2

  [[email protected] shell]$ cat 1.txt

  11 22 33 44 55

  66 22 33 11 33

  324 25 63 634 745

  [[email protected] shell]$

修改ls顯示的時間格式:

ls -l --time-style ‘+%Y/%m/%d %H:%M:%S‘
total 0
-rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file1
-rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file2

--time-style 也可以存在環境變量裏:

export TIME_STYLE=‘+%Y/%m/%d %H:%M:%S‘

ls -l
total 0
-rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file1
-rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file2

Linux中內部命令和外部命令:
UNIX命令有內部命令和外部命令之分。內部命令實際上是shell程序的一部分,其中包含的是一些比較簡練的UNIX系統命令,這些命令由shell程序識別並在shell程序內部完成運行,通常在UNIX系統加載運行時shell就被加載並駐留在系統內存中。外部命令是UNIX系統中的實用程序部分,因為實用程序的功能通常都比較強大,所以它們包含的程序量也會很大,在系統加載時並不隨系統一起被加載到內存中,而是在需要時才將其調進內存。通常外部命令的實體並不包含在shell中,但是其命令執行過程是由shell 程序控制的。shell程序管理外部命令執行的路徑查找、加載存放,並控制命令的執行。
理解了內部命令和外部命令之後,你可能就會有疑問了??那麽我怎麽看一個命令是內部命令還是外部命令呢??

1、在終端下輸入man builtins,即可調出內部命令的大綱和解釋。

2、用命令enable調出來,後者將分行打印出內部命令

[[email protected] shell]# enable ls

bash: enable: ls: not a shell builtin


1. http://blog.sina.com.cn/s/blog_6238358c0100sg5n.html


在Linux使用exec執行命令時報的哪些錯