<?xml version="1.0" encoding="UTF-8" ?>
<rss version="2.0">
<channel>
<title><![CDATA[向东博客 专注WEB应用 构架之美 --- 构架之美，在于尽态极妍 | 应用之美，在于药到病除]]></title> 
<link>http://jackxiang.com/index.php</link> 
<description><![CDATA[赢在IT，Playin' with IT,Focus on Killer Application,Marketing Meets Technology.]]></description> 
<language>zh-cn</language> 
<copyright><![CDATA[向东博客 专注WEB应用 构架之美 --- 构架之美，在于尽态极妍 | 应用之美，在于药到病除]]></copyright>
<item>
<link>http://jackxiang.com/post//</link>
<title><![CDATA[[转]CGIC简明教程]]></title> 
<author>jack &lt;xdy108@126.com&gt;</author>
<category><![CDATA[WEB2.0]]></category>
<pubDate>Mon, 31 May 2010 02:07:28 +0000</pubDate> 
<guid>http://jackxiang.com/post//</guid> 
<description>
<![CDATA[ 
	<br/><br/>本系列的目的是演示如何使用C语言的CGI库“CGIC”完成Web开发的各种要求。<br/><br/>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;基础知识<br/>&nbsp;&nbsp;&nbsp;&nbsp;* 1: 使用CGIC的基本思路<br/>&nbsp;&nbsp;&nbsp;&nbsp;* 2: 获取Get请求字符串<br/>&nbsp;&nbsp;&nbsp;&nbsp;* 3: 反转义<br/>&nbsp;&nbsp;&nbsp;&nbsp;* 4: 获取请求中的参数值<br/>&nbsp;&nbsp;&nbsp;&nbsp; 进阶训练<br/>&nbsp;&nbsp;&nbsp;&nbsp;* 用CGIC实现文件上传<br/><br/><br/>CGIC简明教程1：使用CGIC的基本思路<br/><br/>C语言编程是一项复杂且容易出错的工作，所以在完成复杂任务时，一定要选择合适的库。对于用C语言编写CGI程序则更是如此。<br/>CGIC是非常优秀的C语言CGI库函数。 其下载地址为：www.boutell.com/cgic/#obtain，现在的版本号是2.05。<br/>本站从今天开始，将逐步介绍如何使用CGIC完成各种操作，也可以说是一个Tutorial。<br/>（注：本系列涉及的编程环境都是Linux，Windows用户需要对用到的操作系统命令稍作修改）<br/><br/>本文纲要 ：<br/>CGIC的安装、测试安装、使用CGIC的基本思路；<br/>1) CGIC的下载安装<br/><br/>从上面提供的官方网址下载了CGIC库之后，解开压缩包，里面有大约10个文件，有用的是：<br/>cgic.h：头文件；<br/>cgic.c：CGIC的源代码文件；<br/>cgictest.c：CGIC库的作者提供的一个CGI程序例子；<br/>capture.c：用于调试CGI程序的工具；<br/>Makefile：安装CGIC的脚本文件；<br/>可以看到，整个库实际上就是cgic.c一个文件，可以说是非常的精炼。<br/>我们可以把CGIC安装为操作系统的一个动态链接库，这样我们每次编译的时候，就不需要有cgic.c这个源文件了。<br/>但是由于需要（以后将会看到），我们将修改cgic.c代码，所以我们不把它安装进系统。每次编译的时候，只要把cgic.c和cgic.h放到当前文件夹就好了。<br/>2) 测试安装<br/><br/>在开始编写你自己的CGI程序之前，一定要先走通他的例子程序，免得后来程序出错的时候还不知道是配置有问题，还是你的程序代码有问题。<br/>我们用他自带cgictest.c来实现自己的第一个C语言CGI程序。<br/>你可以新建一个工作目录，用于存放你的CGI程序源代码，把cgic.h, cgic.c, cgictest.c三个文件拷贝到这个目录，然后建立一个Makefile文件，其内容为：<br/><br/>&nbsp;&nbsp; 1. test.cgi:cgictest.c cgic.h cgic.c<br/>&nbsp;&nbsp; 2. gcc -wall cgictest.c cgic.c -o test.cgi<br/><br/>需要提醒的是，第二行开头一定是一个tab键（且仅有一个），不能使用空格。<br/>保存好Makefile的内容之后，执行make命令：<br/>make<br/><br/>我们看到，当前目录下应该多了一个test.cgi文件。<br/><br/>在你的网站根目录下建立一个cgi-bin目录（当然名字可以任意取，但作为习惯，一般叫做cgi-bin），然后在Apache的配置文件里赋予其执行 CGI代码的权限，权限修改完之后要重启Apache。完成之后，把刚才生成的test.cgi放到cgi-bin目录中。此时我们可以在浏览器中输入以下地址进行访问：<br/>http://127.0.0.1/cgi-bin/test.cgi<br/><br/>如果正常的话，应该看到一个网页被展示出来。这样，第一个C语言的CGI程序就运行起来了。<br/>如果浏览器报错，那么多半是配置Apache的时候有些操作没有正确完成。<br/>3) 使用CGIC的基本思路<br/><br/>从cgic.c的代码可以看出，它定义了main函数，而在cgictest.c中定义了一个cgiMain函数。也就是说，对于使用CGIC编写的 CGI程序，都是从cgic.c中的代码进入，在库函数完成了一系列必要的操作（比如解析参数、获取系统环境变量）之后，它才会调用你的代码（从你定义的 cgiMain进入）。<br/><br/>另外一点就是，cgi程序输出HTML页面的方式都是使用printf把页面一行一行地打印出来，比如cgictest.c中的这一段代码：<br/>fprintf(cgiOut, &quot;&lt;textarea NAME=&#92;&quot;address&#92;&quot; ROWS=4 COLS=40&gt;&#92;n&quot;);<br/>fprintf(cgiOut, &quot;Default contents go here. &#92;n&quot;);<br/>fprintf(cgiOut, &quot;&lt;/textarea&gt;&#92;n&quot;);<br/><br/>上面这段代码的运行结果就是在页面上输出一个textarea。第一个参数cgiOut实际上就是stdin，所以我们可以直接使用printf，而不必使用fprintf。不过在调试的时候会用到fprintf来重定向输出。<br/>这种方式与Java Servlet非常类似，Servlet也是通过调用打印语句System.out.println(…)来输出一个页面。（不过后来Java推出了JSP来克服这种不便。）<br/>但是与Servlet不同的地方在于，使用C语言的我们还要自己输出HTML头部（声明文档类型）：<br/>cgiHeaderContentType(&quot;text/html&quot;);<br/><br/>这个语句的调用一定要在所有printf语句之前。而这个语句执行的任务实际上就是：<br/>void cgiHeaderContentType(char *mimeType) &#123;<br/>&nbsp;&nbsp;&nbsp;&nbsp;fprintf(cgiOut, &quot;Content-type: %s&#92;r&#92;n&#92;r&#92;n&quot;, mimeType);<br/>&#125;<br/><br/>这个语句告诉浏览器，这次传来的数据是什么类型，是一个HTML文档，还是一个bin文件… 如果是个HTML文档，就通过浏览器窗口显示，如果是一个bin（二进制）文件，则打开下载窗口，让用户选择是否保存文件以及保存文件的路径。<br/><br/>理解了这几点之后，你就可以编写自己的CGIC程序了。新建一个文件test.c试试：<br/>下载: test.c<br/><br/>&nbsp;&nbsp; 1. #include &lt;stdio.h&gt;<br/>&nbsp;&nbsp; 2. #include &quot;cgic.h&quot;<br/>&nbsp;&nbsp; 3. #include &lt;string.h&gt;<br/>&nbsp;&nbsp; 4. #include &lt;stdlib.h&gt;<br/>&nbsp;&nbsp; 5. int cgiMain() &#123;<br/>&nbsp;&nbsp; 6.&nbsp;&nbsp;&nbsp;&nbsp; cgiHeaderContentType(&quot;text/html&quot;);<br/>&nbsp;&nbsp; 7.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;HTML&gt;&lt;HEAD&gt;&#92;n&quot;);<br/>&nbsp;&nbsp; 8.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;TITLE&gt;My First CGI&lt;/TITLE&gt;&lt;/HEAD&gt;&#92;n&quot;);<br/>&nbsp;&nbsp; 9.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;BODY&gt;&lt;H1&gt;Hello CGIC&lt;/H1&gt;&lt;/BODY&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;10.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;/HTML&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;11.&nbsp;&nbsp;&nbsp;&nbsp; return 0;<br/>&nbsp;&nbsp;12. &#125;<br/><br/>把Makefile文件中的cgitest.c全部换称test.c，保存，再执行make命令即可。<br/>此时通过浏览器访问，会在页面上看到一个大大的“Hello CGIC”。<br/><br/><br/>CGIC简明教程2：获取Get请求字符串<br/><br/><br/>Get请求就是我们在浏览器地址栏输入URL时发送请求的方式，或者我们在HTML中定义一个表单（form）时，把action属性设为“Get”时的工作方式；<br/><br/>Get请求字符串就是跟在URL后面以问号“?”开始的字符串，但不包括问号。比如这样的一个请求：<br/>http://127.0.0.1/cgi-bin/out.cgi?ThisIsTheGetString<br/><br/>在上面这个URL中，“ThisIsTheGetString”就是Get请求字符串。<br/><br/>在进入我们自己编写的cgi代码之前，CGIC库已经事先把这个字符串取到了，我们可以在程序中直接获得，要做的仅仅是在你编写的cgiMain方法前面加入以下声明：<br/>extern char *cgiQueryString;<br/><br/>现在给出一个简单的例子，这个例子跟上一篇的测试程序非常相似，只不过程序的输出是使用者输入的Get请求字符串。<br/>下载: test.c<br/><br/>&nbsp;&nbsp; 1. #include &lt;stdio.h&gt;<br/>&nbsp;&nbsp; 2. #include &quot;cgic.h&quot;<br/>&nbsp;&nbsp; 3. #include &lt;string.h&gt;<br/>&nbsp;&nbsp; 4. #include &lt;stdlib.h&gt;<br/>&nbsp;&nbsp; 5. <br/>&nbsp;&nbsp; 6. extern char *cgiQueryString;<br/>&nbsp;&nbsp; 7. int cgiMain() &#123;<br/>&nbsp;&nbsp; 8.&nbsp;&nbsp;&nbsp;&nbsp; cgiHeaderContentType(&quot;text/html&quot;);<br/>&nbsp;&nbsp; 9.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;HTML&gt;&lt;HEAD&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;10.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;TITLE&gt;My CGIC&lt;/TITLE&gt;&lt;/HEAD&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;11.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;BODY&gt;&quot;);<br/>&nbsp;&nbsp;12.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;H1&gt;%s&lt;/H1&gt;&quot;,cgiQueryString);<br/>&nbsp;&nbsp;13.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;/BODY&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;14.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;/HTML&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;15.&nbsp;&nbsp;&nbsp;&nbsp; return 0;<br/>&nbsp;&nbsp;16. &#125;<br/><br/>假设把这个程序编译成out.cgi（编译方法参见上一篇），并部署到Web服务器的cgi-bin目录下，当用户在浏览器地址栏输入本文开头给出的URL字符串时，浏览器页面上会显示：<br/>ThisIsTheGetString<br/><br/>我们也可以编写一个用于测试的HTML页面：<br/>下载: test.html<br/><br/>&nbsp;&nbsp; 1. &lt;html&gt;<br/>&nbsp;&nbsp; 2. &lt;head&gt;<br/>&nbsp;&nbsp; 3.&nbsp;&nbsp;&nbsp;&nbsp; &lt;title&gt;Test&lt;/title&gt;<br/>&nbsp;&nbsp; 4. &lt;/head&gt;<br/>&nbsp;&nbsp; 5. &lt;body&gt;<br/>&nbsp;&nbsp; 6.&nbsp;&nbsp;&nbsp;&nbsp; &lt;form action=&quot;cgi-bin/out.cgi&quot; method=&quot;get&quot;&gt;<br/>&nbsp;&nbsp; 7.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &lt;input type=&quot;text&quot; name=&quot;theText&quot;&gt;<br/>&nbsp;&nbsp; 8.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &lt;input type=&quot;submit&quot; value=&quot;Continue &amp;rarr;&quot;&gt;<br/>&nbsp;&nbsp; 9.&nbsp;&nbsp;&nbsp;&nbsp; &lt;/form&gt;<br/>&nbsp;&nbsp;10. &lt;/body&gt;<br/>&nbsp;&nbsp;11. &lt;/html&gt;<br/><br/>文件的部署结构应该为：<br/>&#124;test.html<br/>&#124;—-cgi-bin/out.cgi<br/><br/>大家可以试试，通过浏览器访问http://127.0.0.1/test.html，在文本框内输入一些字符，并点击提交按钮，然后就可以看到cgi程序的执行结果：把在文本框输入的字符原样显示在浏览器上。<br/><br/><br/><br/>CGIC简明教程3：反转义<br/><br/>浏览器在发送Get请求时，会把请求字符串进行转义操作（英文术语为: escape）； 比如，我们在地址栏输入（注意最后”it’s me”中的空格）:<br/>http://localhost/~Jack/cgi-bin/out.cgi?it&#039;s me<br/><br/>浏览器会把它转义为：<br/>http://localhost/~Jack/cgi-bin/out.cgi?it&#039;s%20me<br/><br/><br/>在上一篇最后给出的例子中，如果在文本框内输入<br/>it&#039;s me<br/><br/>你会发现，浏览器最终发送的请求为<br/>http://localhost/~Jack/cgi-bin/out.cgi?theText=it%27s+me<br/><br/>通过CGIC，我们可以把这些被转义后的字符还原为我们本来的输入，这个过程就叫“反转义” (Unescape)。<br/>不过这个过程有点像hack他的代码。<br/><br/>整个过程分三个步骤：<br/>1）打开cgic.c，找到这一行语句：<br/>static cgiUnescapeResultType cgiUnescapeChars(char **sp, char *cp, int len);<br/><br/>注意，我们要找的只是这个函数声明，不是函数定义；<br/><br/>2）在这个函数声明语句的上方，你会看到一个结构体定义：<br/><br/>&nbsp;&nbsp; 1. typedef enum &#123;<br/>&nbsp;&nbsp; 2.&nbsp;&nbsp;&nbsp;&nbsp; cgiUnescapeSuccess,<br/>&nbsp;&nbsp; 3.&nbsp;&nbsp;&nbsp;&nbsp; cgiUnescapeMemory<br/>&nbsp;&nbsp; 4. &#125; cgiUnescapeResultType;<br/><br/>把这几行语句复制到cgic.h文件中，并在这里把它注释掉；<br/>同时还要删除在第一步中找到的函数声明语句中的“static”关键字。<br/><br/>3）我们现在就可以使用反转义函数cgiUnescapeChars了：<br/>在你自己的代码（按照惯例，还是test.c）中，加入以下声明语句即可<br/>extern cgiUnescapeResultType cgiUnescapeChars(char **sp, char *cp, int len);<br/><br/>接下来我们给出一段完整的test.c代码<br/>下载: test.c<br/><br/>&nbsp;&nbsp; 1. #include &lt;stdio.h&gt;<br/>&nbsp;&nbsp; 2. #include &quot;cgic.h&quot;<br/>&nbsp;&nbsp; 3. #include &lt;string.h&gt;<br/>&nbsp;&nbsp; 4. #include &lt;stdlib.h&gt;<br/>&nbsp;&nbsp; 5. <br/>&nbsp;&nbsp; 6. extern char *cgiQueryString;<br/>&nbsp;&nbsp; 7. extern cgiUnescapeResultType cgiUnescapeChars(char **sp, char *cp, int len);<br/>&nbsp;&nbsp; 8. int cgiMain() &#123;<br/>&nbsp;&nbsp; 9.&nbsp;&nbsp;&nbsp;&nbsp; char * buffer;<br/>&nbsp;&nbsp;10.&nbsp;&nbsp;&nbsp;&nbsp; cgiHeaderContentType(&quot;text/html&quot;);<br/>&nbsp;&nbsp;11.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;HTML&gt;&lt;HEAD&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;12.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;TITLE&gt;My CGI&lt;/TITLE&gt;&lt;/HEAD&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;13.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;BODY&gt;&quot;);<br/>&nbsp;&nbsp;14.&nbsp;&nbsp;&nbsp;&nbsp; cgiUnescapeChars(&amp;buffer, cgiQueryString, strlen(cgiQueryString));<br/>&nbsp;&nbsp;15.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;H1&gt;%s&lt;/H1&gt;&quot;,buffer);<br/>&nbsp;&nbsp;16.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;/BODY&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;17.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;/HTML&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;18.&nbsp;&nbsp;&nbsp;&nbsp; free(buffer);<br/>&nbsp;&nbsp;19.&nbsp;&nbsp;&nbsp;&nbsp; return 0;<br/>&nbsp;&nbsp;20. &#125;<br/><br/>值得注意的是，buffer的存储空间是cgiUnescapeChars帮你分配的，但最后要由你自己来释放（free），这一点千万不可忘记。<br/><br/>下面你可以结合上一篇给出的测试用html代码试试该cgi程序的运行结果，也可以直接在浏览器地址栏输入一些带有特殊符号的字符串。<br/><br/>最后讲一下为什么不得不用这种hacker的方式来完成该任务，而CGIC不显式提供？<br/>CGIC的出发点是，我们平时只需要解析请求中的键值对，比如：”?q=nice&amp;client=IE”，当我们在服务端查询“q”的值时，我们就能得到“nice”。CGIC有一族函数帮助我们完成这个任务，比如cgiFormString（以后会讲到）。在解析这种请求格式的时候，如果我们提供的参数值含有被转义的字符，那么CGIC就会在内部调用cgiUnescapeChars完成反转义。<br/>但是，有时候我们会发送非常复杂的Get请求字符串，但并不是“键－值”对的格式。这就需要直接使用cgiUnescapeChars进行反转义了。<br/>例如：假设我们有个服务端cgi程序chat.cgi，这是个网络聊天机器人（也许你可以开发自己的Web版MSN机器人、QQ机器人）。如果我们发送如下请求：<br/>http://127.0.0.1/cgi-bin/chat.cgi?&quot;this is a cgi user&quot;<br/><br/>那么chat.cgi就会把“this is a cgi user”当做你对它说的话，经过处理，它会回复一段语句。为了方便，我们并没有写成“键－值”对的形式。这个时候被我们hack的cgiUnescapeChars就能派上用场了。<br/><br/><br/>CGIC简明教程4：获取请求中的参数值<br/><br/>我们在提交一个表单(form)时，怎样把表单内的值提取出来呢？<br/>比如下面这个表单：<br/>&lt;form action=&quot;cgi-bin/out.cgi&quot; method=&quot;POST&quot;&gt;<br/>&nbsp;&nbsp;&nbsp;&nbsp;&lt;input type=&quot;text&quot; name=&quot;name&quot; /&gt;<br/>&nbsp;&nbsp;&nbsp;&nbsp;&lt;input type=&quot;text&quot; name=&quot;number&quot; /&gt;<br/>&nbsp;&nbsp;&nbsp;&nbsp;&lt;input type=&quot;submit&quot; value=&quot;Submit&quot; /&gt;<br/>&lt;/form&gt;<br/><br/>当out.cgi收到请求时，需要把输入框”name”和输入框”number”内的值提取出来。而且不管form中的action是GET还是POST，都要有效。<br/><br/>下面给出示例代码：<br/>下载: test.c<br/><br/>&nbsp;&nbsp; 1. #include &lt;stdio.h&gt;<br/>&nbsp;&nbsp; 2. #include &quot;cgic.h&quot;<br/>&nbsp;&nbsp; 3. #include &lt;string.h&gt;<br/>&nbsp;&nbsp; 4. #include &lt;stdlib.h&gt;<br/>&nbsp;&nbsp; 5. <br/>&nbsp;&nbsp; 6. int cgiMain() &#123;<br/>&nbsp;&nbsp; 7.&nbsp;&nbsp;&nbsp;&nbsp; char name[241];<br/>&nbsp;&nbsp; 8.&nbsp;&nbsp;&nbsp;&nbsp; char number[241];<br/>&nbsp;&nbsp; 9.&nbsp;&nbsp;&nbsp;&nbsp; cgiHeaderContentType(&quot;text/html&quot;);<br/>&nbsp;&nbsp;10.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;HTML&gt;&lt;HEAD&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;11.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;TITLE&gt;My CGI&lt;/TITLE&gt;&lt;/HEAD&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;12.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;BODY&gt;&quot;);<br/>&nbsp;&nbsp;13.&nbsp;&nbsp;&nbsp;&nbsp; cgiFormString(&quot;name&quot;, name, 241);<br/>&nbsp;&nbsp;14.&nbsp;&nbsp;&nbsp;&nbsp; cgiFormString(&quot;number&quot;, number, 241);<br/>&nbsp;&nbsp;15.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;H1&gt;%s&lt;/H1&gt;&quot;,name);<br/>&nbsp;&nbsp;16.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;H1&gt;%s&lt;/H1&gt;&quot;,number);<br/>&nbsp;&nbsp;17.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;/BODY&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;18.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(cgiOut, &quot;&lt;/HTML&gt;&#92;n&quot;);<br/>&nbsp;&nbsp;19.&nbsp;&nbsp;&nbsp;&nbsp; return 0;<br/>&nbsp;&nbsp;20. &#125;<br/><br/>从上面的代码可以看出，第13行和第14行获取了输入框的值。<br/><br/>获取输入参数值在CGIC中其实有一族函数，cgiFormString是其中最常用的一个。<br/>cgiFormStringNoNewlines用来去掉换行符（如果用户是在一个TextArea里输入字符的话）；<br/>cgiFormStringSpaceNeeded用于测试输入值的长度，可以以此为依据，然后按需精确分配缓冲区。<br/><br/><br/>用C语言库(CGIC)编写CGI，实现文件上传<br/><br/><br/>用C语言编写cgi程序的话，多半会用到CGIC。这是个非常流行的库，遇到文件上传之类的应用更是离不开它。官方页面及下载地址为：www.boutell.com/cgic/#obtain<br/><br/>不少网站都有文件上传的功能，本文展示如何用CGIC库编写文件上传的服务端程序，最后给出一段简单的HTML代码，供大家测试使用。<br/>下载: upload.c<br/><br/>&nbsp;&nbsp; 1. #include&lt;stdio.h&gt;<br/>&nbsp;&nbsp; 2. #include&lt;string.h&gt;<br/>&nbsp;&nbsp; 3. #include&lt;unistd.h&gt;<br/>&nbsp;&nbsp; 4. #include&lt;fcntl.h&gt;<br/>&nbsp;&nbsp; 5. #include&lt;sys/stat.h&gt;<br/>&nbsp;&nbsp; 6. #include&quot;cgic.h&quot;<br/>&nbsp;&nbsp; 7. #define BufferLen 1024<br/>&nbsp;&nbsp; 8. int cgiMain(void)&#123;<br/>&nbsp;&nbsp; 9.&nbsp;&nbsp;&nbsp;&nbsp; cgiFilePtr file;<br/>&nbsp;&nbsp;10.&nbsp;&nbsp;&nbsp;&nbsp; int targetFile;<br/>&nbsp;&nbsp;11.&nbsp;&nbsp;&nbsp;&nbsp; mode_t mode;<br/>&nbsp;&nbsp;12.&nbsp;&nbsp;&nbsp;&nbsp; char name[128];<br/>&nbsp;&nbsp;13.&nbsp;&nbsp;&nbsp;&nbsp; char fileNameOnServer[64];<br/>&nbsp;&nbsp;14.&nbsp;&nbsp;&nbsp;&nbsp; char contentType[1024];<br/>&nbsp;&nbsp;15.&nbsp;&nbsp;&nbsp;&nbsp; char buffer[BufferLen];<br/>&nbsp;&nbsp;16.&nbsp;&nbsp;&nbsp;&nbsp; char *tmpStr=NULL;<br/>&nbsp;&nbsp;17.&nbsp;&nbsp;&nbsp;&nbsp; int size;<br/>&nbsp;&nbsp;18.&nbsp;&nbsp;&nbsp;&nbsp; int got,t;<br/>&nbsp;&nbsp;19.&nbsp;&nbsp;&nbsp;&nbsp; cgiHeaderContentType(&quot;text/html&quot;);<br/>&nbsp;&nbsp;20.&nbsp;&nbsp;&nbsp;&nbsp; //取得html页面中file元素的值，应该是文件在客户机上的路径名<br/>&nbsp;&nbsp;21.&nbsp;&nbsp;&nbsp;&nbsp; if (cgiFormFileName(&quot;file&quot;, name, sizeof(name)) !=cgiFormSuccess) &#123;<br/>&nbsp;&nbsp;22.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf(stderr,&quot;could not retrieve filename&#92;n&quot;);<br/>&nbsp;&nbsp;23.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; goto FAIL;<br/>&nbsp;&nbsp;24.&nbsp;&nbsp;&nbsp;&nbsp; &#125;<br/>&nbsp;&nbsp;25.&nbsp;&nbsp;&nbsp;&nbsp; cgiFormFileSize(&quot;file&quot;, &amp;size);<br/>&nbsp;&nbsp;26.&nbsp;&nbsp;&nbsp;&nbsp; //取得文件类型，不过本例中并未使用<br/>&nbsp;&nbsp;27.&nbsp;&nbsp;&nbsp;&nbsp; cgiFormFileContentType(&quot;file&quot;, contentType, sizeof(contentType));<br/>&nbsp;&nbsp;28.&nbsp;&nbsp;&nbsp;&nbsp; //目前文件存在于系统临时文件夹中，通常为/tmp，通过该命令打开临时文件。临时文件的名字与用户文件的名字不同，所以不能通过路径/tmp/userfilename的方式获得文件<br/>&nbsp;&nbsp;29.&nbsp;&nbsp;&nbsp;&nbsp; if (cgiFormFileOpen(&quot;file&quot;, &amp;file) != cgiFormSuccess) &#123;<br/>&nbsp;&nbsp;30.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf(stderr,&quot;could not open the file&#92;n&quot;);<br/>&nbsp;&nbsp;31.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; goto FAIL;<br/>&nbsp;&nbsp;32.&nbsp;&nbsp;&nbsp;&nbsp; &#125;<br/>&nbsp;&nbsp;33.&nbsp;&nbsp;&nbsp;&nbsp; t=-1;<br/>&nbsp;&nbsp;34.&nbsp;&nbsp;&nbsp;&nbsp; //从路径名解析出用户文件名<br/>&nbsp;&nbsp;35.&nbsp;&nbsp;&nbsp;&nbsp; while(1)&#123;<br/>&nbsp;&nbsp;36.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; tmpStr=strstr(name+t+1,&quot;&#92;&#92;&quot;);<br/>&nbsp;&nbsp;37.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if(NULL==tmpStr)<br/>&nbsp;&nbsp;38.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; tmpStr=strstr(name+t+1,&quot;/&quot;);//if &quot;&#92;&#92;&quot; is not path separator, try &quot;/&quot;<br/>&nbsp;&nbsp;39.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if(NULL!=tmpStr)<br/>&nbsp;&nbsp;40.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; t=(int)(tmpStr-name);<br/>&nbsp;&nbsp;41.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; else<br/>&nbsp;&nbsp;42.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; break;<br/>&nbsp;&nbsp;43.&nbsp;&nbsp;&nbsp;&nbsp; &#125;<br/>&nbsp;&nbsp;44.&nbsp;&nbsp;&nbsp;&nbsp; strcpy(fileNameOnServer,name+t+1);<br/>&nbsp;&nbsp;45.&nbsp;&nbsp;&nbsp;&nbsp; mode=S_IRWXU&#124;S_IRGRP&#124;S_IROTH;<br/>&nbsp;&nbsp;46.&nbsp;&nbsp;&nbsp;&nbsp; //在当前目录下建立新的文件，第一个参数实际上是路径名，此处的含义是在cgi程序所在的目录（当前目录））建立新文件<br/>&nbsp;&nbsp;47.&nbsp;&nbsp;&nbsp;&nbsp; targetFile=open(fileNameOnServer,O_RDWR&#124;O_CREAT&#124;O_TRUNC&#124;O_APPEND,mode);<br/>&nbsp;&nbsp;48.&nbsp;&nbsp;&nbsp;&nbsp; if(targetFile&lt;0)&#123;<br/>&nbsp;&nbsp;49.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf(stderr,&quot;could not create the new file,%s&#92;n&quot;,fileNameOnServer);<br/>&nbsp;&nbsp;50.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; goto FAIL;<br/>&nbsp;&nbsp;51.&nbsp;&nbsp;&nbsp;&nbsp; &#125;<br/>&nbsp;&nbsp;52.&nbsp;&nbsp;&nbsp;&nbsp; //从系统临时文件中读出文件内容，并放到刚创建的目标文件中<br/>&nbsp;&nbsp;53.&nbsp;&nbsp;&nbsp;&nbsp; while (cgiFormFileRead(file, buffer, BufferLen, &amp;got) ==cgiFormSuccess)&#123;<br/>&nbsp;&nbsp;54.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if(got&gt;0)<br/>&nbsp;&nbsp;55.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; write(targetFile,buffer,got);<br/>&nbsp;&nbsp;56.&nbsp;&nbsp;&nbsp;&nbsp; &#125;<br/>&nbsp;&nbsp;57.&nbsp;&nbsp;&nbsp;&nbsp; cgiFormFileClose(file);<br/>&nbsp;&nbsp;58.&nbsp;&nbsp;&nbsp;&nbsp; close(targetFile);<br/>&nbsp;&nbsp;59.&nbsp;&nbsp;&nbsp;&nbsp; goto END;<br/>&nbsp;&nbsp;60. FAIL:<br/>&nbsp;&nbsp;61.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(stderr,&quot;Failed to upload&quot;);<br/>&nbsp;&nbsp;62.&nbsp;&nbsp;&nbsp;&nbsp; return 1;<br/>&nbsp;&nbsp;63. END:<br/>&nbsp;&nbsp;64.&nbsp;&nbsp;&nbsp;&nbsp; printf(&quot;File &#92;&quot;%s&#92;&quot; has been uploaded&quot;,fileNameOnServer);<br/>&nbsp;&nbsp;65.&nbsp;&nbsp;&nbsp;&nbsp; return 0;<br/>&nbsp;&nbsp;66. &#125;<br/><br/>假设该文件存储为upload.c，则使用如下命令编辑：<br/>gcc -Wall upload.c cgic.c -o upload.cgi<br/><br/>编译完成后把upload.cgi复制到你部署cgi程序的目录（通常命名为cgi-bin）。<br/>正式部署时，请务必修改用open创建新文件那一行代码。把open的第一个参数设置为目标文件在服务器上存储的绝对路径，或者相对于cgi程序的相对路径。本例中，出于简单考虑，在cgi程序所在目录下创建新文件。<br/><br/>测试用HTML代码:<br/>下载: upload.html<br/><br/>&nbsp;&nbsp; 1. &lt;form target=&quot;_blank&quot; method=&quot;post&quot; action=&quot;cgi-bin/upload.cgi&quot;&gt;<br/>&nbsp;&nbsp; 2.&nbsp;&nbsp;&nbsp;&nbsp; &lt;input name=&quot;file&quot; type=&quot;file&quot; /&gt; &lt;input name=&quot;submit&quot; type=&quot;submit&quot; /&gt;<br/>&nbsp;&nbsp; 3. &lt;/form&gt;<br/><br/>最后的文件目录结构为<br/>/MyWebRoot<br/>&#124;—/upload.html<br/>&#124;—/cgi-bin<br/>&#124;——/upload.cgi<br/>当然，你必须配置能够cgi-bin，并且程序要有权限在cgi-bin目录下创建文件（因为此例把文件上传到cgi-bin目录下）。<br/><br/>那么如何控制上传文件的大小呢？因为你有时会不允许用户上传太大的文件。<br/>通过分析cgic.c的源代码，我们发现它定义了一个变量cgiContentLength，表示请求的长度。但我们需要首先判断这是一个上传文件的请求，然后才能根据cgiContentLength来检查用户是否要上传一个太大的文件。<br/>cgic.c的main函数中进行了一系列if-else判断来检查请求的类型，首先确定这是一个post请求，然后确定数据的编码方式为 “multipart/form-data”，这个判断通过之后，就要开始准备接收数据了。所以我们要在接收数据开始之前使用 cgiContentLength判断大小，如果超过标准，就立即返回，不允许继续操作。<br/>下面贴出修改后代码片段（直接修改cgic.c的源代码即可）：<br/><br/>&nbsp;&nbsp; 1. else if (cgiStrEqNc(cgiContentType, &quot;multipart/form-data&quot;)) &#123;<br/>&nbsp;&nbsp; 2. #ifdef CGICDEBUG<br/>&nbsp;&nbsp; 3. CGICDEBUGSTART<br/>&nbsp;&nbsp; 4.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(dout, &quot;Calling PostMultipartInput&#92;n&quot;);<br/>&nbsp;&nbsp; 5. CGICDEBUGEND<br/>&nbsp;&nbsp; 6. #endif<br/>&nbsp;&nbsp; 7. //我的代码<br/>&nbsp;&nbsp; 8. //UpSize:文件长度上限值，以byte为单位，UpSize是一个int变量，因为cgiContentLength的类型为int<br/>&nbsp;&nbsp; 9.&nbsp;&nbsp;&nbsp;&nbsp; if(cgiContentLength&gt;UpSize)&#123;<br/>&nbsp;&nbsp;10.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; cgiHeaderContentType(&quot;text/html&quot;);<br/>&nbsp;&nbsp;11.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; printf(&quot;File too large!&#92;n&quot;);<br/>&nbsp;&nbsp;12.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; cgiFreeResources();<br/>&nbsp;&nbsp;13.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return -1;<br/>&nbsp;&nbsp;14.&nbsp;&nbsp;&nbsp;&nbsp; &#125;<br/>&nbsp;&nbsp;15. //我的代码结束<br/>&nbsp;&nbsp;16.&nbsp;&nbsp;&nbsp;&nbsp; if (cgiParsePostMultipartInput() != cgiParseSuccess) &#123;<br/>&nbsp;&nbsp;17. #ifdef CGICDEBUG<br/>&nbsp;&nbsp;18. CGICDEBUGSTART<br/>&nbsp;&nbsp;19.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf(dout, &quot;PostMultipartInput failed&#92;n&quot;);<br/>&nbsp;&nbsp;20. CGICDEBUGEND<br/>&nbsp;&nbsp;21. #endif<br/>&nbsp;&nbsp;22.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; cgiFreeResources();<br/>&nbsp;&nbsp;23.&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return -1;<br/>&nbsp;&nbsp;24.&nbsp;&nbsp;&nbsp;&nbsp; &#125;<br/>&nbsp;&nbsp;25. #ifdef CGICDEBUG<br/>&nbsp;&nbsp;26. CGICDEBUGSTART<br/>&nbsp;&nbsp;27.&nbsp;&nbsp;&nbsp;&nbsp; fprintf(dout, &quot;PostMultipartInput succeeded&#92;n&quot;);<br/>&nbsp;&nbsp;28. CGICDEBUGEND<br/>&nbsp;&nbsp;29.&nbsp;&nbsp;&nbsp;&nbsp; #endif<br/>&nbsp;&nbsp;30. &#125;<br/>&nbsp;&nbsp;31. &#125;<br/><br/>变量UpSize表示文件大小的上限。在cgic.c的main中找到相关代码，并修改成上面这样即可。你可以在cgic.c中定义UpSize，也可以在刚才完成的upload.c中定义，然后在cgic.c中用extern方式引用。
]]>
</description>
</item><item>
<link>http://jackxiang.com/post//#blogcomment</link>
<title><![CDATA[[评论] [转]CGIC简明教程]]></title> 
<author> &lt;user@domain.com&gt;</author>
<category><![CDATA[评论]]></category>
<pubDate>Thu, 01 Jan 1970 00:00:00 +0000</pubDate> 
<guid>http://jackxiang.com/post//#blogcomment</guid> 
<description>
<![CDATA[ 
	
]]>
</description>
</item>
</channel>
</rss>