一、簡介
如果你對Unix/Linux有所了解的話,你應該知道他們大都自帶了C和C++的編譯器,分別是GCC和G++。Unix在程序安裝及Make等許多地方使用到了這些編譯器。利用一些控制台命令,C++和PHP。我將向你介紹怎樣生成一個完整的C++程序例子,他可以在用PHP程序來執行,並能獲得相應的輸出結果。
我將先生成C++程序代碼,並編譯它,然後討論我們將如果通過使用PHP的函數passthru來執行這個程序。從某種意義上來說,這邊文章給我們提供一種通過Web頁面來訪問一般程序的方法。
為了能更好的理解這篇文章,你應該有一台運行著apache和最新版本php的unix/Linux服務器。同時也應該掌握C++, unix控制台命令,當然一些PHP的編程經驗也是必需的。
二、編寫一個C++程序
例如,我們可以寫一個能夠通過命令行還接收參數的C++簡單程序,並命名為Sampleapp.然後我們能夠按照下面的方式給他傳遞三個不同的參數 :
Sampleapp ?參數一 ?參數二 ?參數三
這個程序的功能是能輸出傳遞給他的參數的個數和每個參數的值,然後我們可以用PHP腳本程序來執行編譯好的C++程序。
利用你習慣的文本編輯器,新建一個名為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;
}
這個C++程序包含的程序的入口點:main(),main()函數帶了兩個參數:argc(命令行傳入參數的個數)和argv(一個包含了所傳參數實際值的字符型指針數組)。這個兩個參數能被C++編譯器自動捕獲。
cout << endl << "You passed " << argc-1
<< " arguement"
<< (argc-1 == 1 ? "" : "s")
<< "." << endl;
;
這句話的意思是獲得從執行命令行傳入的參數的個數。Argv這個字符型指針數組是從0開始檢索的,它至少包含一個實際的值(即本程序的路徑和名稱),這個值由C++編譯器自動地附加上去。條件操作符”?”是用來判斷命令行傳入地參數是否多於一個。例如,如果命令行過傳入兩個參數,我們地程序將輸出如下信息:
You passed 2 arguments.
cout << (argc-1 == 1 ?
"This" : "These")
<< " arguement"
<< (argc-1 == 1 ? "" : "s")
<< " "
<< (argc-1 == 1 ? "is" : "are")
<< ": " << endl << endl;
接下來,我們同樣用條件操作符來輸出另一句話。不過要記住,即使我們不從程序執行命令行傳入任何參數,main函數地argv[]參數也包含一個值。同樣地,如果我們從命令行傳入兩個參數給程序,程序將輸出如下地信息:
These arguments are:
for(int i = 1;
i < argc; i++)
cout << "[" << i << "] "
<< argv[i] << endl;
最後,main函數逐一的輸出命令行傳入的每個參數,它用到了一個簡單的for(;;)循環語句,這個函數能根據參數的個數將參數值一個一個的輸出。假如我們傳給程序兩個參數”first”和second”, for循環輸出的結果如下:
>[1] ?first
[2] ?second
以上是關於這個C++程序的簡單說明,它的功能十分簡單,就是將命令行傳入的參數用cout函數顯示在輸出屏幕上。
接下來,我們將編譯這個.cpp文件,如果你在windows平台下,需要telnet到所使用的server上。在這裡,我們使用大多Unix機器上都提供的G++編譯器來編譯這個源文件。不過為了確信你的機器安裝了G++,你可以輸入如下命令:which g++。如果G++已經安裝了,Unix shell將顯示出G++所在的全路徑。如果沒有安裝,它將提示你說”command couldn’t be found”。 你可以在這裡下載到G++。在源文件所在的目錄輸入如下G++命令:
g++ -c sampleapp.cpp.
通過這個命令,我們就將.cpp文件編譯成了包含機器代碼的目標文件。通過 ls ?a命令,你可以發現在本目錄下出現了一個新文件sampleapp.o,這就是.cpp源文件被編譯成機器碼的結果。不過我們最終想要的是一個可執行文件,因為我們還要輸入如下的G++命令:
g++ sampleapp.cpp ?o sampleapp
這樣我們就獲得了一個名為sampleapp的可執行文件。不過注意的是,Unix下的可執行文件跟Windows不一樣,它沒有任何後綴。下面我們可以來檢驗一下程序執行的結果,如果如下命令:
sampleapp one -two /three
我們可以看到如下的執行結果:
You passed 3 arguments.
These arguments are:
[1] one
[2] ?two
[3] /three
現在,可執行的C++程序成生完畢,下面我們將生成一個能夠通過web浏覽器來訪問這個程序的PHP教本程序。
三,生成PHP腳本程序
為了能通過Internet來調用我們的C++程序,我們需要生成一個PHP腳本程序。這個PHP腳本程序將有一個Form表單,以便用戶能輸入可以傳給程序Sampleapp的參數。PHP腳本的代碼太長就不在這裡全部列出了,需要的話可以通過下面的地址來下載它。(Php code)
if(@$submit)
{
}
else
{
}
首先,腳本程序檢查看變量$submit是否有值,這個變量$submit的值是程序後面的Form表單提交後傳遞過來的,它缺省為空值。符號@的作用是當變量$submit的值不存在的時忽略相關的錯誤信息。
由於變量$submit缺省為空,所以一開始執行else{}中的代碼,它在浏覽器上簡單的顯示一個Form表單。Form的action屬性設為變量$PHP_SELF,即表單提交後返回本頁。同時Form表單包含了一個文本輸入條,這是用來讓用戶輸入要傳遞給C++程序的命令行參數。Form如下圖所示:
一旦我們輸入執行命令並提交表單,變量$submit(即按鈕Go的名字)就獲得一個值,這樣PHP教本將執行if{}之間的代碼。
if($args == "")
echo "<h1>You didnt enter
any arguments.</h1>";
else
{
echo "<h1>SampleApp Result</h1>";
$command = "/htdocs/sampleapp " .
escapeshellcmd($args);
passthru($command);
}
變量$args是自動產生的,它的值是Form表單中文本輸入條傳過來的值。如果沒有輸入任何信息,程序將簡單的告訴用戶沒有輸入任何值。
如果用戶輸入任何非空的信息,程序將把text域的值,即變量$args傳給C++程序。下面這段代碼就是執行C++的程序的執行命令:
$command = "/htdocs/sampleapp "
. escapeshellcmd($args);
函數eacapeshellcmd是用來當做安全檢查工具,以過濾調一些如”,”,””和””等的特殊字符。這可以防止一些用戶企圖輸入某些字符來調用系統內部命令。
例如,如果你在Form表單的text域中輸入”1 ?two /three”,那麼變量$command的值就為:
/htdocs/sampleapp 1 ?two /three
你能發現我們定義了程序sampleapp的全路徑,在這個例子中,程序文件位於/htdocs目錄下。你可以根據的自己程序所在的目錄做相應的修改。
passthru($command);
最後,我們使用PHP的函數passthru來執行變量$command所包含的命令並且將原始的執行結果輸出到浏覽器上。
w在本文即將結束之前,幾個可能碰到的問題我想說一下。首先,當你執行sampleapp.php教本程序的時候,如果你沒有看到程序的任何輸出信息,或許是開了安全模式。如果這樣,系統將不會允許PHP腳本來執行系統內部程序。
關於如何關閉安全模式,請訪問網頁http://www.php.net/manual/en/features.safe-mode.php,上面有詳細的介紹。其次,在一些Unix系統上,PHP函數passthru不能將內部程序的輸出傳遞給浏覽頁面,如果發生這種情況,可以用system函數來代替passthru函數。
四,結論
從本的例子可以看出,Unix操作系統非常強大,並且PHP允許開發者通過腳本以獨立的線程來執行系統內部程序。本文的所給的例子非常的簡單,但是只要再多花一點功夫,你可以寫一個能更新Mysql數據庫的c++程序,運行其他系統命令的程序或者是操作系統文件/目錄結構的程序。但是,不管怎樣,你都應該確保你的系統安全,絕對不能讓任何其他的腳本程序隨意訪問系統內部程序。