Content-type的几种常见类型及php://input的使用
POST请求的消息主体放在entity body中,服务端根据请求头中的Content-Type字段来获取消息主体的编码方式,进而进行解析数据。
一、application/x-www-form-urlencoded
最常见的 POST 提交数据的方式,原生Form表单,如果不设置 enctype 属性,默认为application/x-www-form-urlencoded 方式提交数据。
首先,Content-Type被指定为 application/x-www-form-urlencoded;其次,提交的表单数据会转换为键值对并按照 key1=val1&key2=val2 的方式进行编码,key 和 val 都进行了 URL 转码。大部分服务端语言都对这种方式有很好的支持。
另外,如利用AJAX 提交数据时,也可使用这种方式。例如 jQuery,Content-Type 默认值都是”application/x-www-form-urlencoded;charset=utf-8”。
二、multipart/form-data
另一个常见的 POST 数据提交的方式, Form 表单的 enctype 设置为multipart/form-data,它会将表单的数据处理为一条消息,以标签为单元,用分隔符(这就是boundary的作用)分开,类似我们上面Content-Type中的例子。
由于这种方式将数据有很多部分,它既可以上传键值对,也可以上传文件,甚至多个文件。当上传的字段是文件时,会有Content-Type来说明文件类型;Content-disposition,用来说明字段的一些信息。每部分都是以 –boundary 开始,紧接着是内容描述信息,然后是回车,最后是字段具体内容(字段、文本或二进制等)。如果传输的是文件,还要包含文件名和文件类型信息。消息主体最后以 –boundary– 标示结束。
三、application/json
Content-Type: application/json 作为响应头比较常见。实际上,现在越来越多的人把它作为请求头,用来告诉服务端消息主体是序列化后的 JSON 字符串,其中一个好处就是JSON 格式支持比键值对复杂得多的结构化数据。由于 JSON 规范的流行,除了低版本 IE 之外的各大浏览器都原生支持JSON.stringify,服务端语言也都有处理 JSON 的函数,使用起来没有困难。
Google 的 AngularJS 中的 Ajax 功能,默认就是提交 JSON 字符串。
四、text/xml
XML的作用不言而喻,用于传输和存储数据,它非常适合万维网传输,提供统一的方法来描述和交换独立于应用程序或供应商的结构化数据,在JSON出现之前是业界一大标准(当然现在也是),相比JSON的优缺点大家有兴趣可以上网search。因此,在POST提交数据时,xml类型也是不可缺少的一种,虽然一般场景上使用JSON可能更轻巧、灵活。
五、binary (application/octet-stream)
在Chrome浏览器的Postman工具中,还可以看到”binary“这一类型,指的就是一些二进制文件类型。如application/pdf,指定了特定二进制文件的MIME类型。就像对于text文件类型若没有特定的子类型(subtype),就使用 text/plain。类似的,二进制文件没有特定或已知的 subtype,即使用 application/octet-stream,这是应用程序文件的默认值,一般很少直接使用 。
对于application/octet-stream,只能提交二进制,而且只能提交一个二进制,如果提交文件的话,只能提交一个文件,后台接收参数只能有一个,而且只能是流(或者字节数组)。
很多web服务器使用默认的 application/octet-stream 来发送未知类型。出于一些安全原因,对于这些资源浏览器不允许设置一些自定义默认操作,导致用户必须存储到本地以使用。一般来说,设置正确的MIME类型很重要。
$data = file_get_contents("php://input");
php://input 是个可以访问请求的原始数据的只读流。 POST 请求的情况下,最好使用 php://input 来代替 $HTTP_RAW_POST_DATA,因为它不依赖于特定的 php.ini 指令。 而且,这样的情况下 $HTTP_RAW_POST_DATA 默认没有填充, 比** always_populate_raw_post_data 潜在需要更少的内存。 enctype="multipart/form-data" 的时候 php://input 是无效的。
1.php://input 可以读取http entity body中指定长度的值,由Content-Length指定长度,不管是POST方式或者GET方法提交过来的数据。但是,一般GET方法提交数据 时,http request entity body部分都为空。
2.php://input 与$HTTP_RAW_POST_DATA读取的数据是一样的,都只读取Content-Type不为multipart/form-data的数据。
3,Coentent-Type仅在取值为application/x-www-data-urlencoded和multipart/form-data两种情况下,PHP才会将http请求数据包中相应的数据填入全局变量$_POST
4,PHP不能识别的Content-Type类型的时候,会将http请求包中相应的数据填入变量$HTTP_RAW_POST_DATA
5. 只有Coentent-Type为multipart/form-data的时候,PHP不会将http请求数据包中的相应数据填入php://input,否则其它情况都会。填入的长度,由Coentent-Length指定。
6.只有Content-Type为application/x-www-data-urlencoded时,php://input数据才跟$_POST数据相一致。
7.php://input数据总是跟$HTTP_RAW_POST_DATA相同,但是php://input比$HTTP_RAW_POST_DATA更凑效,且不需要特殊设置php.ini
8.PHP会将PATH字段的query_path部分,填入全局变量$_GET。通常情况下,GET方法提交的http请求,body为空。 例子 1.php用file_get_contents("php://input")或者$HTTP_RAW_POST_DATA可以接收xml数据 比如: getXML.php;//接收XML地址
1 <?php 2 $xmldata = file_get_contents("php://input"); 3 $data = (array)simplexml_load_string($xmldata); 4 ?>
这里的$data就是包含xml数据的数组,具体php解析xml数据更新详细的方法 sendXML.php
1 <?php 2 $xml = '<xml>xmldata</xml>';//要发送的xml 3 $url = 'http://localhost/test/getXML.php';//接收XML地址 4 $header = 'Content-type: text/xml';//定义content-type为xml 5 $ch = curl_init(); //初始化 6 curl curl_setopt($ch, CURLOPT_URL, $url);//设置链接 7 curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);//设置是否返回信息 8 curl_setopt($ch, CURLOPT_HTTPHEADER, $header);//设置HTTP头 9 curl_setopt($ch, CURLOPT_POST, 1);//设置为POST方式 10 curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);//POST数据 11 $response = curl_exec($ch);//接收返回信息 12 13 if(curl_errno($ch)){//出错则显示错误信息 14 print curl_error($ch); 15 } 16 17 curl_close($ch); //关闭curl链接 18 echo $response;//显示返回信息 19 ?>
2.一个手机上传图片到服务器的小程序
上传文件
1 <?php 2 //@file phpinput_post.php 3 $data=file_get_contents('btn.png'); 4 $http_entity_body = $data; 5 $http_entity_type = 'application/x-www-form-urlencoded'; 6 $http_entity_length = strlen($http_entity_body); 7 $host = '127.0.0.1'; 8 $port = 80; 9 $path = '/image.php'; 10 $fp = fsockopen($host, $port, $error_no, $error_desc, 30); 11 if ($fp){ 12 fputs($fp, "POST {$path} HTTP/1.1\r\n"); 13 fputs($fp, "Host: {$host}\r\n"); 14 fputs($fp, "Content-Type: {$http_entity_type}\r\n"); 15 fputs($fp, "Content-Length: {$http_entity_length}\r\n"); 16 fputs($fp, "Connection: close\r\n\r\n"); 17 fputs($fp, $http_entity_body . "\r\n\r\n"); 18 19 while (!feof($fp)) { 20 $d .= fgets($fp, 4096); 21 } 22 fclose($fp); 23 echo $d; 24 } 25 ?>
接收文件
1 <?php 2 /** 3 *Recieve image data 4 **/error_reporting(E_ALL); 5 function get_contents() { 6 $xmlstr= file_get_contents("php://input"); 7 $filename=time().'.png'; 8 if(file_put_contents($filename,$xmlstr)){ 9 echo 'success'; 10 }else{ 11 echo 'failed'; 12 } 13 } 14 get_contents(); 15 ?>
3.获取HTTP请求原文
1 /** 2 * 获取HTTP请求原文 3 * @return string 4 */ 5 function get_http_raw() { 6 $raw = ''; // (1) 请求行 7 $raw .= $_SERVER['REQUEST_METHOD'].' '.$_SERVER['REQUEST_URI'].' '.$_SERVER['SERVER_PROTOCOL']."\r\n"; 8 // (2) 请求Headers 9 foreach($_SERVER as $key => $value) { 10 if(substr($key, 0, 5) === 'HTTP_') { 11 $key = substr($key, 5); 12 $key = str_replace('_', '-', $key); 13 $raw .= $key.': '.$value."\r\n"; 14 } 15 } 16 // (3) 空行 17 $raw .= "\r\n"; 18 // (4) 请求Body 19 $raw .= file_get_contents('php://input'); 20 return $raw; 21 }
综上,也就是说$_POST在获取参数时并不是都能拿到。
还有就是第三方接口需要传数组时 :