在linux上寫程式、做網管的人,或多或少都會幾種指令碼。指令碼語言靈活的變數類型、強大的Regex
處理能力,再加上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。關於 aw
k 的資料有很多,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;
}
轉自:http://www.233.com/Linux/Instructs/051207/165934648.html