php教程 PHP开发 PHP 입력 스트림 소개 php://input

PHP 입력 스트림 소개 php://input

Dec 23, 2016 am 11:00 AM

php://input에 대한 소개를 위해 공식 PHP 매뉴얼 문서에는 이를 명확하게 설명하는 단락이 있습니다.

“php://input을 사용하면 원시 POST 데이터를 읽을 수 있습니다. $HTTP_RAW_POST_DATA에 비해 메모리 사용량이 적고 특별한 php.ini 지시문이 필요하지 않습니다. enctype="multipart/form-data".
번역하면 다음과 같습니다.
"php://input은 처리되지 않은 POST 데이터를 읽을 수 있습니다. $HTTP_RAW_POST_DATA에 비해 메모리에 대한 부담이 적고 특별한 php.ini 설정이 필요하지 않습니다. php://input은 enctype=multipart/form-data에 사용할 수 없습니다.”

이 개요를 어떻게 이해해야 할까요?! 3개 부분으로 나누어 단계별로 이해해보았습니다.
POST 데이터를 읽어보세요.
multipart/form-data 유형에는 사용할 수 없습니다.
php://input VS $HTTP_RAW_POST_DATA
POST 데이터 읽기

PHPer는 내장 변수 $_POST $에 익숙해야 합니다. _POST와 php://input의 관계와 차이점은 무엇입니까? 또한 클라이언트가 서버와 상호 작용하는 데 가장 일반적으로 사용되는 방법은 POST 외에 GET입니다. , 그것은 사용될 수 있습니다. 이 두 가지 문제는 우리가 이 섹션에서 논의해야 할 주요 내용입니다.
여기서는 테스트와 관찰을 요약하는 것이 매우 효과적인 방법이 될 것임을 알려줍니다. 몇 가지 테스트에 도움이 되는 스크립트입니다.

@file 192.168.0.6:/phpinput_server.php는 수신된 데이터를 인쇄합니다.
@file 192.168.0.8:/phpinput_post.php는 다음을 사용하여 양식 데이터 제출을 시뮬레이션합니다. POST 메서드
@file 192.168.0.8:/phpinput_xmlrpc.php는 POST 메서드를 사용하여 xmlrpc 요청 실행을 시뮬레이션합니다.
@file 192.168.0.8:/phpinput_get.php는 GET 메서드를 사용하여 양식 제출을 시뮬레이션합니다.
phpinput_server .php 및 phpinput_post.php



//@file phpinput_server.php
$raw_post_data = file_get_contents('php://input', 'r ');
echo "------$_POST-----n"
echo var_dump($_POST) "n ";
echo "-------php://input-------------n";
echo $raw_post_data . "n";
? >

//@file phpinput_post.php
$http_entity_body = 'n=' .urldecode('perfgeeks') . );
$http_entity_type = 'application/x-www-form-urlencoded';
$http_entity_length = strlen($http_entity_body)
$host = '192.168.0.6'; 80;
$path = '/phpinput_server.php';
$fp = fsockopen($host, $port, $error_no, $error_desc, 30)
if ($fp) {
fputs($fp, "POST {$path} HTTP/1.1rn");
fputs($fp, "호스트: {$host}rn")
fputs($fp, "콘텐츠 유형: {$http_entity_type}rn ");
fputs($fp, "콘텐츠 길이: {$http_entity_length}rn");
fputs($fp, "연결: closenrn");
fputs($ fp, $http_entity_body . "rnrn");

while (!feof($fp)) {
$d .= fgets($fp, 4096)
}
fclose( $fp);
echo $d;
}
?>
ngrep 도구를 사용하여 http 요청 패킷을 가져올 수 있습니다. 왜냐하면 우리가 감지해야 하는 것은 php://input이기 때문입니다. 그래서 우리는 여기서 http 요청 패킷만 가져옵니다. 테스트 스크립트를 실행해 보겠습니다. phpinput_post.php

@php /phpinput_post.php
HTTP/1.1 200 OK
날짜: 2010년 4월 8일 목요일 03:23:36 GMT
서버: Apache /2.2.3(CentOS)
X-Powered-By: PHP/5.1.6
Content-Length: 160
Connection: close
Content-Type: text/html- 8
-------$_POST------
array(2) {
["n"]=> string(9) "perfgeeks"
["p"]=> string(4) "7788"
}
-------php://input------ -------
n=perfgeeks&p=7788
ngrep을 통해 캡처된 http 요청 패킷은 다음과 같습니다.

