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


    1.POST和GET

     一個CGI程序在於服務器之間的信息傳輸和數據傳輸一般通過兩種方法,即POST和GET。具體是哪一種方法這需要通過CGI的一個環境變量REQUEST_METHOD判斷(具體怎么判斷我會在下面詳細講解),在這之前先講一下URL編碼。
     1.1 URL編碼
      雖然在設置表單信息的傳輸方式時有POST和GET兩種方法,但是不管采取哪種方法,瀏覽器采取的編碼方式卻是完全相同的。編碼規則如下:
      □ 變量之間使用“&”分開
      □ 變量與其對應值之間使用“=”鏈接
      □ 空格符使用“+”代替
      □ 保留的控制字符則使用百分號接相應的十六進制ASCII代替
      □ 空格是非法字符
      □ 任意不可打印的ASCII 控制字符都為非法字符
      □ 某些具有特殊意義的字符也用百分號接相應的十六進制ASCII代替 
[html]  view plain  copy
 
 print?
  1. <body>   
  2. <form name="form1" action="/cgi-bin/pass.cgi" method="get">   
  3. <table align="center">   
  4.         <tr><td align="center" colspan="2"></td></tr>   
  5.         <tr>   
  6.              <td align="right">用戶名</td>   
  7.              <td><input type="text" name="Username"></td>   
  8.         </tr>   
  9.         <tr>   
  10.              <td align="right">密  碼</td>   
  11.              <td><input type="password" name="Password"></td>   
  12.         </tr>   
  13.         <tr>   
  14.              <td><input type="submit" value="登    錄"></td>   
  15.              <td><input type="reset" value="取    消"></td>   
  16.         </tr>   
  17. </table>   
  18. </form>   
  19. </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程序講解。代碼如下:
[cpp]  view plain  copy
 
 print?
  1. #include <stdio.h>   
  2.  #include <stdlib.h>   
  3.  #include <string.h>   
  4.    
  5.  char* getcgidata(FILE* fp, char* requestmethod);   
  6. int main()   
  7. {   
  8.                 char *input;   
  9.                 char *req_method;   
  10.                 char name[64];   
  11.                 char pass[64];   
  12.                 int i = 0;   
  13.                 int j = 0;   
  14.                   
  15.  //         printf("Content-type: text/plain; charset=iso-8859-1\n\n");   
  16.                 printf("Content-type: text/html\n\n");   
  17.                 printf("The following is query reuslt:<br><br>");   
  18.    
  19.                 req_method = getenv("REQUEST_METHOD");   
  20.                 input = getcgidata(stdin, req_method);   
  21.    
  22.                 // 我們獲取的input字符串可能像如下的形式   
  23.                 // Username="admin"&Password="aaaaa"   
  24.                 // 其中"Username="和"&Password="都是固定的   
  25.                 // 而"admin"和"aaaaa"都是變化的,也是我們要獲取的   
  26.                   
  27.                 // 前面9個字符是UserName=   
  28.                 // 在"UserName="和"&"之間的是我們要取出來的用戶名   
  29.                 for ( i = 9; i < (int)strlen(input); i++ )   
  30.                 {   
  31.                              if ( input[i] == '&' )   
  32.                              {   
  33.                                             name[j] = '\0';   
  34.                                             break;   
  35.                              }                                       
  36.                              name[j++] = input[i];   
  37.                 }   
  38.    
  39.                 // 前面9個字符 + "&Password="10個字符 + Username的字符數   
  40.                 // 是我們不要的,故省略掉,不拷貝   
  41.                 for ( i = 19 + strlen(name), j = 0; i < (int)strlen(input); i++ )   
  42.                 {   
  43.                              pass[j++] = input[i];   
  44.                 }   
  45.                 pass[j] = '\0';   
  46.    
  47.                 printf("Your Username is %s<br>Your Password is %s<br> \n", name, pass);   
  48.                   
  49.                 return 0;   
  50. }   
  51.    
  52.  char* getcgidata(FILE* fp, char* requestmethod)   
  53. {   
  54.                 char* input;   
  55.                 int len;   
  56.                 int size = 1024;   
  57.                 int i = 0;   
  58.                   
  59.                 if (!strcmp(requestmethod, "GET"))   
  60.                 {   
  61.                              input = getenv("QUERY_STRING");   
  62.                              return input;   
  63.                 }   
  64.                 else if (!strcmp(requestmethod, "POST"))   
  65.                 {   
  66.                              len = atoi(getenv("CONTENT_LENGTH"));   
  67.                              input = (char*)malloc(sizeof(char)*(size + 1));   
  68.                                
  69.                              if (len == 0)   
  70.                              {   
  71.                                             input[0] = '\0';   
  72.                                             return input;   
  73.                              }   
  74.                                
  75.                              while(1)   
  76.                              {   
  77.                                             input[i] = (char)fgetc(fp);   
  78.                                             if (i == size)   
  79.                                             {   
  80.                                                          input[i+1] = '\0';   
  81.                                                          return input;   
  82.                                             }   
  83.                                               
  84.                                             --len;   
  85.                                             if (feof(fp) || (!(len)))   
  86.                                             {   
  87.                                                          i++;   
  88.                                                          input[i] = '\0';   
  89.                                                          return input;   
  90.                                             }   
  91.                                             i++;   
  92.                                               
  93.                              }   
  94.                 }   
  95.                 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方法數據的獲取方式了。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM