C++ Web编程
-
什么是CGI?
通用网关接口(CGI)是一组标准,这些标准定义了Web服务器和自定义脚本之间如何交换信息。CGI规范目前由NCSA维护,NCSA定义CGI如下-- 通用网关接口(CGI)是外部网关程序与信息服务器(例如HTTP服务器)接口的标准。
- 当前版本是CGI/1.1,CGI/1.2正在开发中。
-
网页浏览
为了理解CGI的概念,让我们看看单击超链接浏览特定网页或URL时会发生什么。- 您的浏览器联系HTTP Web服务器并要求URL,即。文件名。
- Web服务器将解析URL并查找文件名。如果找到请求的文件,则Web服务器将该文件发送回浏览器,否则发送一条错误消息,指示您请求了错误的文件。
- Web浏览器从Web服务器获取响应,并根据收到的响应显示收到的文件或错误消息。
但是,可以通过以下方式设置HTTP服务器:无论何时请求某个目录中的文件,都不会发回该文件。而是将其作为程序执行,并将程序产生的输出发送回浏览器进行显示。通用网关接口(CGI)是一种标准协议,用于使应用程序(称为CGI程序或CGI脚本)能够与Web服务器和客户端进行交互。这些CGI程序可以用Python,PERL,Shell,C或C++等编写。 -
CGI架构图
以下简单程序显示了CGI的简单架构- -
Web服务器配置
在继续进行CGI编程之前,请确保您的Web服务器支持CGI,并且已将其配置为处理CGI程序。HTTP服务器要执行的所有CGI程序都保存在预先配置的目录中。该目录称为CGI目录,按照约定,其命名为/var/www/cgi-bin。按照惯例,CGI文件的扩展名为.cgi,尽管它们是C++可执行文件。默认情况下,Apache Web Server配置为在/var/www/cgi-bin中运行CGI程序。如果要指定任何其他目录来运行CGI脚本,则可以在httpd.conf文件中修改以下部分-<Directory "/var/www/cgi-bin"> AllowOverride None Options ExecCGI Order allow,deny Allow from all </Directory> <Directory "/var/www/cgi-bin"> Options All </Directory>
在这里,我假设您已经成功启动并运行了Web Server,并且能够运行任何其他CGI程序,例如Perl或Shell等。 -
第一个CGI程序
考虑以下C++程序内容-#include <iostream> using namespace std; int main () { cout << "Content-type:text/html\r\n\r\n"; cout << "<html>\n"; cout << "<head>\n"; cout << "<title>Hello World - First CGI Program</title>\n"; cout << "</head>\n"; cout << "<body>\n"; cout << "<h2>Hello World! This is my first CGI program</h2>\n"; cout << "</body>\n"; cout << "</html>\n"; return 0; }
编译以上代码,并将可执行文件命名为cplusplus.cgi。该文件被保存在/var/www/cgi-bin目录中,并且具有以下内容。在运行CGI程序之前,请确保使用chmod 755 cplusplus.cgi UNIX命令使文件具有更改模式让其具备执行权限,以使文件可执行。上面的C++程序是一个简单的程序,它将其输出写入STDOUT文件(即屏幕)。有一个重要的额外功能可用,即第一行打印Content-type:text/html\r\n\r\n。该行发送回浏览器,并指定要在浏览器屏幕上显示的内容类型。现在您必须已经了解了CGI的基本概念,并且可以使用Python编写许多复杂的CGI程序。C++ CGI程序可以与任何其他外部系统(例如RDBMS)进行交互以交换信息。 -
HTTP头
行content-type:text/html\r\n\r\n
是HTTP头的一部分,发送到浏览器以理解内容。所有的HTTP头将在以下形式−Content-type: text/html\r\n\r\n
其他几个重要的HTTP标头,在CGI编程中将经常使用。头 描述 Content-type: 定义返回文件格式的MIME字符串。例子是的content-type:text/html。 Expires: Date 信息失效的日期。浏览器应该使用它来决定何时需要刷新页面。有效的日期字符串的格式应该是1998年1月01日12:00:00 GMT。 Location: URL 应该返回的URL,而不是请求的URL。您可以使用此文件将请求重定向到任何文件。 Last-modified: Date 资源最后一次修改的日期。 Content-length: N 返回的数据的长度(以字节为单位)。浏览器使用这个值来报告一个文件的估计下载时间。 Set-Cookie: String 设置通过字符串传递的cookie。 -
CGI环境变量
所有CGI程序都可以访问以下环境变量。这些变量在编写任何CGI程序时都起着重要作用。环境变量 描述 CONTENT_TYPE 内容的数据类型,当客户端向服务器发送附加内容时使用。例如文件上传等。 CONTENT_LENGTH 仅对POST请求可用的查询信息的长度。 HTTP_COOKIE 以键和值对的形式返回设置好的cookie。 HTTP_USER_AGENT 用户代理请求头字段包含有关发出请求的用户代理的信息。它是web浏览器的名称。 PATH_INFO CGI脚本的路径。 QUERY_STRING 与GET方法请求一起发送的url编码信息。 REMOTE_ADDR 发出请求的远程主机的IP地址。这对于日志记录或身份验证非常有用。 REMOTE_HOST 发出请求的主机的完全限定名。如果此信息不可用,则可以使用REMOTE_ADDR获取IR地址。 REQUEST_METHOD 用于发出请求的方法。最常见的方法是GET和POST。 SCRIPT_FILENAME CGI脚本的完整路径。 SCRIPT_NAME CGI脚本的名称。 SERVER_NAME 服务器的主机名或IP地址。 SERVER_SOFTWARE 服务器正在运行的软件的名称和版本 这是一个小的CGI程序,列出了所有CGI变量。#include <iostream> #include <stdlib.h> using namespace std; const string ENV[ 24 ] = { "COMSPEC", "DOCUMENT_ROOT", "GATEWAY_INTERFACE", "HTTP_ACCEPT", "HTTP_ACCEPT_ENCODING", "HTTP_ACCEPT_LANGUAGE", "HTTP_CONNECTION", "HTTP_HOST", "HTTP_USER_AGENT", "PATH", "QUERY_STRING", "REMOTE_ADDR", "REMOTE_PORT", "REQUEST_METHOD", "REQUEST_URI", "SCRIPT_FILENAME", "SCRIPT_NAME", "SERVER_ADDR", "SERVER_ADMIN", "SERVER_NAME","SERVER_PORT","SERVER_PROTOCOL", "SERVER_SIGNATURE","SERVER_SOFTWARE" }; int main () { cout << "Content-type:text/html\r\n\r\n"; cout << "<html>\n"; cout << "<head>\n"; cout << "<title>CGI Environment Variables</title>\n"; cout << "</head>\n"; cout << "<body>\n"; cout << "<table border = \"0\" cellspacing = \"2\">"; for ( int i = 0; i < 24; i++ ) { cout << "<tr><td>" << ENV[ i ] << "</td><td>"; // attempt to retrieve value of environment variable char *value = getenv( ENV[ i ].c_str() ); if ( value != 0 ) { cout << value; } else { cout << "Environment variable does not exist."; } cout << "</td></tr>\n"; } cout << "</table><\n"; cout << "</body>\n"; cout << "</html>\n"; return 0; }
-
C++ CGI库
对于真实的示例,您将需要通过CGI程序执行许多操作。有一个为C++程序编写的CGI库,您可以从ftp://ftp.gnu.org/gnu/cgicc/下载,并按照以下步骤安装该库-$tar xzf cgicc-X.X.X.tar.gz $cd cgicc-X.X.X/ $./configure --prefix=/usr $make $make install
您可以在“C++ CGI库文档” 中查看可用的相关文档。
-
GET和POST方法
当您需要将一些信息从浏览器传递到Web服务器,最后传递给CGI程序时,您肯定遇到过很多情况。浏览器最经常使用两种方法将此信息传递到Web服务器。这些方法是GET方法和POST方法。GET方法发送附加到页面请求的编码用户信息。页面和编码信息由?分隔。字符如下-http://www.test.com/cgi-bin/cpp.cgi?key1=value1&key2=value2
GET方法是将信息从浏览器传递到web服务器的默认方法,它会产生一个长字符串,出现在浏览器的Location:box中。如果您有密码或其他敏感信息要传递给服务器,请不要使用GET方法。GET方法有大小限制,您可以在请求字符串中传递最多1024个字符。当使用GET方法时,信息通过QUERY_STRING http头传递,并将在您的CGI程序中通过QUERY_STRING环境变量访问。您可以通过将键和值对与任何URL连接起来来传递信息,也可以使用HTML<form>标记使用GET方法传递信息。 -
简单的URL示例:GET方法
这是一个简单的URL,它将使用GET方法将两个值传递给hello_get.py程序。/cgi-bin/cpp_get.cgi?first_name=ZARA&last_name=ALI
下面是一个程序,用于生成cpp_get.cgi CGI程序来处理Web浏览器给出的输入。我们将使用C++ CGI库,它使访问传递的信息变得非常容易-#include <iostream> #include <vector> #include <string> #include <stdio.h> #include <stdlib.h> #include <cgicc/CgiDefs.h> #include <cgicc/Cgicc.h> #include <cgicc/HTTPHTMLHeader.h> #include <cgicc/HTMLClasses.h> using namespace std; using namespace cgicc; int main () { Cgicc formData; cout << "Content-type:text/html\r\n\r\n"; cout << "<html>\n"; cout << "<head>\n"; cout << "<title>Using GET and POST Methods</title>\n"; cout << "</head>\n"; cout << "<body>\n"; form_iterator fi = formData.getElement("first_name"); if( !fi->isEmpty() && fi != (*formData).end()) { cout << "First name: " << **fi << endl; } else { cout << "No text entered for first name" << endl; } cout << "<br/>\n"; fi = formData.getElement("last_name"); if( !fi->isEmpty() &&fi != (*formData).end()) { cout << "Last name: " << **fi << endl; } else { cout << "No text entered for last name" << endl; } cout << "<br/>\n"; cout << "</body>\n"; cout << "</html>\n"; return 0; }
现在,如下编译上述程序:$g++ -o cpp_get.cgi cpp_get.cpp -lcgicc
生成cpp_get.cgi并将其放在您的CGI目录中,然后尝试使用以下链接进行访问-/cgi-bin/cpp_get.cgi?first_name=ZARA&last_name=ALI
这将产生以下结果-First name: ZARA Last name: ALI
-
简单的FORM示例:GET方法
这是一个简单的示例,该示例使用HTML FORM和Submit按钮传递两个值。我们将使用相同的CGI脚本cpp_get.cgi来处理此输入。<form action = "/cgi-bin/cpp_get.cgi" method = "get"> First Name: <input type = "text" name = "first_name"> <br /> Last Name: <input type = "text" name = "last_name" /> <input type = "submit" value = "Submit" /> </form>
-
使用POST方法传递信息
将信息传递到CGI程序的一种通常更可靠的方法是POST方法。这将以与GET方法完全相同的方式打包信息,但不是在?之后将其作为文本字符串发送。在网址中将其作为单独的消息发送。该消息以标准输入的形式进入CGI脚本。相同的cpp_get.cgi程序也将处理POST方法。让我们以与上述相同的示例为例,它使用HTML FORM和Submit按钮传递两个值,但是这次使用POST方法,如下所示:<form action = "/cgi-bin/cpp_get.cgi" method = "post"> First Name: <input type = "text" name = "first_name"><br /> Last Name: <input type = "text" name = "last_name" /> <input type = "submit" value = "Submit" /> </form>
-
将复选框数据传递到CGI程序
当需要选择多个选项时,将使用复选框。 这是带有两个复选框的表单的示例HTML代码-<form action = "/cgi-bin/cpp_checkbox.cgi" method = "POST" target = "_blank"> <input type = "checkbox" name = "maths" value = "on" /> Maths <input type = "checkbox" name = "physics" value = "on" /> Physics <input type = "submit" value = "Select Subject" /> </form>
下面是C++程序,它将生成cpp_checkbox.cgi脚本来处理Web浏览器通过复选框按钮提供的输入。#include <iostream> #include <vector> #include <string> #include <stdio.h> #include <stdlib.h> #include <cgicc/CgiDefs.h> #include <cgicc/Cgicc.h> #include <cgicc/HTTPHTMLHeader.h> #include <cgicc/HTMLClasses.h> using namespace std; using namespace cgicc; int main () { Cgicc formData; bool maths_flag, physics_flag; cout << "Content-type:text/html\r\n\r\n"; cout << "<html>\n"; cout << "<head>\n"; cout << "<title>Checkbox Data to CGI</title>\n"; cout << "</head>\n"; cout << "<body>\n"; maths_flag = formData.queryCheckbox("maths"); if( maths_flag ) { cout << "Maths Flag: ON " << endl; } else { cout << "Maths Flag: OFF " << endl; } cout << "<br/>\n"; physics_flag = formData.queryCheckbox("physics"); if( physics_flag ) { cout << "Physics Flag: ON " << endl; } else { cout << "Physics Flag: OFF " << endl; } cout << "<br/>\n"; cout << "</body>\n"; cout << "</html>\n"; return 0; }
-
将单选按钮数据传递到CGI程序
当只需要选择一个选项时,使用单选按钮。 这是带有两个单选按钮的表单的示例HTML代码-<form action = "/cgi-bin/cpp_radiobutton.cgi" method = "post" target = "_blank"> <input type = "radio" name = "subject" value = "maths" checked = "checked"/> Maths <input type = "radio" name = "subject" value = "physics" /> Physics <input type = "submit" value = "Select Subject" /> </form>
下面是C++程序,它将生成cpp_radiobutton.cgi脚本来处理Web浏览器通过单选按钮给出的输入。#include <iostream> #include <vector> #include <string> #include <stdio.h> #include <stdlib.h> #include <cgicc/CgiDefs.h> #include <cgicc/Cgicc.h> #include <cgicc/HTTPHTMLHeader.h> #include <cgicc/HTMLClasses.h> using namespace std; using namespace cgicc; int main () { Cgicc formData; cout << "Content-type:text/html\r\n\r\n"; cout << "<html>\n"; cout << "<head>\n"; cout << "<title>Radio Button Data to CGI</title>\n"; cout << "</head>\n"; cout << "<body>\n"; form_iterator fi = formData.getElement("subject"); if( !fi->isEmpty() && fi != (*formData).end()) { cout << "Radio box selected: " << **fi << endl; } cout << "<br/>\n"; cout << "</body>\n"; cout << "</html>\n"; return 0; }