T 192.168.0.8:57846 -> AP ]
POST /phpinput_server.php HTTP/1.1..
호스트: 192.168.0.6..Content-Type: application/x-www-form-urlencoded..Co
ntent-Length: 18. . 연결: close....n=perfgeeks&p=7788....
자세히 살펴보면
1, $_POST 데이터, php://input 데이터 및 httpd 엔터티 본문 데이터가 " 일관됨"
2. http 요청의 Content-Type은 application/x-www-form-urlencoded입니다. 이는 http 요청 본문의 데이터가 http의 post 메서드를 사용하여 제출된 양식 데이터이며 urlencode() 에 의해 처리되었습니다.
(참고: 아래에 표시되지 않는 굵은 부분에 주의하세요.)

xml-rpc 요청을 시뮬레이션하는 phpinput_xmlrpc.php 스크립트의 원본 파일 콘텐츠를 살펴보겠습니다. POST 방식으로 제출됩니다.



//@file phpinput_xmlrpc.php
$http_entity_body = "nn jt_userinfon"; 
$http_entity_type = '텍스트/html'; 
$http_entity_length = strlen($http_entity_body); 
$host = '192.168.0.6'; 
$포트 = 80; 
$path = '/phpinput_server.php'; 
$fp = fsockopen($host, $port, $error_no, $error_desc, 30); 
if ($fp) { 
fputs($fp, "POST {$path} HTTP/1.1rn"); 
fputs($fp, "호스트: {$host}rn"); 
fputs($fp, "콘텐츠 유형: {$http_entity_type}rn"); 
fputs($fp, "콘텐츠 길이: {$http_entity_length}rn"); 
fputs($fp, "연결: closenrn"); 
fputs($fp, $http_entity_body . "rnrn"); 
while (!feof($fp)) { 
$d .= fgets($fp, 4096); 


fclose($fp); 
에코 $d; 

?> 서버: Apache/2.2.3(CentOS) 
X-Powered-By: PHP/5.1.6 
콘텐츠 길이: 154 
연결: 닫기 
콘텐츠 유형: text/html ; charset=UTF-8 

-------$_POST------ 
배열(0) { 
}

-------php://input------------- 
 
<메소드콜> 
jt_userinfo 
 
执行这个脚本的时候,저희는 ngrep抓取的http请求数据包如下 

T 192.168.0.8:45570 -> 192.168.0.6:80 [AP] 
POST /phpinput_server.php HTTP/1.1.. 
호스트: 192.168.0.6..Content-Type: text/html..Content-Length: 75..Connec 
tion: close.....<메소드 호출>. jt_userinfo< 
/name>.....请求中的body数据是xml数据格式。 
2,服务端$_POST打印了个httpentitybody不一致了。这跟上个例子不一样了, 다양한 콘텐츠 유형 text/xml,而不是application/x-www-form-urlencoded 
3,而php://input数据还是跟http 엔터티 본문数据一致。也就是php://input数据和$_POST数据不一致了。

저희는 GET 방식을 사용하고 있습니다. put_server.php文件,将$ _POST改成$_GET。 

<?php 
//@file phpinput_server.php 
$raw_post_data = file_get_contents(&#39;php://input&#39;, &#39;r&#39;); 
echo "-------\$_GET------------------\n"; 
echo var_dump($_GET) . "\n"; 
echo "-------php://input-------------\n"; 
echo $raw_post_data . "\n"; 
?> 
<?php 
//@file phpinput_get.php 
$query_path = &#39;n=&#39; . urldecode(&#39;perfgeeks&#39;) . &#39;&p=&#39; . urldecode(&#39;7788&#39;); 
$host = &#39;192.168.0.6&#39;; 
$port = 80; 
$path = &#39;/phpinput_server.php&#39;; 
$d = &#39;&#39;; 
$fp = fsockopen($host, $port, $error_no, $error_desc, 30); 
if ($fp) { 
fputs($fp, "GET {$path}?{$query_path} HTTP/1.1\r\n"); 
fputs($fp, "Host: {$host}\r\n"); 
fputs($fp, "Connection: close\r\n\r\n"); 

while (!feof($fp)) { 
$d .= fgets($fp, 4096); 
} 
fclose($fp); 
echo $d; 
} 
?>
로그인 후 복사

同样,我们执行下一phpinput_get.php测试脚本,它模拟了一个通常情况下的GET方法提交表单数据。

@php /phpinput_get.php
HTTP/1.1 200 OK
Date: Thu, 08 Apr 2010 07:38:15 GMT
Server: Apache/2.2.3 (CentOS)
X-Powered-By: PHP/5.1.6
Content-Length: 141
Connection: close
Content-Type: text/html; charset=UTF-8

-------$_GET------------------
array(2) {
["n"]=>
string(9) "perfgeeks"
["p"]=>
string(4) "7788"
}

-------php://input-------------
在这个时候,使用ngrep工具,捕获的相应的http请求数据包如下

T 192.168.0.8:36775 -> 192.168.0.6:80 [AP]
GET /phpinput_server.php?n=perfgeeks&p=7788 HTTP/1.1..
Host: 192.168.0.6..Connection: close....
比较POST方法提交的http请求,通常GET方法提交的请求中,entity body为空。同时,不会指定Content-Type和Content-Length。但是,如果强硬数据http entity body,并指明正确地Content-Type和Content-Length,那么php://input还可是读取得到http entity body数据,但不是$_GET数据。

所根据,上面几个探测,我们可以作出以下总结:
1,Content- Type取值为application/x-www-form-urlencoded时,php会将http请求body相应数据会填入到数 组$_POST,填入到$_POST数组中的数据是进行urldecode()解析的结果。(其实,除了该Content-Type,还有 multipart/form-data表示数据是表单数据,稍后我们介绍)
2,php://input数据,只要Content-Type不为 multipart/form-data(该条件限制稍后会介绍)。那么php://input数据与http entity body部分数据是一致的。该部分相一致的数据的长度由Content-Length指定。
3,仅当Content-Type为application/x-www-form-urlencoded且提交方法是POST方法时,$_POST数据与php://input数据才是”一致”(打上引号,表示它们格式不一致,内容一致)的。其它情况,它们都不一致。
4,php://input读取不到$_GET数据。是因为$_GET数据作为query_path写在http请求头部(header)的PATH字段,而不是写在http请求的body部分。

这也帮助我们理解了,为什么xml_rpc服务端读取数据都是通过file_get_contents(‘php://input', ‘r')。而不是从$_POST中读取,正是因为xml_rpc数据规格是xml,它的Content-Type是text/xml。
php://input碰到了multipart/form-data

上传文件的时候,表单的写法是这样的

<form enctype="multipart/form-data" action="phpinput_server.php" method="POST" > 
<input type="text" name="n" /> 
<input type="file" name="f" /> 
<input type="submit" value="upload now" /> 
</form>
로그인 후 복사

那么,enctype=multipart/form-data这里的意义,就是将该次http请求头部(head)中的Content-Type设置为multipart/form-data。请查阅RFC1867对 它的描述。multipart/form-data也表示以POST方法提交表单数据,它还伴随了文件上传,所以会跟application/x- www-form-urlencoded数据格式不一样。它会以一更种更合理的,更高效的数据格式传递给服务端。我们提交该表单数据,并且打印出响应结 果,如下:

-------$_POST------------------
array(1) { ["n"]=> string(9) "perfgeeks" }
-------php://input-------------
同时,我们通过ngrep抓取的相应的http请求数据包如下:

########
T 192.168.0.8:3981 -> 192.168.0.6:80 [AP]
POST /phpinput_server.php HTTP/1.1..Host: 192.168.0.6..Connection: kee
p-alive..User-Agent: Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US) A
ppleWebKit/533.2 (KHTML, like Gecko) Chrome/5.0.342.3 Safari/533.2..Re
ferer: http://192.168.0.6/phpinput_server.php..Content-Length: 306..Ca
che-Control: max-age=0..Origin: http://192.168.0.6..Content-Type: mult
ipart/form-data; boundary=----WebKitFormBoundarybLQwkp4opIEZn1fA..Acce
pt: application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q
=0.8,image/png,*/*;q=0.5..Accept-Encoding: gzip,deflate,sdch..Accept-L
anguage: zh-CN,zh;q=0.8..Accept-Charset: GBK,utf-8;q=0.7,*;q=0.3..Cook
ie: SESS3b0e658f87cf58240de13ab43a399df6=lju6o5bg8u04lv1ojugm2ccic6...
.
##
T 192.168.0.8:3981 -> 192.168.0.6:80 [AP]
------WebKitFormBoundarybLQwkp4opIEZn1fA..Content-Disposition: form-da
ta; name="n"....perfgeeks..------WebKitFormBoundarybLQwkp4opIEZn1fA..C
ontent-Disposition: form-data; name="f"; filename="test.txt"..Content-
Type: text/plain....i am file..multipart/form-data..------WebKitFormBo
undarybLQwkp4opIEZn1fA--..
##
从响应输出来比对,$_POST数据跟请求提交数据相符,即$_POST = array(‘n' => ‘perfgeeks')。这也跟http请求body中的数据相呼应,同时说明PHP把相应的数据填入$_POST全局变量。而php://input 输出为空,没有输出任何东西,尽管http请求数据包中body不为空。这表示,当Content-Type为multipart/form-data的 时候,即便http请求body中存在数据,php://input也为空,PHP此时,不会把数据填入php://input流。所以,可以确定: php://input不能用于读取enctype=multipart/form-data数据。

我们再比较这次通过ngrep抓取的http请求数据包,我们会发现,最大不同的一 点是Content-Type后面跟了boundary定义了数据的分界符,bounday是随机生成的。另外一个大不一样的,就是http entity body中的数据组织结构不一样了。

上一节,我们概述了,当Content-Type为application/x- www-form-urlencoded时,php://input和$_POST数据是“一致”的,为其它Content-Type的时候,php: //input和$_POST数据数据是不一致的。因为只有在Content-Type为application/x-www-form- urlencoded或者为multipart/form-data的时候,PHP才会将http请求数据包中的body相应部分数据填入$_POST全 局变量中,其它情况PHP都忽略。而php://input除了在数据类型为multipart/form-data之外为空外,其它情况都可能不为空。 通过这一节,我们更加明白了php://input与$_POST的区别与联系。所以,再次确认,php://input无法读取 enctype=multipart/form-data数据,当php://input遇到它时,永远为空,即便http entity body有数据。
php://input VS $http_raw_post_data

相信大家对php://input已经有一定深度地了解了。那 么$http_raw_post_data是什么呢?$http_raw_post_data是PHP内置的一个全局变量。它用于,PHP在无法识别的 Content-Type的情况下,将POST过来的数据原样地填入变量$http_raw_post_data。它同样无法读取Content- Type为multipart/form-data的POST数据。需要设置php.ini中的 always_populate_raw_post_data值为On,PHP才会总把POST数据填入变量$http_raw_post_data。

把脚本phpinput_server.php改变一下,可以验证上述内容

<?php 
$raw_post_data = file_get_contents(&#39;php://input&#39;, &#39;r&#39;); 
$rtn = ($raw_post_data == $HTTP_RAW_POST_DATA) ? 1 : 0; 
echo $rtn; 
?>
로그인 후 복사

执行测试脚本 

@php phpinput_post.php 
@php phpinput_get.php 
@php phpinput_xmlrpc.php 

得出的结果输出都是一样的,即都为1,表示php://input和$HTTP_RAW_POST_DATA是相同的。至于对内存的压力,我们这里就不做细致地测试了。有兴趣的,可以通过xhprof进行测试和观察。 

以此,我们这节可以总结如下: 
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的数据。 
学习笔记 

1,Coentent-Type仅在取值为application/x-www-data-urlencoded和multipart/form-data两种情况下,PHP才会将http请求数据包中相应的数据填入全局变量$_POST 
2,PHP不能识别的Content-Type类型的时候,会将http请求包中相应的数据填入变量$HTTP_RAW_POST_DATA 
3, 只有Coentent-Type不为multipart/form-data的时候,PHP不会将http请求数据包中的相应数据填入php://input,否则其它情况都会。填入的长度,由Coentent-Length指定。 
4,只有Content-Type为application/x-www-data-urlencoded时,php://input数据才跟$_POST数据相一致。 
5,php://input数据总是跟$HTTP_RAW_POST_DATA相同,但是php://input比$HTTP_RAW_POST_DATA更凑效,且不需要特殊设置php.ini 
6,PHP会将PATH字段的query_path部分,填入全局变量$_GET。通常情况下,GET方法提交的http请求,body为空。

更多PHP输入流php://input介绍相关文章请关注PHP中文网!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)