«
如何用PHP实现通过Web执行C/C++程序

时间:2008-5-31    作者:Deri    分类: 分享


   <p>  一、简介</p><p>  如果你对Unix/Linux有所了解的话,你应该知道他们大都自带了C和C++的编译器,分别是GCC和G++。Unix在程序安装及Make等许多地方使用到了这些编译器。利用一些控制台命令,C++和PHP。我将向你介绍怎样生成一个完整的C++程序例子,他可以在用PHP程序来执行,并能获得相应的输出结果。</p><p>  我将先生成C++程序代码,并编译它,然后讨论我们将如果通过使用PHP的函数passthru来执行这个程序。从某种意义上来说,这边文章给我们提供一种通过Web页面来访问一般程序的方法。</p><p>  为了能更好的理解这篇文章,你应该有一台运行着apache和最新版本php的unix/Linux服务器。同时也应该掌握C++, unix控制台命令,当然一些PHP的编程经验也是必需的。</p><p>  二、编写一个C++程序</p><p>  例如,我们可以写一个能够通过命令行还接收参数的C++简单程序,并命名为Sampleapp.然后我们能够按照下面的方式给他传递三个不同的参数 :</p><code>Sampleapp ?参数一 ?参数二 ?参数三</code></p><p>  这个程序的功能是能输出传递给他的参数的个数和每个参数的值,然后我们可以用PHP脚本程序来执行编译好的C++程序。</p><p>  利用你习惯的文本编辑器,新建一个名为Sampleapp.cpp的文件,再此文件中输入如下的代码:</p><code>#include <iostream.h><br />int main(int argc, char* argv[])<br />{<br />cout << endl << "You passed "<br /><< argc-1 << " arguement"<br />  << (argc-1 == 1 ? "" : "s")<br /><< "." << endl;<br />  cout << (argc-1 == 1 ? "This" : "These")<br />  << " arguement"<br /><< (argc-1 == 1 ? "" : "s")<br /><< " "<br />  << (argc-1 == 1 ? "is" : "are") << ": "<br /><< endl << endl;<br />  for(int i = 1; i < argc; i++)<br />  cout << "[" << i << "] "<br /><< argv[i] << endl;<br />  return 0;<br />  }</code></p>
