为什么要进行CGI编程?
在HTML中,当客户填写了表单,并按下了发送(submit)按钮后,表单的内容被发送到了服务器端,一般的,这时就需要有一个服务器端脚本来对表单的内容进行一些处理,或者是把它们保存起来,或者是按内容进行一些查询,或者是一些别的什么。没有了CGI,WEB的世界就完全失去了它的交互性,所有的信息都变成单向的了,而不能够有任何的反馈。
有的人认为可以用JavaScript来代替CGI程序,这其实是一个概念上的错误。JavaScript只能够在客户浏览器中运行,而CGI却是工作在服务器上的。他们所做的工作有一些交集,比如表单数据验证一类的,但是JavaScript是绝对无法取代CGI的。但可以这样说,如果一项工作即能够用JavaScript来做,又可以用CGI来做,那么绝对要使用JavaScript,在执行的速度上,JavaScript比CGI有着先天的优势。只有那些在客户端解决不了的问题,比如和某个远程数据库交互,这时就应该使用CGI了。
简单的说来,CGI是用来沟通HTML表单和服务器端程序的接口(interface)。说它是接口,也就是说CGI并不是一种语言,而是可以被其他语言所应用的一个规范集。理论上讲,你可以用任何的程序语言来编写CGI程序,只要在编程的时候符合CGI规范所定义的一些东西就可以了。由于C语言在平台无关性上表现不错(几乎在任何的系统平台下都有其相应编译器),而且对大多数程序员而言都算得上很熟悉(不像Perl),因此,C是CGI编程的首选语言之一。这儿我们介绍的,就是如何使用C来编写CGI程序。
作为CGI编程的最为简单的例子,就是进行表单的处理。因而在这篇文章中,我们主要介绍的就是如何用C来编写CGI程序来进行表但处理。
GET表单的处理
对于那些使用了属性“METHOD=GET”的表单(或者没有METHOD属性,这时候GET是其缺省值),CGI定义为:当表单被发送到服务器断后,表单中的数据被保存在服务器上一个叫做QUERY_STRING的环境变量中。这种表单的处理相对简单,只要读取环境变量就可以了。这一点对不同的语言有不同的做法。在C语言中,你可以用库函数getenv(定义在标准库函数stdlib中)来把环境变量的值作为一个字符串来存取。你可以在取得了字符串中的数据后,运用一些小技巧进行类型的转换,这都是比较简单的了。在CGI程序中的标准输出(output)(比如在C中的stdout文件流)也是经过重定义了的。它并没有在服务器上产生任何的输出内容,而是被重定向到客户浏览器。这样,如果编写一个C的CGI程序的时候,把一个HTML文档输出到它的stdout上,这个HTML文档会被在客户端的浏览器中显示出来。这也是CGI程序的一个基本原理。
我们来看看具体的程序实现,下面是一段HTML表单:
< FORM ACTION="/cgi-bin/mult.cgi" >
< P >请在下面填入乘数和被乘数,按下确定后可以看到结果。
< INPUT NAME="m" SIZE="5" >
< INPUT NAME="n" SIZE="5" >< BR >
< INPUT TYPE="SUBMIT" VALUE="确定" >
< /FORM >
我们要实现的功能很简单,就是把表单中输入的数值乘起来,然后输出结果。其实这个功能完全可以用JavaScript来实现,但为了让程序尽量的简单易懂,我还是选择了这个小小的乘法来作为示例。
下面就是处理这个表单的CGI程序,对应于FORM标签中的ACTION属性值。
#include < stdio.h >
#include < stdlib.h >
int main(void)
{
char *data;
long m,n;
printf("%s%c%c ","Content-Type:text/html;charset=gb2312",13,10);
printf("< TITLE >乘法结果< /TITLE > ");
printf("< H3 >乘法结果< /H3 > ");
data = getenv("QUERY_STRING");
if(data == NULL)
printf("< P >错误!数据没有被输入或者数据传输有问题");
else if(sscanf(data,"m=%ld&n=%ld",&m,&n)!=2)
printf("< P >错误!输入数据非法。表单中输入的必须是数字。");
else
printf("< P >%ld和%ld的成绩是:%ld。",m,n,m*n);
return 0;
}
具体的C语法就不多讲了,我们来看看它作为CGI程序所特殊的地方。
前面已经提到标准输出的内容就是要被显示在浏览器中的内容。第一行的输出内容是必须的,也是一个CGI程序所特有的:printf("%s%c%c ","Content-Type:text/html",13,10),这个输出是作为HTML的文件头。因为CGI不仅可以像浏览器输出HTML文本,而且可以输出图像,声音之类的东西。这一行告诉浏览器如何处理接受到的内容。在Content-Type的定义后面跟有两行的空行,这也是不可缺少的。因为所有CGI程序的头部输出都是相近的,因而可以为其定义一个函数,来节省编程的时间。这是CGI编程常用的一个技巧。
程序在后面调用了用了库函数getevn来得到QUERY_STRING的内容,然后使用sscanf函数把每个参数值取出来,要注意的是sscanf函数的用法。其他的就没有什么了,和一般的C程序没有区别。
把程序编译后,改名为mult.cgi放在/cgi-bin/目录下面,就可以被表单调用了。这样,一个处理GET方式表单的CGI程序就大功告成了。
POST表单处理
下面我们来考虑另外一种表单传送方法:POST。假设我们要实现的任务是这样的:把表单中客户输入的一段文本内容添加到服务器上的一个文本文件的后面。这可以看作是一个留言版程序的雏形。显然,这个工作是无法用JavaScript这种客户端脚本来实现,也算得上真正意义上的CGI程序了。
看起来这个问题和上面讲的内容很相近,仅仅是用不同的表单和不同的脚本(程序)而已。但实际上,这中间是有一些区别的。在上面的例子中,GET的处理方法可以看作是“纯查询(pure query)”类型的,也就是说,它与状态无关。同样的数据可以被提交任意的次数,而不会引起任何的问题(除了服务器的一些小小的开销)。但是现在的任务就不同了,至少它要改变一个文件的内容。因而,可以说它是与状态有关的。这也算是POST和GET的区别之一。而且,GET对于表单的长度是有限制的,而POST则不然,这也是在这个任务中选用POST方法的主要原因。但相对的,对GET的处理速度就要比POST快一些。
在CGI的定义中,对于POST类型的表单,其内容被送到CGI程序的标准输入(在C语言中是stdin),而被传送的长度被放在环境变量CONTENT_LENGTH中。因而我们要做的就是,在标准输入中读入CONTENT_LENGTH长度的字符串。从标准输出读入数据听起来似乎要比从环境变量中读数据来的要容易一些,其实则不然,有一些细节地方要注意,这在下面的程序中可以看到。特别要注意的一点就是:CGI程序和一般的程序有所不同,一般的程序在读完了一个文件流的内容之后,会得到一个EOF的标志。但在CGI程序的表单处理过程中,EOF是永远不会出现的,所以千万不要读多于CONTENT_LENGTH长度的字符,否这会有什么后果,谁也不知道(CGI规范中没有定义,一般根据服务器不同而有不同得处理方法)。
我们来看看到底如何从POST表单收集数据到CGI程序,下面給出了一個比较简单的C源代碼:
#include < stdio.h >
#include < stdlib.h >
#define MAXLEN 80
#define EXTRA 5
/* 4个字节留给字段的名字"data", 1个字节留给"=" */
#define MAXINPUT MAXLEN+EXTRA+2
/* 1个字节留给换行符,还有一个留给后面的NULL */
#define DATAFILE "../data/data.txt"
/* 要被添加数据的文件 */
void unencode(char *src, char *last, char *dest)
{
for(; src != last; src++, dest++)
if(*src == "+")
*dest = " ";
else if(*src == "%") {
int code;
if(sscanf(src+1, "%2x", &code) != 1) code = "?";
*dest = code;
src +=2; }
else
*dest = *src;
*dest = " ";
*++dest = "";
}
int main(void)
{
char *lenstr;
char input[MAXINPUT], data[MAXINPUT];
long len;
printf("%s%c%c ",
"Content-Type:text/html;charset=gb2312",13,10);
printf("< TITLE >Response< /TITLE > ");
lenstr = getenv("CONTENT_LENGTH");
if(lenstr == NULL || sscanf(lenstr,"%ld",&len)!=1 || len > MAXLEN)
printf("< P >表单提交错误");
else {
FILE *f;
fgets(input, len+1, stdin);
unencode(input+EXTRA, input+len, data);
f = fopen(DATAFILE, "a");
if(f == NULL)
printf("< P >对不起,意外错误,不能够保存你的数据 ");
else
fputs(data, f);
fclose(f);
printf("< P >非常感谢,您的数据已经被保存< BR >%s",data);
}
return 0;
}
从本质上来看,程序先从CONTENT_LENGTH环境变量中得到数据的字长,然后读取相应长度的字符串。因为数据内容在传输的过程中是经过了编码的,所以必须进行相应的解码。编码的规则很简单,主要的有这几条:
1. 表单中每个每个字段用字段名后跟等号,再接上上这个字段的值来表示,每个字段之间的内容用&连结;
2. 所有的空格符号用加号代替,所以在编码码段中出现空格是非法的;
3. 特殊的字符比如标点符号,和一些有特定意义的字符如“+”,用百分号后跟其对应的ACSII码值来表示。
例如:如果用户输入的是:
Hello there!
那么数据传送到服务器的时候经过编码,就变成了data=Hello+there%21 上面的unencode()函数就是用来把编码后的数据进行解码的。在解码完成后,数据被添加到data.txt文件的尾部,并在浏览其中回显出来。
把文件编译完成后,把它改名为collect.cgi后放在CGI目录中就可以被表单调用了。下面给出了其相应的表单:
< FORM ACTION="/cgi-bin/collect.cgi" METHOD="POST" >
< P >请输入您的留言(最多80个字符):< BR >< INPUT NAME="data" SIZE="60" MAXLENGTH="80" >< BR >
< INPUT TYPE="SUBMIT" VALUE="确定" >
< /FORM >
事实上,这个程序只能作为例子,是不能够正式的使用的。它漏掉了很关键的一个问题:当有多个用户同时像文件写入数据是,肯定会有错误发生。而对于一个这样的程序而言,文件被同时写入的几率是很大的。因此,在比较正式的留言版程序中,都需要做一些更多的考虑,比如加入一个信号量,或者是借助于一个钥匙文件等。因为那只是编程的技巧问题,在这儿就不多说了。
最后,我们来写一个浏览data.txt文件的的CGI程序,这只需要把内容输出到stdout就可以了:
#include < stdio.h >
#include < stdlib.h >
#define DATAFILE "../data/data.txt"
int main(void)
{
FILE *f = fopen(DATAFILE,"r");
int ch;
if(f == NULL) {
printf("%s%c%c ",
"Content-Type:text/html;charset=gb2312",13,10);
printf("< TITLE >错误 < /TITLE > ");
printf("< P >< EM >意外错误,无法打开文件< /EM >"); }
else {
printf("%s%c%c ",
"Content-Type:text/plain",13,10);
while((ch=getc(f)) != EOF)
putchar(ch);
fclose(f); }
return 0;
}
这个程序唯一要注意的是:它并没有把data.txt 包装成HTML格式后再输出,而是直接作为简单文本(plain text)输出,这只要在输出的头部用text/plain类型代替text/html就可以了,浏览器会根据Content-Type的类型自动的选择相应的处理方法。
要触发这个程序也很简单,因为没有数据要输入,所以只需一个按钮就可以搞定了:
< FORM ACTION="/cgi-bin/viewdata.cgi" >
< P >< INPUT TYPE="SUBMIT" VALUE="察看" >
< /FORM >
到这儿,一些基本的用C编写CGI程序的原理就将完了。当然,就凭讲的这些内容,还很难编写出一个好的CGI程序,这需要进一步的学习CGI的规范定义,以及一些其他的CGI编程特有的技巧。
这篇文章的目的,也就是要你了解一下CGI编程的概念。事实上,现在的一些主流的服务器端脚本编程语言如ASP,PHP,JSP等,都基本上具备了CGI编程的大部分的功能,但他们在使用上的,确实是比无论用什么语言进行CGI编程都要容易的多。所以在进行服务器端编程的时候,一般都会首先考虑使用这些脚本编程语言。只有当他们也解决不了,比如要进行一些更为底层的编程的时候,才会用到CGI。
|
相关推荐
C语言因其平台无关性和广泛使用,常被用于编写CGI程序。 在HTML表单中,有两种提交方式:GET和POST。GET方式的表单数据会被编码并附加到URL后面,通过服务器的QUERY_STRING环境变量传递。CGI程序可以通过调用`...
JavaScript可以在客户端执行,进行表单验证等任务,但它无法直接与远程数据库交互,而这是CGI的主要应用场景之一。 CGI并不是一种特定的编程语言,而是一种规范,允许任何语言编写的程序作为Web服务器的接口。在...
在本文中,我们将深入探讨使用C语言进行CGI编程的基本概念和实践。 首先,为什么需要CGI编程?HTML表单提交后,通常需要在服务器端进行处理,如存储数据、查询信息或执行其他业务逻辑。CGI为这种服务器端交互提供了...
本篇文章将深入探讨C语言与CGI结合在Linux下的开发实践,以及`cgilib-2.01-bin`这个库的特点和使用方法。 首先,理解CGI的基本原理是至关重要的。CGI允许Web服务器调用外部程序(如用C语言编写的应用),处理来自...
### C语言开发CGI初步了解 #### CGI概念与工作原理 CGI,全称为Common Gateway Interface,即通用网关接口,是一种用于HTML表单与...通过本篇介绍,初学者可以开始探索C语言CGI的世界,逐步构建更复杂的Web应用程序。
CGIC是C语言的一个库,它简化了在C语言中编写CGI程序的过程。本教程适合初学者,旨在引导他们通过CGIC库进行Web开发。 1. **基础知识** - **使用CGIC的基本思路**:CGIC库提供了一系列的函数,用于处理HTTP请求,...
文档列举了多篇相关的参考文献,这些文献涉及了基于Web的C语言考试系统设计、C语言考试系统的开发与实现等多个方面。通过分析这些文献,可以了解在C语言机考模式研究中,已有的工作、存在的问题以及未来可能的研究...
- 使用`arm-linux-gcc -o test.cgi test.c`命令编译C语言源文件`test.c`,生成名为`test.cgi`的可执行文件。这里使用了`arm-linux-gcc`编译器,说明程序是针对ARM架构的Linux系统编译的。 3. **调试步骤**: - **...
CGIC(Common Gateway Interface in C)是一个C语言编写的CGI库,它简化了用C语言编写CGI程序的过程。这篇教程的目标是教会读者如何使用CGIC库来满足Web开发的需求。 ### 基础知识 #### 1. 使用CGIC的基本思路 ...
在CGI(Common Gateway Interface)编程中,cgic库是一个常用的C语言库,它简化了处理Web表单数据的过程。本篇文章将详细讲解cgic库的一些核心函数及其用途。 1. `cgiFormString`: 此函数用于从HTTP请求的表单数据...
CGIC简明教程是针对初学者的一系列指导,旨在教授如何使用C语言的CGI库——CGIC来开发Web应用程序。CGIC库提供了一个简洁的接口,使得在C语言中编写CGI(通用网关接口)程序变得更加简单。这篇教程分为基础知识和...
描述中提到的链接指向了ITEYE博客上的一篇文章,虽然描述为空,但我们可以推测这篇文章可能详细讨论了如何使用C语言编写Web服务器库,或者是关于Lighttpd的使用、配置或开发经验。遗憾的是,由于无法直接访问该链接...
本篇文档将介绍Linux CGI编程的基础知识、常用标签、编程方法和重要的技术细节。 首先,CGI是服务器端的一种接口标准,它定义了Web服务器如何向扩展的应用程序(如脚本)传递客户端的请求,以及应用程序如何将处理...
本篇文章将深入探讨BOA中的CGI(通用网关接口)程序和CGI可执行脚本的实现,以及BOA如何处理汉字和其他编码问题。 首先,CGI是Web服务器提供的一种机制,使得服务器能够调用外部程序来处理客户端请求并返回结果。在...
本篇主要探讨电子商务应用系统实现的概览,以及传统程序语言与公共网关接口CGI在电子商务中的应用。 3.3.1 电子商务应用系统实现概述 电子商务系统是基于Web的分布式结构,它将客户端和服务器端的开发工作相结合。...
本篇文章将深入探讨如何使用C语言编写一个基础的Web服务器,结合“MyWebServer.zip”这个压缩包中的资源文件,我们将分析其基本结构、工作原理以及实现的关键技术。 首先,我们要理解C语言在开发Web服务器中的优势...
随着互联网的发展,动态Web应用的需求增加,C语言和CGI(通用网关接口)成为实现动态功能的主要手段,但其复杂性和高昂的开发成本限制了其普及。1996年之后,微软推出的COM(组件对象模型)和ASP(活动服务器页面)...
开发动态网站需要使用C语言或CGI(Common Gateway Interface),这导致开发成本高且难度大。 - **两层架构的问题**:早期网站采用两层架构,即表示层和数据访问层,这种架构限制了应用的扩展性和集成性。 #### 四、...