详解PHP输入流php://input

在使用xml-rpc的时候,server端获取client数据,主要是通过php输入流input,而不是$_POST数组。所以,这里主要探讨php输入流php://input

对一php://input介绍,PHP官方手册文档有一段话对它进行了很明确地概述:
“php://input allows you to read raw POST data. It is a less memory intensive alternative to $HTTP_RAW_POST_DATA and does not need any special php.ini directives. php://input is not available with enctype=”multipart/form-data”. 
翻译过来,是这样: 
“php://input可以读取没有处理过的POST数据。相较于$HTTP_RAW_POST_DATA而言,它给内存带来的压力较小,并且不需要特殊的php.ini设置。php://input不能用于enctype=multipart/form-data” 

我们应该怎么去理解这段概述呢?!我把它划分为三部分,逐步去理解。 
读取POST数据 
不能用于multipart/form-data类型 
php://input VS $HTTP_RAW_POST_DATA 
读取POST数据 

PHPer们一定很熟悉$_POST这个内置变量。$_POST与 php://input存在哪些关联与区别呢?另外,客户端向服务端交互数据,最常用的方法除了POST之外,还有GET。既然php://input作 为PHP输入流,它能读取GET数据吗?这二个问题正是我们这节需要探讨的主要内容。 
经验告诉我们,从测试与观察中总结,会是一个很凑效的方法。这里,我写了几个脚本来帮助我们测试。

程序01:打印出接收到的数据

1 <?php 
2 $raw_post_data = file_get_contents('php://input', 'r'); 
3 echo "-------\$_POST------------------\n"; 
4 echo var_dump($_POST) . "\n"; 
5 echo "-------php://input-------------\n"; 
6 echo $raw_post_data . "\n"; 
7 ?> 

程序02:模拟以POST方法提交表单数据

 1 <?php 
 2 $http_entity_body = 'n=' . urldecode('perfgeeks') . '&p=' . urldecode('7788'); 
 3 $http_entity_type = 'application/x-www-form-urlencoded'; 
 4 $http_entity_length = strlen($http_entity_body); 
 5 $host = '192.168.0.6'; 
 6 $port = 80; 
 7 $path = '/phpinput_server.php'; 
 8 $fp = fsockopen($host, $port, $error_no, $error_desc, 30); 
 9 if ($fp) { 
10 fputs($fp, "POST {$path} HTTP/1.1\r\n"); 
11 fputs($fp, "Host: {$host}\r\n"); 
12 fputs($fp, "Content-Type: {$http_entity_type}\r\n"); 
13 fputs($fp, "Content-Length: {$http_entity_length}\r\n"); 
14 fputs($fp, "Connection: close\r\n\r\n"); 
15 fputs($fp, $http_entity_body . "\r\n\r\n"); 
16 
17 while (!feof($fp)) { 
18 $d .= fgets($fp, 4096); 
19 } 
20 fclose($fp); 
21 echo $d; 
22 } 
23 ?> 

我们可以通过使用工具ngrep抓取http请求包(因为我们需要探知的是php://input,所以我们这里只抓取http Request数据包)

 

posted @ 2015-12-16 18:06  Steven*  阅读(1588)  评论(0编辑  收藏  举报