1.POST和GET
一個CGI程序在於服務器之間的信息傳輸和數據傳輸一般通過兩種方法,即POST和GET。具體是哪一種方法這需要通過CGI的一個環境變量REQUEST_METHOD判斷(具體怎麼判斷我會在下面詳細講解),在這之前先講一下URL編碼。
1.1 URL編碼
雖然在設置表單信息的傳輸方式時有POST和GET兩種方法,但是不管採取哪種方法,瀏覽器採取的編碼方式卻是完全相同的。編碼規則如下:
□ 變量之間使用“&”分開
□ 變量與其對應值之間使用“=”鏈接
□ 空格符使用“+”代替
□ 保留的控制字符則使用百分號接相應的十六進制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>
<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>
Username=Tom&Password=1234
下面講解POST和GET具體的具體工作方式
2.POST和GET工作方式
2.1 POST
如果在form表單中method使用POST方法,那麼服務器會將會把從表單中填入的數據接收,並傳給相應的CGI程序(就是action中指定的CGI程序),同時把REQUEST_METHOD環境變量設置爲POST,而相應的CGI程序檢查該環境變量,以確定其工作在POST接收數據方式,然後讀取這個數據。注意使用POST這種方法傳輸數據時,Http在數據發送完後,並不會發送相應的數據傳輸完畢提示信息,所以Http服務器提供了另一個環境變量CONTENET_LENGTH,該環境變量記錄了傳輸過來了多少個字節長度的數據(單位爲字節),所以在編寫CGI程序時,如果method爲POST,就需要通過該變量來限定讀取的數據的長度(如何實現,下面講解)。
另外還有個環境變量CONTENET_TYPE,記錄從瀏覽器端發送來的數據類型,現在一般發送的MIME類型爲Content-type: text/html\n\n,具體怎麼使用在CGI中下面介紹。在確認兩個環境變量的內容都符合後,就開始按下列規則解析表單傳輸過來的數據,就是URL編碼的逆過程(不再贅述)。
2.2 GET
基本上GET方法和POST方法相同,不同的是,使用GET方法時,數據被存儲到一個叫做QUERY_STRING的環境變量中了,具體如何得到該變量裏的內容,會在下面的例子中詳細講述。
說了這麼多,通過實例看一下,具體實現時如何編寫CGI程序。
表單仍然和上面的HTML代碼相同。下面通過一個返回所填內容的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/plain; charset=iso-8859-1\n\n");
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);
// 我們獲取的input字符串可能像如下的形式
// Username="admin"&Password="aaaaa"
// 其中"Username="和"&Password="都是固定的
// 而"admin"和"aaaaa"都是變化的,也是我們要獲取的
// 前面9個字符是UserName=
// 在"UserName="和"&"之間的是我們要取出來的用戶名
for ( i = 9; i < (int)strlen(input); i++ )
{
if ( input[i] == '&' )
{
name[j] = '\0';
break;
}
name[j++] = input[i];
}
// 前面9個字符 + "&Password="10個字符 + Username的字符數
// 是我們不要的,故省略掉,不拷貝
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;
}
#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/plain; charset=iso-8859-1\n\n");
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);
// 我們獲取的input字符串可能像如下的形式
// Username="admin"&Password="aaaaa"
// 其中"Username="和"&Password="都是固定的
// 而"admin"和"aaaaa"都是變化的,也是我們要獲取的
// 前面9個字符是UserName=
// 在"UserName="和"&"之間的是我們要取出來的用戶名
for ( i = 9; i < (int)strlen(input); i++ )
{
if ( input[i] == '&' )
{
name[j] = '\0';
break;
}
name[j++] = input[i];
}
// 前面9個字符 + "&Password="10個字符 + Username的字符數
// 是我們不要的,故省略掉,不拷貝
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");
通過它告訴服務器要輸出的內容是文本內容或者HTML,在編寫CGI程序時容易遺留這一行,則會提示服務器內部出錯,無法完成你的請求,需要注意的是後面兩個“\n\n”,這是必須的,具體爲什麼,我也不清楚,這樣寫是正確。在這個地方,有的網友做的時候漢字輸出後是亂碼,這樣的話,可以在“\n\n”,之前輸出編碼信息,在window下一般爲gb2312.
往下走,就是這一行了:
req_method = getenv("REQUEST_METHOD");這是通過getenv()函數得到環境變量的值,在調用函數裏判斷採用的那種方法,然後做出相應的操作。
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));
{
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()函數,判斷具體的方法,如果是GET方法,則通過getenv()函數直接獲取QUERY_STRING中的內容,返回給主函數。繼續往下走,就是當method爲POST時,如何通過環境變量CONTENET_LENGTH來限制接收數據的數量,這一句
if (getenv(″CONTENT-LENGTH″))判斷CONTENET_LENGTH是否存在,但是在編程時可以直接使用atoi()函數,所以代碼中我註釋掉了這一行(編程時自己注意差別)
len=atoi (getenv(″CONTENT-LENGTH″));
此行首先檢查環境變量CONTENT-LENGTH是否存在的同時,將此環境變量的值轉換成整數,並賦給變量len。請注意Web服務器並不以文件結束符來終止它的輸出,所以如果不檢查環境變量CONTENT-LENGTH,CGI程序就無法知道什麼時候輸入結束了。
len=atoi (getenv(″CONTENT-LENGTH″));
此行首先檢查環境變量CONTENT-LENGTH是否存在的同時,將此環境變量的值轉換成整數,並賦給變量len。請注意Web服務器並不以文件結束符來終止它的輸出,所以如果不檢查環境變量CONTENT-LENGTH,CGI程序就無法知道什麼時候輸入結束了。
下面這句
input = (char*)malloc(sizeof(char)*(size + 1));就是申請一段內存空間,用於數據存儲。
再往下,就是C語言基礎了,這裏不再贅述。
一般理解了這個例子就可以掌握POST和GET方法數據的獲取方式了。
3.在編寫HTML代碼時,我用到了,需要不在網頁中顯示文本填寫框,但是需要單擊按鈕時,隱藏的傳數據給CGI程序,用來供CGI程序判斷,並作出相應的處理(其實對於做網頁的人來說都簡單的不值一提,只對像我這種搞嵌入式對網站開發只瞭解皮毛的人有用,希望對您有用)如下格式
對應的HTML代碼如下
<form action="./cgi-bin/hello.cgi" target =main ><input type="hidden"name="Username" value="0"><input style= 'width: 119px; height:49px' type="submit" value="教室">
簡單講一下,在單擊網頁中顯示的“教室”按鈕時,出給CGI程序的是Username=0,這樣一串字符串。將這個例子,只是想講一下怎麼樣隱式的傳輸數據,主要是使用了“hedden”屬性。
寫在最後,我在做着一塊的項目時,這一塊內容的知識相對比較少,而且比較分散,所以我就總結了一下,希望對大家有用。
參考資料:《CGI編程指南》機械工業出版社
http://blog.sina.com.cn/s/blog_3f839e670100bzzh.html(代碼來自網頁博主)