程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> PHP綜合 >> PHP Wrapper在SAE上的應用方法

PHP Wrapper在SAE上的應用方法

編輯:PHP綜合

本文講述了PHP Wrapper在SAE上的應用方法。分享給大家供大家參考,具體如下:

一、PHP Wrapper是什麼

自PHP 4.3開始,PHP開始允許用戶通過stream_wrapper_register()自定義URL風格的協議。用戶使用fopen(), copy()等文件系統函數對封裝協議進行操作時,PHP會調用注冊協議時所提供的類中相應的函數。
PHP手冊中給了一個例子,它將VariableStream類注冊為var://協議,通過這個協議,用戶可以使用文件系統函數直接讀寫全局變量。例如,用戶可以通過 “var://foo” 讀寫 $GLOBALS['foo'] 。

二、SAE為什麼需要PHP Wrapper

出於性能和安全方面的考慮,SAE平台上禁用了本地文件讀寫和對外的數據抓取。相應的,我們提供了對應的服務來做同樣的事情。

由於新服務的接口和PHP本身的接口不太一樣,專門為我們平台開發的程序當然不會存在問題,但是大量已有的程序和開源項目,就面臨著繁雜的遷移工作。而使用PHP Wrapper對我們的服務的接口進行封裝之後,用戶就可以更方便地將程序遷移到SAE平台。

三、如何寫PHP Wrapper

要通過PHP Wrapper封裝一個協議,首先,我們需要寫一個 streamWrapper 類,類名可自定義,類的格式為:

streamWrapper {
public resource $context ;
__construct ( void )
public bool dir_closedir ( void )
public bool dir_opendir ( string $path , int $options )
public string dir_readdir ( void )
public bool dir_rewinddir ( void )
public bool mkdir ( string $path , int $mode , int $options )
public bool rename ( string $path_from , string $path_to )
public bool rmdir ( string $path , int $options )
public resource stream_cast ( int $cast_as )
public void stream_close ( void )
public bool stream_eof ( void )
public bool stream_flush ( void )
public bool stream_lock ( mode $operation )
public bool stream_open ( string $path , string $mode , int $options , string &$opened_path )
public string stream_read ( int $count )
public bool stream_seek ( int $offset , int $whence = SEEK_SET )
public bool stream_set_option ( int $option , int $arg1 , int $arg2 )
public array stream_stat ( void )
public int stream_tell ( void )
public int stream_write ( string $data )
public bool unlink ( string $path )
public array url_stat ( string $path , int $flags )
}

類中各方法說明:

streamWrapper::__construct — 構造函數,僅在stream_open前被調用
streamWrapper::dir_closedir — 關閉目錄句柄,響應closedir()函數
streamWrapper::dir_opendir — 打開目錄句柄,響應opendir()函數
streamWrapper::dir_readdir — 從目錄句柄讀取條目,響應readdir()函數
streamWrapper::dir_rewinddir — 倒回目錄句柄,響應rewinddir()函數
streamWrapper::mkdir — 創建目錄,響應mkdir()函數
streamWrapper::rename — 目錄或文件重命名,響應rename()函數
streamWrapper::rmdir — 刪除目錄,響應rmdir()函數
streamWrapper::stream_cast — 檢索基礎資源,響應stream_select()函數
streamWrapper::stream_close — 關閉資源,響應fclose()函數
streamWrapper::stream_eof — 檢查文件指針是否已經在文件末尾,響應feof()函數
streamWrapper::stream_flush — 清除輸出緩存,響應fflush()函數
streamWrapper::stream_lock — 咨詢文件鎖定,響應flock()函數
streamWrapper::stream_open — 打開文件或URL為流,響應fopen()函數
streamWrapper::stream_read — 從流中讀取內容,響應fread(), fgets()函數
streamWrapper::stream_seek — 在流中定位指針,響應fseek()函數
streamWrapper::stream_set_option — 改變流設置
streamWrapper::stream_stat — 檢索文件資源的信息,響應fstat()函數
streamWrapper::stream_tell — 檢索流中指針的位置,響應ftell()函數
streamWrapper::stream_write — 向流中寫入內容,響應fwrite(), fputs()函數
streamWrapper::unlink — 刪除文件,響應unlink()函數
streamWrapper::url_stat — 檢索文件的信息,響應所有stat()相關的函數,例如file_exists(), is_dir(), is_file(), filesize(), fileinode()等等

詳細說明請參考PHP手冊:http://cn2.php.net/manual/en/class.streamwrapper.php

寫好streamWrapper類之後,使用 stream_wrapper_register () 將這個類注冊到Wrapper中,就可以開始使用了。函數使用方法為:

bool stream_wrapper_register ( string $protocol , string $classname [, int $flags = 0 ] )

例如:

stream_wrapper_register("saemc", "SaeMemcacheWrapper");

由於SAE平台不支持對本地文件的寫操作,因此Smarty之類的一些需要在本地寫文件的開源項目就沒辦法直接在SAE平台上使用,而有了saemc Wrapper,用戶就可以將Smarty編譯的模板保存在MC中,很方便的將Smarty遷移到SAE平台上來。

在附件中我們為大家提供了SAE上Memcache Wrapper的實現代碼,大家可以下載此附件進行測試。

在測試之前,需要先在本地啟動一個端口為22222的Memcached服務:

memcached -m 10 -p 22222 -u nobody -l 127.0.0.1

然後使用下面代碼就可以測試了:

//包含附件代碼,注冊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的實現:

http://stdlib.sinaapp.com/?f=sae_include.function.php

四、寫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緩存。

而實際上,PHP只有在對本地文件進行unlink, rename和rmdir操作時會清除stat緩存,而在通過其他的wrapper進行unlink, rename和rmdir操作時,並不會清除stat緩存。因此在寫wrapper時我們要自己在unlink等方法中通過clearstatcache()來清除stat緩存。

點擊此處下載附件。

更多關於PHP相關內容感興趣的讀者可查看本站專題:《php curl用法總結》、《php socket用法總結》、《PHP網絡編程技巧總結》、《PHP基本語法入門教程》、《php操作office文檔技巧總結(包括word,excel,access,ppt)》、《php日期與時間用法總結》、《php面向對象程序設計入門教程》、《php字符串(string)用法總結》、《php+mysql數據庫操作入門教程》及《php常見數據庫操作技巧匯總》

希望本文所述對大家PHP程序設計有所幫助。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved