C與指令碼的混合程式設計 (轉)

amyz發表於2007-11-16
C與指令碼的混合程式設計 (轉)[@more@]

:namespace prefix = o ns = "urn:schemas--com::office" />

C與指令碼的混合

 

作者:  陳軼飛

最後:  -09-09

關鍵詞:  c、指令碼、awk、、

 

在上寫、做網管的人,或多或少都會幾種指令碼。指令碼語言靈活的變數型別、強大的正則處理能力,再加上linux本身的管道、重定向以及豐富的命令列工具,讓你程式設計起來遊刃有餘。

而C語言固然有種種優勢,但不可否認,很多場合下,用指令碼語言更為方便,比如我們將舉例說明的對的處理。

 

先看看我們示例程式的任務:

 

假設我們有一個用c寫的程式,它有一個配置檔案 user.conf,儲存了一些資訊,user.conf定義如下:

1)、以 # 開頭的行為註釋行,不做處理

2)、允許空行

3)、如果不是1和2,那麼就是有效的資料,格式如下


# user.conf: configure file for user
# username  age  sex  country
tom  20  male us
chen 22  female cn

 

每一列分為4個欄位,欄位之間用一個或多個空白字元(空格或者製表符)隔開,欄位依次是 姓名、年齡、性別、國家

 

我們的c程式要完成對 user.conf的新增、刪除、編輯、查詢

 

這樣一個簡單的任務,用c處理起來不算複雜,不過也是要花點功夫的,而如果用指令碼語言來做,卻很簡單,能不能在c中指令碼來完成任務了?

 

Awk是linux上一種指令碼語言,它的長處在於處理有一定格式規則的檔案,例如我們們的user.conf。關於 awk 的資料有很多,oreilly公司出了專門的 awk 程式設計的書籍,網上也是可以到的。你也可以直接 man  awk看看。

 

我們先看看如何用 shell 結合 awk來完成上述任務:

1)  新增一條記錄

例如,要新增 jack  18  male  us 這樣一條記錄,可以簡單的用重定向功能

Echo  –e  “jack  18  male  us” >>  user.conf

 

現在,這條記錄被新增到 user.conf末尾了。

 

2)  刪除一條記錄

例如,現在要刪除使用者 chen 的資訊

 

cat user.conf | awk ‘!/^chen[[:blank:]]+/ {print}’ > tmp.conf; mv –f tmp.conf user.conf

 

3)、編輯一條記錄
現在,想把 tom的性別改為 female

 

cat user.conf | awk ‘{if($0 ~ /^tom[[:blank:]]+/) print $1 $2 female $3; else print}’
 
透過 system()這個,我們就可以在 c 中呼叫以上指令碼,完成任務了。
但是,system() 用起來還是覺得不爽,它的不足是隻能指令碼,卻無法獲得指令碼的輸出資料,而這通常是我們進一步處理的資料來源。(在shell和perl中,可以透過反引號( `` )來取得命令的輸出結果)。 一個解決辦法是把輸出結果重定向到一個臨時檔案中,然後在c中讀取檔案,獲取資料,最後當然還要刪除這個檔案。不過,這個方法總是讓人覺得有一點點不爽,如果能直接把指令碼執行中輸出的資料輸到我們的緩衝區來就更好了。


我寫了個小函式,叫 my_system(),透過管道以及重定向,實現了以上想法。函式原型如下:

 

int my_system(const char* pCmd, char* pResult, int size);

 

輸出資料被儲存到 pResult所指向的緩衝區中,緩衝區大小為 size,最多可以儲存 size-1的資料。

 

函式的實現放在本文的最後

 

有了這個函式以後,在 c中呼叫指令碼就更方便了,我們可以透過它來實現對 user.conf的查詢。

 

4)、查詢一個記錄
例如,我們要獲取 tom 的性別
可以用指令碼這樣來實現:


cat user.conf | awk ‘/^tom[[:blank:]]+/ {print $3}’
 
指令碼的執行結果是 tom的性別 male被輸出到螢幕上

 

在我們的 c程式中,如此呼叫 my_system(),

 

char  buf[101];
my_system(“cat user.conf | awk ‘/^tom[[:blank:]]+/ {print $3}’”, buf, 101);
 

呼叫完以後,buf中的資料就是 “male”了,怎麼樣,還算方便吧?

 

以上只是用結合指令碼完成了一個比較簡單的任務,所以我沒有把這些指令碼單獨形成指令碼檔案。如果你善於使用 perl、shell、awk,那麼可以寫出更強大的指令碼檔案來處理更復雜的問題,然後透過類似 my_system( )的方法,在 c/c++等其它語言中取得指令碼的輸出結果,實現有趣的“混合程式設計”。

 

希望你能從中得到樂趣!


#include
#include
#include
#include
#include

static int my_system(const char* pCmd, char* pResult, int size)

{

  int fd[2];

  int pid;

  int count;

  int left;

  char* p = 0;

  int maxlen = size – 1;

  memset(pResult, 0, size);

 

  if(pipe(fd))

  {

  printf("pipe errorn");

  return –1;

  }

  if((pid = fork()) == 0)

  {// chile process

 

  int fd2[2];

  if(pipe(fd2))

  {

  printf("pipe2 errorn");

  return –1;

  }

  close(1);

  dup2(fd2[1],1);

  close(fd[0]);

  close(fd2[1]);

  system(pCmd);

  read(fd2[0], pResult, maxlen);

 pResult[strlen(pResult)-1] = 0;

  write(fd[1], pResult, strlen(pResult));

  close(fd2[0]);

  exit(0);

  }

 

  // parent process

  close(fd[1]);

  p = pResult; 

  left = maxlen;

  while((count = read(fd[0], p, left))) {

  p += count;

  left -= count;

  if(left == 0)

  break;

  }

  close(fd[0]);

  return 0;

}

 

int main(void)

{

  char result[1025];

  my_system("/sbin/ifconfig", result, 1025);

  printf("the result isnn%sn", result);

  return 0;

}

 


來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/10752019/viewspace-982675/,如需轉載,請註明出處,否則將追究法律責任。

相關文章