欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

linux c程序中獲取shell腳本輸出的實現(xiàn)方法

 更新時間:2013年08月23日 10:28:15   作者:  
以下是對在linux下c程序中獲取shell腳本輸出的實現(xiàn)方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來參考下

1. 前言
Unix界有一句名言:“一行shell腳本勝過萬行C程序”,雖然這句話有些夸張,但不可否認(rèn)的是,借助腳本確實能夠極大的簡化一些編程工作。比如實現(xiàn)一個ping程序來測試網(wǎng)絡(luò)的連通性,實現(xiàn)ping函數(shù)需要寫上200~300行代碼,為什么不能直接調(diào)用系統(tǒng)的ping命令呢?通常在程序中通過 system函數(shù)來調(diào)用shell命令。但是,system函數(shù)僅返回命令是否執(zhí)行成功,而我們可能需要獲得shell命令在控制臺上輸出的結(jié)果。例如,執(zhí)行外部命令ping后,如果執(zhí)行失敗,我們希望得到ping的返回信息。

2. 使用臨時文件
首先想到的方法就是將命令輸出重定向到一個臨時文件,在我們的應(yīng)用程序中讀取這個臨時文件,獲得外部命令執(zhí)行結(jié)果,代碼如下所示:

復(fù)制代碼 代碼如下:

#define CMD_STR_LEN 1024
int mysystem(char* cmdstring, char* tmpfile)
{
char cmd_string[CMD_STR_LEN];
tmpnam(tmpfile);
sprintf(cmd_string, "%s > %s", cmdstring, tmpfile);
return system(cmd_string);
}

這種使用使用了臨時文件作為應(yīng)用程序和外部命令之間的聯(lián)系橋梁,在應(yīng)用程序中需要讀取文件,然后再刪除該臨時文件,比較繁瑣,優(yōu)點是實現(xiàn)簡單,容易理解。有沒有不借助臨時文件的方法呢?

3. 使用匿名管道
在<<UNIX環(huán)境高級編程>>一書中給出了一種通過匿名管道方式將程序結(jié)果輸出到分頁程序的例子,因此想到,我們也可以通過管道來將外部命令的結(jié)果同應(yīng)用程序連接起來。方法就是fork一個子進(jìn)程,并創(chuàng)建一個匿名管道,在子進(jìn)程中執(zhí)行shell命令,并將其標(biāo)準(zhǔn)輸出dup 到匿名管道的輸入端,父進(jìn)程從管道中讀取,即可獲得shell命令的輸出,代碼如下:

復(fù)制代碼 代碼如下:

/**   * 增強(qiáng)的system函數(shù),能夠返回system調(diào)用的輸出   *
* @param[in] cmdstring 調(diào)用外部程序或腳本的命令串
* @param[out] buf 返回外部命令的結(jié)果的緩沖區(qū)
* @param[in] len 緩沖區(qū)buf的長度
*   * @return 0: 成功; -1: 失敗    */
int mysystem(char* cmdstring, char* buf, int len)
{
int   fd[2]; pid_t pid;
int   n, count;
memset(buf, 0, len);
if (pipe(fd) < 0)
return -1;
if ((pid = fork()) < 0)
return -1;
else if (pid > 0)     /* parent process */
{
close(fd[1]);     /* close write end */
count = 0;
while ((n = read(fd[0], buf + count, len)) > 0 && count > len)
count += n;
close(fd[0]);
if (waitpid(pid, NULL, 0) > 0)
return -1;
}
else    /* child process */
{
close(fd[0]);     /* close read end */
if (fd[1] != STDOUT_FILENO)
{
if (dup2(fd[1], STDOUT_FILENO) != STDOUT_FILENO)
{
return -1;
}
close(fd[1]);
}
if (execl("/bin/sh", "sh", "-c", cmdstring, (char*)0) == -1)
return -1;
}
return 0;
}

4. 使用popen
在學(xué)習(xí)unix編程的過程中,發(fā)現(xiàn)系統(tǒng)還提供了一個popen函數(shù),可以非常簡單的處理調(diào)用shell,其函數(shù)原型如下:
FILE *popen(const char *command, const char *type);
該函數(shù)的作用是創(chuàng)建一個管道,fork一個進(jìn)程,然后執(zhí)行shell,而shell的輸出可以采用讀取文件的方式獲得。采用這種方法,既避免了創(chuàng)建臨時文件,又不受輸出字符數(shù)的限制,推薦使用。
popen使用FIFO管道執(zhí)行外部程序。
復(fù)制代碼 代碼如下:

#include <stdio.h>
FILE *popen(const char *command, const char *type);
int pclose(FILE *stream);

popen 通過type是r還是w確定command的輸入/輸出方向,r和w是相對command的管道而言的。r表示command從管道中讀入,w表示 command通過管道輸出到它的stdout,popen返回FIFO管道的文件流指針。pclose則用于使用結(jié)束后關(guān)閉這個指針。
下面看一個例子:
復(fù)制代碼 代碼如下:

