PHP Wrapper在SAE上的应用方法(2)

//包含附件代码,注册saemc Wrapper include_once('wrapper.php'); //测试 saemc Wrapper $fp = fopen( "saemc://test.txt", "w+" ) or die("fopen faild!"); fwrite( $fp, "line1\n" ) or die("fwrite line1 faild!"); fwrite( $fp, "line2\n" ) or die("fwrite line2 faild!"); fwrite( $fp, "line3\n" ) or die("fwrite line3 faild!"); var_dump(ftell($fp)); fseek( $fp, 0 ); while ( !feof( $fp ) ) { $c = fgets( $fp ) or die("fgets faild!"); var_dump($c); } fclose( $fp ); var_dump(file_get_contents("saemc://test.txt")); var_dump(file_put_contents("saemc://path/test.txt", "hello world!\n")); var_dump(file_put_contents("saemc://path/test.txt", "hello world!\n", FILE_APPEND)); var_dump(file_get_contents("saemc://path/test.txt")); var_dump(copy("saemc://path/test.txt", "saemc://path/test_new.txt")); var_dump(file_get_contents("saemc://path/test_new.txt")); var_dump(unlink("saemc://path/test.txt")); var_dump(file_get_contents("saemc://path/test.txt")); var_dump(rename("saemc://path/test_new.txt", "saemc://path/test.txt")); var_dump(file_get_contents("saemc://path/test.txt")); echo "====test include====\n"; include_once("saemc://path/test.txt");

测试页面的输出结果:

int(18) string(6) "line1 " string(6) "line2 " string(6) "line3 " string(18) "line1 line2 line3 " int(13) int(13) string(26) "hello world! hello world! " bool(true) string(26) "hello world! hello world! " bool(true) bool(false) bool(true) string(26) "hello world! hello world! " ====test include==== hello world! hello world!

我们提供的 Memcache Wrapper并没有实现目录操作的一些方法和Memcache的Timeout,大家可以参考PHP手册,尝试实现目录操作,或者通过context使这个Wrapper支持Memcache的Timeout。

另外,大家可以到下面这个地址查看SAE Stdlib中sae_include的源码,在其中还有我们为Storage服务封装的saestor Wrapper和为Fetchurl服务重新封装的http Wrapper的实现:

四、写Wrapper时的一些注意事项

1. 构造函数

streamWrapper 类很特别,它的构造函数并不是每次都调用的。只有在你的操作触发了stream_open相关的操作时才会调用,比如你用file_get_contents()了。而当你的操作触发和stream无关的函数时,比如file_exists会触发url_stat方法,这个时候构造函数是不会被调用的。

2. 读实现

Wrapper里边有Position和Seek等概念,但是很多服务其实是一次性就读取全部数据的,这个可以在stream_open的时候一次性读回,放到一个属性中,以后seek和tell的时候直接操作属性里边存放的数据就可以了。

3. 追加写实现

有很多服务是一次性写入所有数据,不支持追加写的功能(比如Memcache),这就需要我们自己在Wrapper中来实现追加写。可以将整个value一次性读取出来,将需要追加写的数据追加在读取出来的内容后面之后,再一次性写回。

但是这种追加写的实现方式性能会比较差,尤其是内容体积较大之后,一次性读取所有内容会非常消耗资源,因此在某些服务中我们不得不舍弃对追加写的支持。

4. url_stat的实现

在streamWrapper类的实现中,url_stat的实现是个难点。必须正确的实现url_stat才能使is_writable和is_readable等查询文件元信息的函数正常工作。

而我们需要为我们的虚设备伪造这些值。以mc为例,我们给大家一些参考数据:

url_stat应该返回一个数组,分13个项,内容如下:

dev 设备号 - 写0即可;
ino inode号 - 写0即可;
mode 文件mode - 这个是文件的权限控制符号,稍后详细说明;
nlink link - 写0即可;
uid uid - Linux上用posix_get_uid可以取到,windows上为0;
gid gid - Linux上用posix_get_gid可以取到,windows上为0;
rdev 设备类型 - 当为inode设备时有值;
size - 文件大小;
atime - 最后读时间 格式为unix时间戳;
mtime - 最后写时间;
ctime - 创建时间;
blksize - blocksize of filesystem IO 写零即可;
blocks - number of 512-byte blocks allocated 写零即可;

其中mode的值必须写对:

如果是文件,其值为:

0100000 + 文件权限,如 0100000 + 0777。

如果是目录,其值为:

040000 + 目录权限,如 0400000 + 0777。

5. 关于stat的缓存

PHP会在同一个页面的执行过程中对文件的元信息进行缓存。
根据PHP文档对 clearstatcache() 这个方法的说明得知:在使用 stat(), lstat(), file_exists(), is_writable(), is_readable(), is_executable(), is_file(), is_dir(), is_link(), filectime(), fileatime(), filemtime(), fileinode(), filegroup(), fileowner(), filesize(), filetype(), 或 fileperms() 方法查询文件信息时,PHP会将文件的stat的缓存以提高性能。 clearstatcache()方法可以用来清除这个缓存,当unlink()会自动清除stat缓存。

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/39d735735d6474a6907e6f525459836b.html