php的各种 I/O流 以及用法
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模式)
<?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 协议,打开了一个标准输入的操作句柄,然后可读取用户在控制台输入的数据
上面的代码也可写成:
<?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模式运行)
<?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模式运行)
<?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方式请求)
服务端代码:
<?php
$data = file_get_contents("php://input",'r');
echo "下面是php://input\n";
var_dump($data);
echo 1;
echo "下面是 POST:\n";
var_dump($_POST);
复制
客户端请求代码:
<?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使用)
<?php
file_put_contents("php://output","仙士可最帅");
复制
php://fd
php://fd 允许直接访问指定的文件描述符。 例如 php://fd/3 引用了文件描述符 3。用法:
在linux中,一切皆文件,当启动系统时,先会启动STDIN标准输入(文件描述符0),之后是STDOUT文件描述符1,STDERR文件描述符2 使用php://fd,可直接调用该文件:
<?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 是以字节为单位、保留在内存的最大数据量,超过则使用临时文件。例如:
<?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手册的例子吧
<?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");
复制
- 点赞
- 收藏
- 关注作者
评论(0)