PHP 日本大会 2024

注册为流包装器的示例类

下面的示例实现了一个 var:// 协议处理程序,允许使用标准文件系统流函数(例如 fread())对命名的全局变量进行读写访问。下面实现的 var:// 协议,给定 URL “var://foo” 将读取/写入 $GLOBALS["foo"] 的数据。

示例 #1 用于读取/写入全局变量的流

<?php

class VariableStream {
var
$position;
var
$varname;

function
stream_open($path, $mode, $options, &$opened_path)
{
$url = parse_url($path);
$this->varname = $url["host"];
$this->position = 0;

return
true;
}

function
stream_read($count)
{
$ret = substr($GLOBALS[$this->varname], $this->position, $count);
$this->position += strlen($ret);
return
$ret;
}

function
stream_write($data)
{
$left = substr($GLOBALS[$this->varname], 0, $this->position);
$right = substr($GLOBALS[$this->varname], $this->position + strlen($data));
$GLOBALS[$this->varname] = $left . $data . $right;
$this->position += strlen($data);
return
strlen($data);
}

function
stream_tell()
{
return
$this->position;
}

function
stream_eof()
{
return
$this->position >= strlen($GLOBALS[$this->varname]);
}

function
stream_seek($offset, $whence)
{
switch (
$whence) {
case
SEEK_SET:
if (
$offset < strlen($GLOBALS[$this->varname]) && $offset >= 0) {
$this->position = $offset;
return
true;
} else {
return
false;
}
break;

case
SEEK_CUR:
if (
$offset >= 0) {
$this->position += $offset;
return
true;
} else {
return
false;
}
break;

case
SEEK_END:
if (
strlen($GLOBALS[$this->varname]) + $offset >= 0) {
$this->position = strlen($GLOBALS[$this->varname]) + $offset;
return
true;
} else {
return
false;
}
break;

default:
return
false;
}
}

function
stream_metadata($path, $option, $var)
{
if(
$option == STREAM_META_TOUCH) {
$url = parse_url($path);
$varname = $url["host"];
if(!isset(
$GLOBALS[$varname])) {
$GLOBALS[$varname] = '';
}
return
true;
}
return
false;
}
}

stream_wrapper_register("var", "VariableStream")
or die(
"注册协议失败");

$myvar = "";

$fp = fopen("var://myvar", "r+");

fwrite($fp, "line1\n");
fwrite($fp, "line2\n");
fwrite($fp, "line3\n");

rewind($fp);
while (!
feof($fp)) {
echo
fgets($fp);
}
fclose($fp);
var_dump($myvar);

?>

以上示例将输出

line1
line2
line3
string(18) "line1
line2
line3
"

添加笔记

用户贡献笔记 1 条笔记

22
pfe dot skh at gmail dot com
14 年前
与数据库一起使用的流示例

需求
一个 MySQL 数据库,其中包含名为 data 的表,结构如下:
CREATE TABLE IF NOT EXISTS `data` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`data` varchar(255) NOT NULL,
`when_inserted` datetime NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=MyISAM DEFAULT CHARSET=latin1 AUTO_INCREMENT=1;

现在使用流式实现

<?php

class DBStream {
private
$_pdo;
private
$_ps;
private
$_rowId = 0;

function
stream_open($path, $mode, $options, &$opath)
{
$url = parse_url($path);
$url['path'] = substr($url['path'], 1);
try{
$this->_pdo = new PDO("mysql:host={$url['host']};dbname={$url['path']}", $url['user'], isset($url['pass'])? $url['pass'] : '', array());
} catch(
PDOException $e){ return false; }
switch (
$mode){
case
'w' :
$this->_ps = $this->_pdo->prepare('INSERT INTO data VALUES(null, ?, NOW())');
break;
case
'r' :
$this->_ps = $this->_pdo->prepare('SELECT id, data FROM data WHERE id > ? LIMIT 1');
break;
default : return
false;
}
return
true;
}

function
stream_read()
{
$this->_ps->execute(array($this->_rowId));
if(
$this->_ps->rowCount() == 0) return false;
$this->_ps->bindcolumn(1, $this->_rowId);
$this->_ps->bindcolumn(2, $ret);
$this->_ps->fetch();
return
$ret;
}

function
stream_write($data)
{
$this->_ps->execute(array($data));
return
strlen($data);
}

function
stream_tell()
{
return
$this->_rowId;
}

function
stream_eof()
{
$this->_ps->execute(array($this->_rowId));
return (bool)
$this->_ps->rowCount();
}

function
stream_seek($offset, $step)
{
//无需实现
}
}

stream_register_wrapper('db', 'DBStream');

$fr = fopen('db://testuser@localhost/testdb', 'r');
$fw = fopen('db://testuser:testpassword@localhost/testdb', 'w');
//以上两种形式均被接受:对于前者,将使用默认密码 ""

$alg = hash_algos();
$al = $alg[array_rand($alg)];
$data = hash($al, rand(rand(0, 9), rand(10, 999))); //一些待写入的随机数据
fwrite($fw, $data); //将数据写入包装器
while($a = fread($fr, 256)){ //一个从包装器读取数据的循环
echo $a . '<br />';
}
?>

希望对您有帮助,干杯;)
To Top