嵌入式应用中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 ,具体怎么使用在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 ");   
  16.                 printf("Content-type: text/html ");   
  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] = '';   
  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] = '';   
  46.    
  47.                 printf("Your Username is %s<br>Your Password is %s<br>  ", 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] = '';   
  72.                                             return input;   
  73.                              }   
  74.                                
  75.                              while(1)   
  76.                              {   
  77.                                             input[i] = (char)fgetc(fp);   
  78.                                             if (i == size)   
  79.                                             {   
  80.                                                          input[i+1] = '';   
  81.                                                          return input;   
  82.                                             }   
  83.                                               
  84.                                             --len;   
  85.                                             if (feof(fp) || (!(len)))   
  86.                                             {   
  87.                                                          i++;   
  88.                                                          input[i] = '';   
  89.                                                          return input;   
  90.                                             }   
  91.                                             i++;   
  92.                                               
  93.                              }   
  94.                 }   
  95.                 return NULL;  

 下面开讲:首先注意这行代码 printf("Content-type: text/html ");
通过它告诉服务器要输出的内容是文本内容或者HTML,在编写CGI程序时容易遗留这一行,则会提示服务器内部出错,无法完成你的请求,需要注意的是后面两个“ ”,这是必须的,具体为什么,我也不清楚,这样写是正确。在这个地方,有的网友做的时候汉字输出后是乱码,这样的话,可以在“ ”,之前输出编码信息,在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方法数据的获取方式了。
原文地址:https://www.cnblogs.com/lidabo/p/5756251.html