php://协议
首先,我们来说一下一个php提供的协议:"php://"
或许有人看到这个会懵逼,这是什么东东?这有啥用?这咋用?我是谁?我在哪?我要去往何处?
恩,大家可以翻开php手册,搜索一下
php://是php内置的一个类url操作的协议,它运行我们访问php各种I/O流,至于什么是I/O流,举个例子:
echo "hello world"; 这个字符串会经过php的处理,最后输出到用户端/控制台,而这个就是"hello world"就是 输出的 流,程序把这个字符串处理成一串串的二进制,输出到了用户端/控制台,这样的字符串就叫做输出流
同理,用户发起一个post请求,将数据传给服务器,服务器接收,这样的字符串就叫做输入流
好了,该协议的用法先放着,我们来看看php的各种I/O流
STDIN 输入流
STDIN输入流为 php的标准输入流,一般是指键盘输入到程序缓冲区的数据
在php中,主要是指在php-cli运行模式下,用户使用键盘输入到控制台的数据,例如:(注意,需要使用php-cli模式)
1
2
3
4
5
6
7
8
9
10
11
|
<?php /** * Created by PhpStorm.
* User: tioncico
* Date: 18-10-20
* Time: 下午5:20
*/
echo "请输入你的名字:\n" ;
$stdin = fopen ( "php://stdin" , ‘r‘ );
$data = fgets ( $stdin );
echo "{$data}大哥,你好啊!" ;
|
可看到,上面使用了php://stdin 协议,打开了一个标准输入的操作句柄,然后可读取用户在控制台输入的数据
上面的代码也可写成:
1
2
3
4
5
6
7
8
9
10
|
<?php /** * Created by PhpStorm.
* User: tioncico
* Date: 18-10-20
* Time: 下午5:20
*/
echo "请输入你的名字:\n" ;
$data = trim( fgets (STDIN));
echo "{$data}大哥,你好啊!" ;
|
STDIN常量是一个已经打开的stdin流,可节省几行代码,也可节省小部分打开stdin的性能
当然,值得注意的点是:
1:php://stdin打开的其实是STDIN常量已经打开的复制,所以就算关闭了php://stdin,也是关闭的复制,STDIN已打开的不会被关闭
2:php://stdin是只读的
3:请直接使用STDIN常量,而不使用php://stdin
4:使用fgets读取,只能读取一行数据(检测到回车就返回),可以使用stream_get_contents 控制读取数量,用于一次性读取包含换行的输入数据
STDOUT输出流
STDOUT和STDIN正好相反,是标准输出流,它将运行php将字符串默认输出到控制台(可使用freopen重定向到文件),例如:(注意,使用php-cli模式运行)
1
2
3
4
5
6
7
8
9
10
11
|
<?php /** * Created by PhpStorm.
* User: tioncico
* Date: 18-10-20
* Time: 下午5:20
*/
$stdout = fopen ( "php://stdout" , ‘w‘ );
fwrite( $stdout , "这是输出1\n" );
echo "这是输出2\n" ;
fwrite(STDOUT, "这是输出3\n" );
|
和stdin注意点一样,这里不复制了,但需要加上一条:
在php-cli模式中,作用和echo相同,但是实现方式不一样
STDERR 标准错误
STRERR 标准错误和STDOUT差不多,都是将字符串默认打印到控制台(可使用freopen重定向到文件),但是这个是打印错误用的,区分就是STDERR会将打印的字符串变成红色(需要终端支持)例如:(注意,使用php-cli模式运行)
1
2
3
4
5
6
7
8
9
10
11
12
|
<?php /** * Created by PhpStorm.
* User: tioncico
* Date: 18-10-20
* Time: 下午5:20
*/
$stderr = fopen ( "php://stderr" , ‘w‘ );
fwrite( $stderr , "这是输出1\n" );
echo "这是输出2\n" ;
fwrite(STDERR, "这是输出3\n" );
$a = $b ;
|
从上面的STDOUT注意点到这里可以发现,STDERR和echo语句的输出顺序是不相同的,个人猜测是echo的缓冲区和STDERR不相同
php://input
从这里开始,就和上面的3个程序标准I/O流关系不大啦,咱们继续往下看,该部分参考:http://www.nowamagic.net/academy/detail/12220520,有时间可详细阅读,本文只做功能简单介绍
php://input 是个可以访问请求的原始数据的只读流。
通俗来讲:php://input可接收用户请求过来的原始数据流(大多数时候作用于POST请求),例如:(需要使用web方式请求)
服务端代码:
1
2
3
4
5
6
7
|
<?php $data = file_get_contents ( "php://input" , ‘r‘ );
echo "下面是php://input\n" ;
var_dump( $data );
echo 1;
echo "下面是 POST:\n" ;
var_dump( $_POST );
|
客户端请求代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
<?php /** * Created by PhpStorm.
* User: tioncico
* Date: 18-10-27
* Time: 下午11:06
*/
function send_post( $url , $post_data ) {
$postdata = http_build_query( $post_data );
$options = array (
‘http‘ => array (
‘method‘ => ‘POST‘ ,
‘header‘ => ‘Content-type:application/x-www-form-urlencoded‘ ,
‘content‘ => $postdata ,
‘timeout‘ => 15 * 60 // 超时时间(单位:s)
)
);
$context = stream_context_create( $options );
$result = file_get_contents ( $url , false, $context );
return $result ;
} //使用方法 $post_data = array (
‘username‘ => ‘tioncico‘ ,
‘password‘ => ‘123456‘
); $data = send_post( ‘http://test.cn‘ , $post_data );
var_dump( $data );
|
输出:
php://output
php://output 是一个只写的数据流, 允许你以 print 和 echo 一样的方式 写入到输出缓冲区。可以说功能和echo 一样(可以在web和php-cli使用)
1
2
|
<?php file_put_contents ( "php://output" , "仙士可最帅" );
|
php://fd
php://fd 允许直接访问指定的文件描述符。 例如 php://fd/3 引用了文件描述符 3。用法:
在linux中,一切皆文件,当启动系统时,先会启动STDIN标准输入(文件描述符0),之后是STDOUT文件描述符1,STDERR文件描述符2
使用php://fd,可直接调用该文件:
1
2
3
4
5
6
|
<?php $stdin = fopen ( "php://fd/0" , ‘r‘ );
$data = fgets ( $stdin );
echo "这是STDIN输入的:{$data}\n" ;
file_put_contents ( "php://fd/2" , "这是STDERR\n" );
file_put_contents ( "php://fd/1" , "这是STDOUT\n" );
|
php://memory 和 php://temp
php://memory 和 php://temp 是一个类似文件 包装器的数据流,允许读写临时数据。 两者的唯一区别是 php://memory 总是把数据储存在内存中, 而 php://temp 会在内存量达到预定义的限制后(默认是 2MB)存入临时文件中。 临时文件位置的决定和 sys_get_temp_dir() 的方式一致。 php://temp 的内存限制可通过添加 /maxmemory:NN 来控制,NN 是以字节为单位、保留在内存的最大数据量,超过则使用临时文件。例如:
1
2
3
4
5
6
7
8
9
10
11
12
|
<?php // Set the limit to 5 MB. $fiveMBs = 5 * 1024 * 1024;
$temp_f = fopen ( "php://temp/maxmemory:$fiveMBs" , ‘r+‘ );//maxmemory可以调整存储空间大小
fputs ( $temp_f , "hello,仙士可,我是temp的数据\n" );
rewind ( $temp_f ); //由于写入数据时,指针已经到了末尾,需要重置指针才能读取到数据
echo stream_get_contents( $temp_f );
$fiveMBs = 5 * 1024 * 1024;
$memory_f = fopen ( "php://memory" , ‘r+‘ );//maxmemory可以调整存储空间大小
fputs ( $memory_f , "hello,仙士可,我是memory的数据\n\n" );
rewind ( $memory_f ); //由于写入数据时,指针已经到了末尾,需要重置指针才能读取到数据
echo stream_get_contents( $memory_f );
|
注意:关闭文件之后数据将无法读取
php://filter
php://filter 是一种元封装器, 设计用于数据流打开时的筛选过滤应用。 这对于一体式(all-in-one)的文件函数非常有用,类似 readfile()、 file() 和 file_get_contents(), 在数据流内容读取之前没有机会应用其他过滤器。
php://filter 目标使用以下的参数作为它路径的一部分。 复合过滤链能够在一个路径上指定。详细使用这些参数可以参考具体范例。
名称 | 描述 |
---|---|
resource=<要过滤的数据流> | 这个参数是必须的。它指定了你要筛选过滤的数据流。 |
read=<读链的筛选列表> | 该参数可选。可以设定一个或多个过滤器名称,以管道符(|)分隔。 |
write=<写链的筛选列表> | 该参数可选。可以设定一个或多个过滤器名称,以管道符(|)分隔。 |
<;两个链的筛选列表> | 任何没有以 read= 或 write= 作前缀 的筛选器列表会视情况应用于读或写链。 |
可选项
属性 | 支持 |
---|---|
受限于 allow_url_fopen | No |
受限于 allow_url_include | 仅 php://input、 php://stdin、 php://memory 和 php://temp。 |
允许读取 | 仅 php://stdin、 php://input、 php://fd、 php://memory 和 php://temp。 |
允许写入 | 仅 php://stdout、 php://stderr、 php://output、 php://fd、 php://memory 和 php://temp。 |
允许追加 | 仅 php://stdout、 php://stderr、 php://output、 php://fd、 php://memory 和 php://temp(等于写入) |
允许同时读写 | 仅 php://fd、 php://memory 和 php://temp。 |
支持 stat() | 仅 php://memory 和 php://temp。 |
支持 unlink() | No |
支持 rename() | No |
支持 mkdir() | No |
支持 rmdir() | No |
仅仅支持 stream_select() | php://stdin、 php://stdout、 php://stderr、 php://fd 和 php://temp。 |
该协议个人不是很会用,并且个人觉得没啥必要学习,所以直接复制php手册的例子吧
1
2
3
4
5
6
|
<?php /* 这会以大写字母输出 www.example.com 的全部内容 */ readfile( "php://filter/read=string.toupper/resource=http://www.example.com" );
/* 这会和以上所做的一样,但还会用 ROT13 加密。 */ readfile( "php://filter/read=string.toupper|string.rot13/resource=http://www.example.com" );
|