<p> </p>

   <p>  这个C++程序包含的程序的入口点:main(),main()函数带了两个参数:argc(命令行传入参数的个数)和argv(一个包含了所传参数实际值的字符型指针数组)。这个两个参数能被C++编译器自动捕获。</p><code>cout << endl << "You passed " << argc-1<br /><< " arguement"<br />  << (argc-1 == 1 ? "" : "s")<br /><< "." << endl;<br />;</code></p><p>  这句话的意思是获得从执行命令行传入的参数的个数。Argv这个字符型指针数组是从0开始检索的,它至少包含一个实际的值(即本程序的路径和名称),这个值由C++编译器自动地附加上去。条件操作符”?”是用来判断命令行传入地参数是否多于一个。例如,如果命令行过传入两个参数,我们地程序将输出如下信息:</p><code>You passed 2 arguments.<br />cout << (argc-1 == 1 ?<br />"This" : "These")<br />  << " arguement"<br /><< (argc-1 == 1 ? "" : "s")<br /><< " "<br />  << (argc-1 == 1 ? "is" : "are")<br /><< ": " << endl << endl;</code></p><p>  接下来,我们同样用条件操作符来输出另一句话。不过要记住,即使我们不从程序执行命令行传入任何参数,main函数地argv[]参数也包含一个值。同样地,如果我们从命令行传入两个参数给程序,程序将输出如下地信息:</p><code>These arguments are:<br />for(int i = 1;<br />i < argc; i++)<br />  cout << "[" << i << "] "<br /><< argv[i] << endl;</code></p>
 <p> </p>

   <p>  最后,main函数逐一的输出命令行传入的每个参数,它用到了一个简单的for(;;)循环语句,这个函数能根据参数的个数将参数值一个一个的输出。假如我们传给程序两个参数”first”和second”, for循环输出的结果如下:</p><code>[1] ?first<br />[2] ?second</code></p><p>  以上是关于这个C++程序的简单说明,它的功能十分简单,就是将命令行传入的参数用cout函数显示在输出屏幕上。</p><p>  接下来,我们将编译这个.cpp文件,如果你在windows平台下,需要telnet到所使用的server上。在这里,我们使用大多Unix机器上都提供的G++编译器来编译这个源文件。不过为了确信你的机器安装了G++,你可以输入如下命令:which g++。如果G++已经安装了,Unix shell将显示出G++所在的全路径。如果没有安装,它将提示你说”command couldn’t be found”。 你可以在这里下载到G++。在源文件所在的目录输入如下G++命令:</p><code>g++ -c sampleapp.cpp.</code></p><p>  通过这个命令,我们就将.cpp文件编译成了包含机器代码的目标文件。通过 ls ?a命令,你可以发现在本目录下出现了一个新文件sampleapp.o,这就是.cpp源文件被编译成机器码的结果。不过我们最终想要的是一个可执行文件,因为我们还要输入如下的G++命令:</p><code>g++ sampleapp.cpp ?o sampleapp</code></p><p>  这样我们就获得了一个名为sampleapp的可执行文件。不过注意的是,Unix下的可执行文件跟Windows不一样,它没有任何后缀。下面我们可以来检验一下程序执行的结果,如果如下命令:</p><code>sampleapp one -two /three</code></p><p>  我们可以看到如下的执行结果:</p><code>You passed 3 arguments.<br />  These arguments are:<br />[1] one<br />[2] ?two<br />[3] /three</code></p>
 <p> </p>

   <p>  现在,可执行的C++程序成生完毕,下面我们将生成一个能够通过web浏览器来访问这个程序的PHP教本程序。</p><p>  三,生成PHP脚本程序</p><p>  为了能通过Internet来调用我们的C++程序,我们需要生成一个PHP脚本程序。这个PHP脚本程序将有一个Form表单,以便用户能输入可以传给程序Sampleapp的参数。PHP脚本的代码太长就不在这里全部列出了,需要的话可以通过下面的地址来下载它。(Php code)</p><code>if(@$submit)<br />  {<br />  }<br />  else<br />  {<br />  }</code></p><p>  首先,脚本程序检查看变量$submit是否有值,这个变量$submit的值是程序后面的Form表单提交后传递过来的,它缺省为空值。符号@的作用是当变量$submit的值不存在的时忽略相关的错误信息。</p><p>  由于变量$submit缺省为空,所以一开始执行else{}中的代码,它在浏览器上简单的显示一个Form表单。Form的action属性设为变量$PHP_SELF,即表单提交后返回本页。同时Form表单包含了一个文本输入条,这是用来让用户输入要传递给C++程序的命令行参数。Form如下图所示:</p><p>  一旦我们输入执行命令并提交表单,变量$submit(即按钮Go的名字)就获得一个值,这样PHP教本将执行if{}之间的代码。</p><code>if($args == "")<br />echo "<h1>You didn't enter<br />any arguments.</h1>";<br />  else<br />  {<br />  echo "<h1>SampleApp Result</h1>";<br />  $command = "/htdocs/sampleapp " .<br />escapeshellcmd($args);<br />  passthru($command);<br />  }</code></p><p>  变量$args是自动产生的,它的值是Form表单中文本输入条传过来的值。如果没有输入任何信息,程序将简单的告诉用户没有输入任何值。</p>
 <p> </p>

   <p>  如果用户输入任何非空的信息,程序将把text域的值,即变量$args传给C++程序。下面这段代码就是执行C++的程序的执行命令:</p><code>$command = "/htdocs/sampleapp "<br />. escapeshellcmd($args);</code></p><p>  函数eacapeshellcmd是用来当做安全检查工具,以过滤调一些如”,”,””和””等的特殊字符。这可以防止一些用户企图输入某些字符来调用系统内部命令。</p><p>  例如,如果你在Form表单的text域中输入”1 ?two /three”,那么变量$command的值就为:</p><code>/htdocs/sampleapp 1 ?two /three</code></p><p>  你能发现我们定义了程序sampleapp的全路径,在这个例子中,程序文件位于/htdocs目录下。你可以根据的自己程序所在的目录做相应的修改。</p><code>passthru($command);</code></p><p>  最后,我们使用PHP的函数passthru来执行变量$command所包含的命令并且将原始的执行结果输出到浏览器上。</p><p>  w在本文即将结束之前,几个可能碰到的问题我想说一下。首先,当你执行sampleapp.php教本程序的时候,如果你没有看到程序的任何输出信息,或许是开了安全模式。如果这样,系统将不会允许PHP脚本来执行系统内部程序。</p><p>  关于如何关闭安全模式,请访问网页http://www.php.net/manual/en/features.safe-mode.php,上面有详细的介绍。其次,在一些Unix系统上,PHP函数passthru不能将内部程序的输出传递给浏览页面,如果发生这种情况,可以用system函数来代替passthru函数。</p><p>  四,结论</p><p>  从本的例子可以看出,Unix操作系统非常强大,并且PHP允许开发者通过脚本以独立的线程来执行系统内部程序。本文的所给的例子非常的简单,但是只要再多花一点功夫,你可以写一个能更新Mysql数据库的c++程序,运行其他系统命令的程序或者是操作系统文件/目录结构的程序。但是,不管怎样,你都应该确保你的系统安全,绝对不能让任何其他的脚本程序随意访问系统内部程序。</p>