一,簡(jiǎn)介
如果你對(duì)Unix/Linux有所了解的話,你應(yīng)該知道他們大都自帶了C和C++的編譯器,分別是GCC和G++。Unix在程序安裝及Make等許多地方使用到了這些編譯器。利用一些控制臺(tái)命令,C++和PHP, 我將向你介紹怎樣生成一個(gè)完整的C++程序例子,他可以在用PHP程序來執(zhí)行,并能獲得相應(yīng)的輸出結(jié)果。我將先生成C++程序代碼,并編譯它,談后討論我們將如果通過使用PHP的函數(shù)passthru來執(zhí)行這個(gè)程序。從某種意義上來說,這邊文章給我們提供一種通過Web頁(yè)面來訪問一般程序的方法。
為了能更好的理解這篇文章,你應(yīng)該有一臺(tái)運(yùn)行著apache和最新版本php的unix/Linux服務(wù)器。同時(shí)也應(yīng)該掌握C++, unix控制臺(tái)命令,當(dāng)然一些PHP的編程經(jīng)驗(yàn)也是必需的。
二,編寫一個(gè)C++程序
例如,我們可以寫一個(gè)能夠通過命令行還接收參數(shù)的C++ 簡(jiǎn)單程序,并命名為Sampleapp.然后我們能夠按照下面的方式給他傳遞三個(gè)不同的參數(shù) :
Sampleapp –參數(shù)一 –參數(shù)二 –參數(shù)三
這個(gè)程序的功能是能輸出傳遞給他的參數(shù)的個(gè)數(shù)和每個(gè)參數(shù)的值,然后我們可以用PHP腳本程序來執(zhí)行編譯好的C++程序。
利用你習(xí)慣的文本編輯器,新建一個(gè)名為Sampleapp.cpp 的文件,再此文件中輸入如下的代碼:
#include <iostream.h>
int main(int argc, char* argv[])
{
cout << endl << "You passed " << argc-1 << " arguement"
<< (argc-1 == 1 ? "" : "s") << "." << endl;
cout << (argc-1 == 1 ? "This" : "These")
<< " arguement" << (argc-1 == 1 ? "" : "s") << " "
<< (argc-1 == 1 ? "is" : "are") << ": " << endl << endl;
for(int i = 1; i < argc; i++)
cout << "[" << i << "] " << argv[i] << endl;
return 0;
}
這個(gè)C++程序包含的程序的入口點(diǎn):main(),main()函數(shù)帶了兩個(gè)參數(shù):argc(命令行傳入?yún)?shù)的個(gè)數(shù))和argv(一個(gè)包含了所傳參數(shù)實(shí)際值的字符型指針數(shù)組)。這個(gè)兩個(gè)參數(shù)能被C++編譯器自動(dòng)捕獲。
cout << endl << "You passed " << argc-1 << " arguement"
<< (argc-1 == 1 ? "" : "s") << "." << endl;;
這句話的意思是獲得從執(zhí)行命令行傳入的參數(shù)的個(gè)數(shù)。Argv這個(gè)字符型指針數(shù)組是從0開始檢索的,它至少包含一個(gè)實(shí)際的值(即本程序的路徑和名稱),這個(gè)值由C++編譯器自動(dòng)地附加上去。條件操作符”?”是用來判斷命令行傳入地參數(shù)是否多于一個(gè)。例如,如果命令行過傳入兩個(gè)參數(shù),我們地程序?qū)⑤敵鋈缦滦畔?
You passed 2 arguments.
cout << (argc-1 == 1 ? "This" : "These")
<< " arguement" << (argc-1 == 1 ? "" : "s") << " "
<< (argc-1 == 1 ? "is" : "are") << ": " << endl << endl;
接下來,我們同樣用條件操作符來輸出另一句話。不過要記住,即使我們不從程序執(zhí)行命令行傳入任何參數(shù),main函數(shù)地argv[]參數(shù)也包含一個(gè)值。同樣地,如果我們從命令行傳入兩個(gè)參數(shù)給程序,程序?qū)⑤敵鋈缦碌匦畔ⅲ?/p>
These arguments are:
for(int i = 1; i < argc; i++)
cout << "[" << i << "] " << argv[i] << endl;
最后,main函數(shù)逐一的輸出命令行傳入的每個(gè)參數(shù),它用到了一個(gè)簡(jiǎn)單的for(;;)循環(huán)語(yǔ)句,這個(gè)函數(shù)能根據(jù)參數(shù)的個(gè)數(shù)將參數(shù)值一個(gè)一個(gè)的輸出。假如我們傳給程序兩個(gè)參數(shù)”first”和second”, for循環(huán)輸出的結(jié)果如下:
[1] –first
[2] –second
以上是關(guān)于這個(gè)C++程序的簡(jiǎn)單說明,它的功能十分簡(jiǎn)單,就是將命令行傳入的參數(shù)用cout函數(shù)顯示在輸出屏幕上。
接下來,我們將編譯這個(gè).cpp文件,如果你在windows平臺(tái)下,需要telnet到所使用的server上。在這里,我們使用大多Unix機(jī)器上都提供的G++編譯器來編譯這個(gè)源文件。不過為了確信你的機(jī)器安裝了G++,你可以輸入如下命令:which g++。如果G++已經(jīng)安裝了,Unix shell將顯示出G++所在的全路徑。如果沒有安裝,它將提示你說”command couldn’t be found”. 你可以在這里下載到G++.
在源文件所在的目錄輸入如下G++命令:
g++ -c sampleapp.cpp.
通過這個(gè)命令,我們就將.cpp文件編譯成了包含機(jī)器代碼的目標(biāo)文件。通過 ls –a命令,你可以發(fā)現(xiàn)在本目錄下出現(xiàn)了一個(gè)新文件sampleapp.o,這就是.cpp源文件被編譯成機(jī)器碼的結(jié)果。不過我們最終想要的是一個(gè)可執(zhí)行文件,因?yàn)槲覀冞€要輸入如下的G++命令:
g++ sampleapp.cpp –o sampleapp
這樣我們就獲得了一個(gè)名為sampleapp的可執(zhí)行文件。不過注意的是,Unix下的可執(zhí)行文件跟Windows不一樣,它沒有任何后綴。
下面我們可以來檢驗(yàn)一下程序執(zhí)行的結(jié)果,如果如下命令:
sampleapp one -two /three
我們可以看到如下的執(zhí)行結(jié)果:
You passed 3 arguments.
These arguments are:
[1] one
[2] –two
[3] /three
現(xiàn)在,可執(zhí)行的C++程序成生完畢,下面我們將生成一個(gè)能夠通過 web瀏覽器來訪問這個(gè)程序的PHP教本程序。
三,生成PHP腳本程序
為了能通過Internet來調(diào)用我們的C++程序,我們需要生成一個(gè)PHP腳本程序。這個(gè)PHP腳本程序?qū)⒂幸粋€(gè)Form表單,以便用戶能輸入可以傳給程序Sampleapp的參數(shù)。PHP腳本的代碼太長(zhǎng)就不在這里全部列出了,需要的話可以通過下面的地址來下載它。(Php code)
if(@$submit)
{
}
else
{
}
首先,腳本程序檢查看變量$submit是否有值,這個(gè)變量$submit的值是程序后面的Form表單提交后傳遞過來的,它缺省為空值。符號(hào)@的作用是當(dāng)變量$submit的值不存在的時(shí)忽略相關(guān)的錯(cuò)誤信息。
由于變量$submit缺省為空,所以一開始執(zhí)行else{}中的代碼,它在瀏覽器上簡(jiǎn)單的顯示一個(gè)Form表單。Form的action屬性設(shè)為變量$PHP_SELF,即表單提交后返回本頁(yè)。同時(shí)Form表單包含了一個(gè)文本輸入條,這是用來讓用戶輸入要傳遞給C++程序的命令行參數(shù)。Form如下圖所示:
一旦我們輸入執(zhí)行命令并提交表單,變量$submit(即按鈕Go的名字)就獲得一個(gè)值,這樣PHP教本將執(zhí)行if{}之間的代碼。
if($args == "")
echo "<h1>You didn‘t enter any arguments.</h1>";
else
{
echo "<h1>SampleApp Result</h1>";
$command = "/htdocs/sampleapp " . escapeshellcmd($args);
passthru($command);
}
變量$args是自動(dòng)產(chǎn)生的,它的值是Form表單中文本輸入條傳過來的值。如果沒有輸入任何信息,程序?qū)⒑?jiǎn)單的告訴用戶沒有輸入任何值。
如果用戶輸入任何非空的信息,程序?qū)裻ext域的值,即變量$args傳給C++程序。下面這段代碼就是執(zhí)行C++的程序的執(zhí)行命令:
$command = "/htdocs/sampleapp " . escapeshellcmd($args);
函數(shù)eacapeshellcmd是用來當(dāng)做安全檢查工具,以過濾調(diào)一些如”,”,”\”和”\\”等的特殊字符。這可以防止一些用戶企圖輸入某些字符來調(diào)用系統(tǒng)內(nèi)部命令。
例如,如果你在Form表單的text域中輸入”1 –two /three”,那么變量$command的值就為: /htdocs/sampleapp 1 –two /three
你能發(fā)現(xiàn)我們定義了程序sampleapp的全路徑,在這個(gè)例子中,程序文件位于/htdocs目錄下。你可以根據(jù)的自己程序所在的目錄做相應(yīng)的修改。
passthru($command);
最后,我們使用PHP的函數(shù)passthru來執(zhí)行變量$command所包含的命令并且將原始的執(zhí)行結(jié)果輸出到瀏覽器上。在我的服務(wù)器上,返回結(jié)果的HTML頁(yè)面如下:
w在本文即將結(jié)束之前,幾個(gè)可能碰到的問題我想說一下。首先,當(dāng)你執(zhí)行sampleapp.php教本程序的時(shí)候,如果你沒有看到程序的任何輸出信息,或許是開了安全模式。如果這樣,系統(tǒng)將不會(huì)允許PHP腳本來執(zhí)行系統(tǒng)內(nèi)部程序。關(guān)于如何關(guān)閉安全模式,請(qǐng)?jiān)L問網(wǎng)頁(yè)http://www.php.net/manual/en/features.safe-mode.php,上面有詳細(xì)的介紹。其次,在一些Unix系統(tǒng)上,PHP函數(shù)passthru不能將內(nèi)部程序的輸出傳遞給瀏覽頁(yè)面,如果發(fā)生這種情況,可以用system函數(shù)來代替passthru函數(shù)。
四,結(jié)論
從本的例子可以看出,Unix操作系統(tǒng)非常強(qiáng)大,并且PHP允許開發(fā)者通過腳本以獨(dú)立的線程來執(zhí)行系統(tǒng)內(nèi)部程序。本文的所給的例子非常的簡(jiǎn)單,但是只要再多花一點(diǎn)功夫,你可以寫一個(gè)能更新Mysql數(shù)據(jù)庫(kù)的c++程序,運(yùn)行其他系統(tǒng)命令的程序或者是操作系統(tǒng)文件/目錄結(jié)構(gòu)的程序。但是,不管怎樣,你都應(yīng)該確保你的系統(tǒng)安全,絕對(duì)不能讓任何其他的腳本程序隨意訪問系統(tǒng)內(nèi)部程序。
如果你想獲得更多關(guān)于本文提到的相關(guān)信息,可以查看下面列出的一些相關(guān)鏈接和圖書。
相關(guān)鏈接:
http://galton.uchicago.edu/~gosset/Compdocs/gcc.html
http://www.andrews.edu/~maier/tutor.html
http://www.php.net/manual/en/function.escapeshellcmd.php
相關(guān)圖書:
Advanced Programming in the Unix Environment
Beginning C++: The Complete Language
Linux Apache Web Server Administration
Professional Apache
聯(lián)系客服