本文實例講述了PHP流Streams、包裝器wrapper概念與用法。分享給大家供大家參考,具體如下:
流Streams這個概念是在php4.3引進的,是對流式數據的抽象,用于統一數據操作,比如文件數據、網絡數據、壓縮數據等,以使可以共享同一套函數,php的文件系統函數就是這樣的共享,比如file_get_contents()函數即可打開本地文件也可以訪問url就是這一體現。簡單點講,流就是表現出流式數據行為的資源對象。
以線性方式進行讀寫,并可以在流里面任意位置進行搜索。
流有點類似數據庫抽象層,在數據庫抽象層方面,不管使用何種數據庫,在抽象層之上都使用相同的方式操作數據,而流是對數據的抽象,它不管是本地文件還是遠程文件還是壓縮文件等等,只要來的是流式數據,那么操作方式就是一樣的。
有了流這個概念就引申出了包裝器wrapper這個概念,每個流都對應一種包裝器,流是從統一操作這個角度產生的一個概念,而包裝器呢是從理解流數據內容出發產生的一個概念,也就是這個統一的操作方式怎么操作或配置不同的內容;
這些內容都是以流的方式呈現,但內容規則是不一樣的,比如http協議傳來的數據是流的方式,但只有http包裝器才理解http協議傳來的數據的意思,可以這么理解,流就是一根流水的管子,只不過它流出的是數據,包裝器就是套在流這根管子外層的一個解釋者,它理解流出的數據的意思,并能操作它。
官方手冊說:“一個包裝器是告訴流怎么處理特殊協議或編碼的附加代碼”明白這句話的意思了嗎?
包裝器可以嵌套,一個流外面包裹了一個包裝器后,還可以在外層繼續包裹包裝器,這個時候里層的包裝器相對于外層的包裝器充當流的角色
在php自身底層實現的c語言開發文檔有這樣的解釋:
流API操作一對不同級別:在基本級別,api定義了php_stream對象表示流式數據源,在稍微高一點的級別,api定義了php_stream_wrapper對象。
它包裹低一級別的php_stream對象,以提供取回URL的內容和元數據、添加上下文參數的能力,調整包裝器行為;
每一種流打開后都可以應用任意數量的過濾器在上面,流數據會經過過濾器的處理,筆者認為過濾器這個詞用得有點不準確,有些誤導人。
從字面意思看好像是去掉一些數據的感覺,應該稱為數據調整器,因為它既可去掉一些數據,也可以添加,還可以修改,但歷史原因約定俗成,也就稱為過濾器了,大家心里明白就好。
我們經常看到下面的詞,來解釋下他們的區別:
資源和數據:資源是比較宏觀的說法,通常包含數據,而數據是比較具象的說法,在開發程序的時候經常說是數據,而在軟件規劃時說是資源,他們是近義詞,就像軟件設計和程序開發的區別一樣。
上下文和參數:上下文是比較宏觀的說法,經常用在溝通上面,具體點講就是一次溝通本身的參數,而參數這個說法往往用在比較具體的事情上面,比如說函數
上面解釋了概念性的東西,下面來看看具體內容:
php支持的協議和包裝器請看這里:http://php.net/manual/zh/wrappers.php:
(筆者注:原標題是:支持的協議和封裝協議,中文翻譯有點誤導,準確的講就是支持的協議和包裝器,從英文版面就很清楚)
默認的支持了一些協議和包裝器,請用stream_get_wrappers()函數查看.也可以自定義一個包裝器,用stream_wrapper_register()注冊
盡管RFC 3986里面可以使用:做分割符,但php只允許://,所以url請使用"scheme://target"這樣的格式
file:// — 訪問本地文件系統,在用文件系統函數時默認就使用該包裝器
http:// — 訪問 HTTP(s) 網址
ftp:// — 訪問 FTP(s) URLs
php:// — 訪問各個輸入/輸出流(I/O streams)
zlib:// — 壓縮流
data:// — 數據(RFC 2397)
glob:// — 查找匹配的文件路徑模式
phar:// — PHP 歸檔
ssh2:// — Secure Shell 2
rar:// — RAR
ogg:// — 音頻流
expect:// — 處理交互式的流
如何實現一個自定義的包裝器:
在用fopen、fwrite、fread、fgets、feof、rewind、file_put_contents、file_get_contents等等文件系統函數操作流時,數據是先傳給定義的包裝器類對象,包裝器再去操作流。
如何實現一個自定義的流包裝器呢?php提供了一個類原型,只是原型而已,不是接口也不是類,不能用于繼承:
streamWrapper { /* 屬性 */ public resource $context ; /* 方法 */ __construct ( void ) __destruct ( 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 ( int $operation ) public bool stream_metadata ( string $path , int $option , mixed $value ) 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 bool stream_truncate ( int $new_size ) public int stream_write ( string $data ) public bool unlink ( string $path ) public array url_stat ( string $path , int $flags ) }