嵌入式應用中CGI編程中POST、GET及環境變量詳解

    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>
     如果我們在用戶名後面填寫Tom,密碼後填寫1234,則在點擊提交後傳給服務器的變量格式如下:
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;
}
     爲了方便網友測試全部代碼貼上。
     下面開講:首先注意這行代碼 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));
   此處通過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程序就無法知道什麼時候輸入結束了。
下面這句 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(代碼來自網頁博主) 
       
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章