嵌入式应用中CGI编程中POST、GET及环境变量详解

原载地址:http://3633188.blog.51cto.com/3623188/828095

    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方法数据的获取方式了。

时间: 2024-10-31 16:26:36

嵌入式应用中CGI编程中POST、GET及环境变量详解的相关文章

为excel vba中添加、删除模块并插入全过程图文详解

  为excel vba中添加.删除模块并插入全过程图文详解         方法/步骤 1.点按快速启动栏excel 程序图标 进入excel 界面 点击选中任意单元格 然后按alt+f11 进入vbe界面 2.点击菜单栏 插入命令 在弹出的活动菜单中点按模块命令 3.另一种方式插入模块的方法可以在工程资管管理器中鼠标点击空白处 右键单击鼠标 在弹出的快捷菜单中选择插入命令 二级菜单中选择模块命令 4.如图所示模块1.模块2分别是通过菜单栏插入命令 和工程资源管理器点击右键创建的模块 5.如果

Eclipse中改变默认的workspace的方法及说明详解_java

eclipse中改变默然的workspace的方法可以有以下几种: 1.在创建project的时候,手动选择使用新的workspace,如创建一个web project,在向导中的Location选项,取消使用"Use default location",同时在下面选择新的workspace. 2.在file菜单中选择switch workspace项,即可选择一个新的workspace 3.在eclipse安装目录下configuration/.settings目录下的 org.ec

Linux中字符模式下设置和删除环境变量?

Linux中字符模式下设置和删除环境变量? bash下 设置:export 变量名=变量值删除:unset 变量名 csh下 设置:setenv 变量名 变量值删除:unsetenv 变量名

Android中Spinner(下拉框)控件的使用详解_Android

android给我们提供了一个spinner控件,这个控件主要就是一个列表,那么我们就来说说这个控件吧,这个控件在以前的也看见过,但今天还是从新介绍一遍吧. Spinner位于 android.widget包下,每次只显示用户选中的元素,当用户再次点击时,会弹出选择列表供用户选择,而选择列表中的元素同样来自适配器.Spinner是View类得一个子类. 1.效果图 2.创建页面文件(main.xml) <Spinner android:id="@+id/spinner1" and

js中使用使用原型(prototype)定义方法的好处详解_javascript技巧

经常在前端面试或是和其他同行沟通是,在谈到构造在JS定义构造函数的方法是最好使用原型的方式:将方法定义到构造方法的prototype上,这样的好处是,通过该构造函数生成的实例所拥有的方法都是指向一个函数的索引,这样可以节省内存. 当然,这种说法没有任何问题,只是在实现上,并非只有使用prototype的方式才能达到这样的效果,我们可以将方法以函数的形式定义在构造函数之外,然后在构造函数中通过this.method = method的方式,这样生成的实例的方法也都通过索引指向一个函数,具体如下:

jQuery中each()、find()和filter()等节点操作方法详解(推荐)_jquery

1.each(callback) 官方解释: 返回值:jQuery 概述 以每一个匹配的元素作为上下文来执行一个函数. 意味着,每次执行传递进来的函数时,函数中的this关键字都指向一个不同的DOM元素(每次都是一个不同的匹配元素).而且,在每次执行函数时,都会给函数传递一个表示作为执行环境的元素在匹配的元素集合中所处位置的数字值作为参数(从零开始的整型). 返回 'false' 将停止循环 (就像在普通的循环中使用 'break').返回 'true' 跳至下一个循环(就像在普通的循环中使用'

iOS 泛型中nullable、null resettable、null kindof 用法详解_IOS

 iOS9新出的关键字:用来修饰属性,或者方法的参数,方法的返回值 iOS9新出关键字nonnull,nullable,null_resettable,_Null_unspecified 需要注意的一点只能修饰对象,不能修饰基本数据类型. 虽然在项目的代码编写中不会经常用到,不过在调用苹果系统方法的时候还是会经常遇到,需要做一个总结 nullable作用:表示可以为空 nullable书写规范: // 方式一: @property (nonatomic, strong, nullable) NS

yii2中结合gridview如何使用modal弹窗实例代码详解_php实例

在上篇文章给大家介绍了Yii2中如何使用modal弹窗(基本使用),即以创建为例. 实际开发中,我们往往还会遇到列表页数据修改要使用modal的情况,如果是一般的循环展示,相信大多数人看了modal的基本使用都会操作,但是结合gridview估计有些人就开始吃不消了,我们看看如何解决这个问题! 1.gridview的操作增加[更新]按钮,并指定data-toggle data-target class以及data-id的值 [ 'class' => 'yii\grid\ActionColumn'

JS中call/apply、arguments、undefined/null方法详解_javascript技巧

a.call和apply方法详解 -------------------------------------------------------------------------------- call方法: 语法:call([thisObj[,arg1[, arg2[, [,.argN]]]]]) 定义:调用一个对象的一个方法,以另一个对象替换当前对象. 说明: call 方法可以用来代替另一个对象调用一个方法.call 方法可将一个函数的对象上下文从初始的上下文改变为由 thisObj 指