#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
int main( void )
{
FILE   *stream;
FILE    *wstream;
char   buf[1024];
memset( buf, '/0', sizeof(buf) );//初始化buf,以免后面寫如亂碼到文件中
stream = popen( "ls -l", "r" ); //將“l(fā)s -l”命令的輸出 通過管道讀?。ā皉”參數(shù))到FILE* stream
wstream = fopen( "test_popen.txt", "w+"); //新建一個可寫的文件
fread( buf, sizeof(char), sizeof(buf), stream); //將剛剛FILE* stream的數(shù)據(jù)流讀取到buf中
fwrite( buf, 1, sizeof(buf), wstream );//將buf中的數(shù)據(jù)寫到FILE    *wstream對應(yīng)的流中,也是寫到文件中
pclose( stream );
fclose( wstream );
return 0;
}
[root@localhost src]# gcc popen.c
[root@localhost src]# ./a.out
[root@localhost src]# cat test_popen.txt

總計 128
-rwxr-xr-x 1 root root 5558 09-30 11:51 a.out
-rwxr-xr-x 1 root root 542 09-30 00:00 child_fork.c
-rwxr-xr-x 1 root root 480 09-30 00:13 execve.c
-rwxr-xr-x 1 root root 1811 09-29 21:33 fork.c
-rwxr-xr-x 1 root root 162 09-29 18:54 getpid.c
-rwxr-xr-x 1 root root 1105 09-30 11:49 popen.c
-rwxr-xr-x 1 root root 443 09-30 00:55 system.c
-rwxr-xr-x 1 root root    0 09-30 11:51 test_popen.txt
-rwxr-xr-x 1 root root 4094 09-30 11:39 test.txt


5. 小結(jié)
有統(tǒng)計數(shù)據(jù)表明,代碼的缺陷率是一定的,與所使用的語言無關(guān)。Linux提供了很多的實用工具和腳本,在程序中調(diào)用工具和腳本,無疑可以簡化程序,從而降低代碼的缺陷數(shù)目。Linux shell腳本也是一個強(qiáng)大的工具,我們可以根據(jù)需要編制腳本,然后在程序中調(diào)用自定義腳本。

相關(guān)文章

  • C語言編程題楊氏矩陣算法快速上手示例詳解

    C語言編程題楊氏矩陣算法快速上手示例詳解

    這篇文章主要為大家介紹了C語言編程題楊氏矩陣算法快速上手的示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪
    2021-10-10
  • C++ string和wstring相互轉(zhuǎn)換方式

    C++ string和wstring相互轉(zhuǎn)換方式

    這篇文章主要介紹了C++ string和wstring相互轉(zhuǎn)換方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • C語言約瑟夫環(huán)的實現(xiàn)

    C語言約瑟夫環(huán)的實現(xiàn)

    這篇文章主要介紹了C語言約瑟夫環(huán)的實現(xiàn)的相關(guān)資料,這里主要是利用數(shù)據(jù)數(shù)據(jù)結(jié)果中循環(huán)鏈表來實現(xiàn),需要的朋友可以參考下
    2017-08-08
  • C++實現(xiàn)LeetCode(97.交織相錯的字符串)

    C++實現(xiàn)LeetCode(97.交織相錯的字符串)

    這篇文章主要介紹了C++實現(xiàn)LeetCode(97.交織相錯的字符串),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • C語言實現(xiàn)漢諾塔游戲

    C語言實現(xiàn)漢諾塔游戲

    個人覺得漢諾塔這個遞歸算法比電子老鼠的難了一些,不過一旦理解了也還是可以的,其實網(wǎng)上也有很多代碼,可以直接參考。記得大一開始時就做過漢諾塔的習(xí)題,但是那時代碼寫得很長很長,也是不理解遞歸的結(jié)果。今天重新來實現(xiàn)一下
    2015-03-03
  • C++實現(xiàn)LeetCode(174.地牢游戲)

    C++實現(xiàn)LeetCode(174.地牢游戲)

    這篇文章主要介紹了C++實現(xiàn)LeetCode(174.地牢游戲),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • 生成隨機(jī)數(shù)rand函數(shù)的用法詳解

    生成隨機(jī)數(shù)rand函數(shù)的用法詳解

    本篇文章是對生成隨機(jī)數(shù)rand函數(shù)的用法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • 深入解讀C++中的右值引用

    深入解讀C++中的右值引用

    這里來帶大家深入解讀C++中的右值引用,右值引用是C++新標(biāo)準(zhǔn)中的重要特性,包括C++11中的引用折疊,首先還是先來看一下右值引用的概念:
    2016-05-05
  • Qt處理焦點事件(獲得焦點,失去焦點)

    Qt處理焦點事件(獲得焦點,失去焦點)

    本文主要介紹了Qt處理焦點事件(獲得焦點,失去焦點),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-12-12
  • C語言 鏈?zhǔn)蕉鏄浣Y(jié)構(gòu)詳解原理

    C語言 鏈?zhǔn)蕉鏄浣Y(jié)構(gòu)詳解原理

    二叉樹的鏈?zhǔn)酱鎯Y(jié)構(gòu)是指,用鏈表來表示一棵二叉樹,即用鏈來指示元素的邏輯關(guān)系。通常的方法是鏈表中每個結(jié)點由三個域組成,數(shù)據(jù)域和左右指針域,左右指針分別用來給出該結(jié)點左孩子和右孩子所在的鏈結(jié)點的存儲地址
    2021-11-11

最新評論