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