在linux上寫(xiě)程序、做網(wǎng)管的人,或多或少都會(huì)幾種腳本。腳本語(yǔ)言靈活的變量類(lèi)型、強(qiáng)大的正則表達(dá)式處理能力,再加上linux系統(tǒng)本身的管道、重定向以及豐富的命令行工具,讓你編程起來(lái)游刃有余。
而C語(yǔ)言固然有種種優(yōu)勢(shì),但不可否認(rèn),很多場(chǎng)合下,用腳本語(yǔ)言更為方便,比如我們將舉例說(shuō)明的對(duì)配置文件的處理。
先看看我們示例程序的任務(wù):
假設(shè)我們有一個(gè)用c寫(xiě)的程序,它有一個(gè)配置文件 user.conf,保存了一些用戶(hù)信息,user.conf定義如下:
1)、以 # 開(kāi)頭的行為注釋行,不做處理
2)、允許空行
3)、如果不是1和2,那么就是有效的數(shù)據(jù),格式如下
# user.conf: configure file for user
# username age sex country
tom 20 male us
chen 22 female cn
每一列分為4個(gè)字段,字段之間用一個(gè)或多個(gè)空白字符(空格或者制表符)隔開(kāi),字段依次是 姓名、年齡、性別、國(guó)家
我們的c程序要完成對(duì) user.conf的添加、刪除、編輯、查詢(xún)
這樣一個(gè)簡(jiǎn)單的任務(wù),用c處理起來(lái)不算復(fù)雜,不過(guò)也是要花點(diǎn)功夫的,而如果用腳本語(yǔ)言來(lái)做,卻很簡(jiǎn)單,能不能在c中調(diào)用腳本來(lái)完成任務(wù)了?
Awk是linux上一種腳本語(yǔ)言,它的長(zhǎng)處在于處理有一定格式規(guī)則的文件,例如咱們的user.conf。關(guān)于 awk 的資料有很多,oreilly公司出了專(zhuān)門(mén)的 awk 編程的書(shū)籍,網(wǎng)上也是可以下載到的。你也可以直接 man awk看看。
我們先看看如何用 shell 結(jié)合 awk來(lái)完成上述任務(wù):
1) 添加一條記錄
例如,要添加 jack 18 male us 這樣一條記錄,可以簡(jiǎn)單的用重定向功能
Echo –e “jack 18 male us” 》》 user.conf
現(xiàn)在,這條記錄被添加到 user.conf末尾了。
2) 刪除一條記錄
例如,現(xiàn)在要?jiǎng)h除用戶(hù) chen 的信息
cat user.conf | awk ‘!/^chen[[:blank:]] / {print}“ 》 tmp.conf; mv –f tmp.conf user.conf
3)、編輯一條記錄
現(xiàn)在,想把 tom的性別改為 female
cat user.conf | awk ‘{if($0 ~ /^tom[[:blank:]] /) print $1 $2 female $3; else print}”
通過(guò) system()這個(gè)函數(shù),我們就可以在 c 中調(diào)用以上腳本,完成任務(wù)了。
但是,system() 用起來(lái)還是覺(jué)得不爽,它的不足是只能執(zhí)行腳本,卻無(wú)法獲得腳本的輸出數(shù)據(jù),而這通常是我們進(jìn)一步處理的數(shù)據(jù)來(lái)源。(在shell和perl中,可以通過(guò)反引號(hào)( `` )來(lái)取得命令的輸出結(jié)果)。一個(gè)解決辦法是把輸出結(jié)果重定向到一個(gè)臨時(shí)文件中,然后在c中讀取文件,獲取數(shù)據(jù),最后當(dāng)然還要?jiǎng)h除這個(gè)文件。不過(guò),這個(gè)方法總是讓人覺(jué)得有一點(diǎn)點(diǎn)不爽,如果能直接把腳本執(zhí)行中輸出的數(shù)據(jù)輸?shù)轿覀兊木彌_區(qū)來(lái)就更好了。
我寫(xiě)了個(gè)小函數(shù),叫 my_system(),通過(guò)管道以及重定向,實(shí)現(xiàn)了以上想法。函數(shù)原型如下:
int my_system(const char* pCmd, char* pResult, int size);
輸出數(shù)據(jù)被保存到 pResult所指向的緩沖區(qū)中,緩沖區(qū)大小為 size,最多可以保存 size-1的數(shù)據(jù)。
函數(shù)的實(shí)現(xiàn)放在本文的最后
有了這個(gè)函數(shù)以后,在 c中調(diào)用腳本就更方便了,我們可以通過(guò)它來(lái)實(shí)現(xiàn)對(duì) user.conf的查詢(xún)。
4)、查詢(xún)一個(gè)記錄
例如,我們要獲取 tom 的性別
可以用腳本這樣來(lái)實(shí)現(xiàn):
cat user.conf | awk ‘/^tom[[:blank:]] / {print $3}“
腳本的執(zhí)行結(jié)果是 tom的性別 male被輸出到屏幕上
在我們的 c程序中,如此調(diào)用 my_system(),
char buf[101];
my_system(“cat user.conf | awk ‘/^tom[[:blank:]] / {print $3}””, buf, 101);
調(diào)用完以后,buf中的數(shù)據(jù)就是 “male”了,怎么樣,還算方便吧?
以上只是用結(jié)合腳本完成了一個(gè)比較簡(jiǎn)單的任務(wù),所以我沒(méi)有把這些腳本單獨(dú)形成腳本文件。如果你善于使用 perl、shell、awk,那么可以寫(xiě)出更強(qiáng)大的腳本文件來(lái)處理更復(fù)雜的問(wèn)題,然后通過(guò)類(lèi)似 my_system( )的方法,在 c/c 等其它語(yǔ)言中取得腳本的輸出結(jié)果,實(shí)現(xiàn)有趣的“混合編程”。
希望你能從中得到樂(lè)趣!
#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 error\n”);
return –1;
}
if((pid = fork()) == 0)
{// chile process
int fd2[2];
if(pipe(fd2))
{
printf(“pipe2 error\n”);
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 is\n\n%s\n”, result);
return 0;
}
責(zé)任編輯:Ct
評(píng)論
查看更多