1.POST和GET
一個(gè)CGI程序在于服務(wù)器之間的信息傳輸和數(shù)據(jù)傳輸一般通過(guò)兩種方法,即POST和GET。具體是哪一種方法這需要通過(guò)CGI的一個(gè)環(huán)境變量REQUEST_METHOD判斷(具體怎么判斷我會(huì)在下面詳細(xì)講解),在這之前先講一下URL編碼。
1.1 URL編碼
雖然在設(shè)置表單信息的傳輸方式時(shí)有POST和GET兩種方法,但是不管采取哪種方法,瀏覽器采取的編碼方式卻是完全相同的。編碼規(guī)則如下:
□ 變量之間使用“&”分開(kāi)
□ 變量與其對(duì)應(yīng)值之間使用“=”鏈接
□ 空格符使用“+”代替
□ 保留的控制字符則使用百分號(hào)接相應(yīng)的十六進(jìn)制ASCII代替
□ 空格是非法字符
□ 任意不可打印的ASCII 控制字符都為非法字符
□ 某些具有特殊意義的字符也用百分號(hào)接相應(yīng)的十六進(jìn)制ASCII代替
- <body>
- <form name="form1" action="/cgi-bin/pass.cgi" method="get">
- <table align="center">
- <tr><td align="center" colspan="2"></td></tr>
- <tr>
- <td align="right">用戶名</td>
- <td><input type="text" name="Username"></td>
- </tr>
- <tr>
- <td align="right">密 碼</td>
- <td><input type="password" name="Password"></td>
- </tr>
- <tr>
- <td><input type="submit" value="登 錄"></td>
- <td><input type="reset" value="取 消"></td>
- </tr>
- </table>
- </form>
- </body>
如果我們?cè)谟脩裘竺嫣顚?xiě)Tom,密碼后填寫(xiě)1234,則在點(diǎn)擊提交后傳給服務(wù)器的變量格式如下:
Username=Tom&Password=1234
下面講解POST和GET具體的具體工作方式
2.POST和GET工作方式
2.1 POST
如果在form表單中method使用POST方法,那么服務(wù)器會(huì)將會(huì)把從表單中填入的數(shù)據(jù)接收,并傳給相應(yīng)的CGI程序(就是action中指定的CGI程序),同時(shí)把REQUEST_METHOD環(huán)境變量設(shè)置為POST,而相應(yīng)的CGI程序檢查該環(huán)境變量,以確定其工作在POST接收數(shù)據(jù)方式,然后讀取這個(gè)數(shù)據(jù)。注意使用POST這種方法傳輸數(shù)據(jù)時(shí),Http在數(shù)據(jù)發(fā)送完后,并不會(huì)發(fā)送相應(yīng)的數(shù)據(jù)傳輸完畢提示信息,所以Http服務(wù)器提供了另一個(gè)環(huán)境變量CONTENET_LENGTH,該環(huán)境變量記錄了傳輸過(guò)來(lái)了多少個(gè)字節(jié)長(zhǎng)度的數(shù)據(jù)(單位為字節(jié)),所以在編寫(xiě)CGI程序時(shí),如果method為POST,就需要通過(guò)該變量來(lái)限定讀取的數(shù)據(jù)的長(zhǎng)度(如何實(shí)現(xiàn),下面講解)。
另外還有個(gè)環(huán)境變量CONTENET_TYPE,記錄從瀏覽器端發(fā)送來(lái)的數(shù)據(jù)類(lèi)型,現(xiàn)在一般發(fā)送的MIME類(lèi)型為Content-type: text/html\n\n,具體怎么使用在CGI中下面介紹。在確認(rèn)兩個(gè)環(huán)境變量的內(nèi)容都符合后,就開(kāi)始按下列規(guī)則解析表單傳輸過(guò)來(lái)的數(shù)據(jù),就是URL編碼的逆過(guò)程(不再贅述)。
2.2 GET
基本上GET方法和POST方法相同,不同的是,使用GET方法時(shí),數(shù)據(jù)被存儲(chǔ)到一個(gè)叫做QUERY_STRING的環(huán)境變量中了,具體如何得到該變量里的內(nèi)容,會(huì)在下面的例子中詳細(xì)講述。
說(shuō)了這么多,通過(guò)實(shí)例看一下,具體實(shí)現(xiàn)時(shí)如何編寫(xiě)CGI程序。
表單仍然和上面的HTML代碼相同。下面通過(guò)一個(gè)返回所填內(nèi)容的CGI程序講解。代碼如下:
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
-
- char* getcgidata(FILE* fp, char* requestmethod);
- int main()
- {
- char *input;
- char *req_method;
- char name[64];
- char pass[64];
- int i = 0;
- int j = 0;
-
-
- printf("Content-type: text/html\n\n");
- printf("The following is query reuslt:<br><br>");
-
- req_method = getenv("REQUEST_METHOD");
- input = getcgidata(stdin, req_method);
-
-
-
-
-
-
-
-
- for ( i = 9; i < (int)strlen(input); i++ )
- {
- if ( input[i] == '&' )
- {
- name[j] = '\0';
- break;
- }
- name[j++] = input[i];
- }
-
-
-
- for ( i = 19 + strlen(name), j = 0; i < (int)strlen(input); i++ )
- {
- pass[j++] = input[i];
- }
- pass[j] = '\0';
-
- printf("Your Username is %s<br>Your Password is %s<br> \n", name, pass);
-
- return 0;
- }
-
- char* getcgidata(FILE* fp, char* requestmethod)
- {
- char* input;
- int len;
- int size = 1024;
- int i = 0;
-
- if (!strcmp(requestmethod, "GET"))
- {
- input = getenv("QUERY_STRING");
- return input;
- }
- else if (!strcmp(requestmethod, "POST"))
- {
- len = atoi(getenv("CONTENT_LENGTH"));
- input = (char*)malloc(sizeof(char)*(size + 1));
-
- if (len == 0)
- {
- input[0] = '\0';
- return input;
- }
-
- while(1)
- {
- input[i] = (char)fgetc(fp);
- if (i == size)
- {
- input[i+1] = '\0';
- return input;
- }
-
- --len;
- if (feof(fp) || (!(len)))
- {
- i++;
- input[i] = '\0';
- return input;
- }
- i++;
-
- }
- }
- return NULL;
下面開(kāi)講:首先注意這行代碼 printf(
"Content-type: text/html\n\n");
通過(guò)它告訴服務(wù)器要輸出的內(nèi)容是文本內(nèi)容或者HTML,在編寫(xiě)CGI程序時(shí)容易遺留這一行,則會(huì)提示服務(wù)器內(nèi)部出錯(cuò),無(wú)法完成你的請(qǐng)求,需要注意的是后面兩個(gè)“\n\n”,這是必須的,具體為什么,我也不清楚,這樣寫(xiě)是正確。在這個(gè)地方,有的網(wǎng)友做的時(shí)候漢字輸出后是亂碼,這樣的話,可以在“\n\n”,之前輸出編碼信息,在window下一般為gb2312.
往下走,就是這一行了: req_method = getenv("REQUEST_METHOD");這是通過(guò)getenv()函數(shù)得到環(huán)境變量的值,在調(diào)用函數(shù)里判斷采用的那種方法,然后做出相應(yīng)的操作。
if (!strcmp(requestmethod, "GET"))
{
input = getenv("QUERY_STRING");
return input;
}
else if (!strcmp(requestmethod, "POST"))
{ //if (getenv(″CONTENT-LENGTH″))
len = atoi(getenv("CONTENT_LENGTH"));
input = (char*)malloc(sizeof(char)*(size + 1));
此處通過(guò)strcmp()函數(shù),判斷具體的方法,如果是GET方法,則通過(guò)getenv()函數(shù)直接獲取QUERY_STRING中的內(nèi)容,返回給主函數(shù)。繼續(xù)往下走,就是當(dāng)method為POST時(shí),如何通過(guò)環(huán)境變量CONTENET_LENGTH來(lái)限制接收數(shù)據(jù)的數(shù)量,這一句 if (getenv(″CONTENT-LENGTH″))判斷CONTENET_LENGTH是否存在,但是在編程時(shí)可以直接使用atoi()函數(shù),所以代碼中我注釋掉了這一行(編程時(shí)自己注意差別)
len=atoi (getenv(″CONTENT-LENGTH″));
此行首先檢查環(huán)境變量CONTENT-LENGTH是否存在的同時(shí),將此環(huán)境變量的值轉(zhuǎn)換成整數(shù),并賦給變量len。請(qǐng)注意Web服務(wù)器并不以文件結(jié)束符來(lái)終止它的輸出,所以如果不檢查環(huán)境變量CONTENT-LENGTH,CGI程序就無(wú)法知道什么時(shí)候輸入結(jié)束了。
下面這句 input = (char*)malloc(sizeof(char)*(size + 1));就是申請(qǐng)一段內(nèi)存空間,用于數(shù)據(jù)存儲(chǔ)。
再往下,就是C語(yǔ)言基礎(chǔ)了,這里不再贅述。
一般理解了這個(gè)例子就可以掌握POST和GET方法數(shù)據(jù)的獲取方式了。