深入理解PHP中的static和yield關(guān)鍵字
前言
本文主要給大家介紹了關(guān)于PHP中static和yield關(guān)鍵字的相關(guān)內(nèi)容,分享出來(lái)供大家參考學(xué)習(xí),下面話(huà)不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹吧。
先來(lái)說(shuō)說(shuō) static 關(guān)鍵字。本篇只講靜態(tài)方法的使用與后期綁定的知識(shí)點(diǎn)。
static 什么時(shí)候用來(lái)修飾方法
static 關(guān)鍵字大家都知道是用來(lái)修飾方法與屬性的。 那么大家在項(xiàng)目中會(huì)在哪些場(chǎng)景下使用它?
我遇到過(guò)幾個(gè)項(xiàng)目,要求所有的方法全部 static 化,當(dāng)然控制器方法不能這么干。原因之一就是:靜態(tài)方法執(zhí)行效率高?那么我們基于此來(lái)分析一下。
首先執(zhí)行效率高我是沒(méi)有意見(jiàn)的。哪么是不是因?yàn)樗矢?,就該毫無(wú)節(jié)制的使用在項(xiàng)目中?討論這個(gè)問(wèn)題先來(lái)回顧下編程語(yǔ)言的歷史。在早一點(diǎn)的時(shí)候,還沒(méi)有面向?qū)ο螅捎玫亩际墙Y(jié)構(gòu)化編程,當(dāng)時(shí)基本上所有的方法都是 靜態(tài)方法,然后有了面向?qū)ο?,產(chǎn)生了實(shí)例化的概念。
從上面簡(jiǎn)短的發(fā)展過(guò)程可以看出,如果僅僅為了性能,哪么面向?qū)ο蠛孟駴](méi)有存在的必要。那么這些大師為了要在 c++ java 這些語(yǔ)言中引入面向?qū)ο蟆⒁雽?shí)例化的感念呢?我覺(jué)得是因?yàn)榘殡S發(fā)展,項(xiàng)目越來(lái)越大,需要更好的組織代碼方式與編程思維。
再回過(guò)頭來(lái)看 static ,它定義的靜態(tài)方法,效率確實(shí)高,但是會(huì)持續(xù)占用內(nèi)存,只有在程序退出時(shí)才結(jié)束生命周期,期間無(wú)法進(jìn)行銷(xiāo)毀等副作用是其一;其二從設(shè)計(jì)模式上來(lái)說(shuō),它具有強(qiáng)耦合性,外部可修改 static 屬性;其三static定義的方法沒(méi)有辦法override來(lái)重寫(xiě),ioc di等概念無(wú)用武之地;其四在進(jìn)行單元測(cè)試時(shí),靜態(tài)方法讓人頭痛。
那么通過(guò)上面所說(shuō),感覺(jué)以后還是別用 static 方法了,老老實(shí)實(shí)的實(shí)例化然后調(diào)用方法?咱們得理性,不能極端到什么地方都用,也不能一丁點(diǎn)都不用。一句話(huà):學(xué)會(huì)面向?qū)ο蟮姆绞絹?lái)思考。我們寫(xiě)代碼的第一考慮點(diǎn)我覺(jué)得是:可擴(kuò)展性(應(yīng)對(duì)業(yè)務(wù)快速變化),可維護(hù)性(線(xiàn)上問(wèn)題及時(shí)修復(fù))。高效率應(yīng)該是最后再來(lái)考慮(因?yàn)閮?yōu)化效率的手段非常之多,并不一定非要給每個(gè)方法加個(gè): static)。如果從面向?qū)ο蟮慕嵌瘸霭l(fā),這個(gè)方法完全獨(dú)立跟類(lèi)屬性無(wú)關(guān),那么就用 static 吧。
總之是站在面向?qū)ο蟮慕嵌?,軟件設(shè)計(jì)的層次來(lái)考慮語(yǔ)法的使用,而不是為了效率破壞掉代碼的美。
static 后期靜態(tài)綁定
這一點(diǎn)php的文檔做了詳細(xì)的介紹,但是我以前一直很少關(guān)注這個(gè)地方,基本上都是使用 self:: 的方式進(jìn)行靜態(tài)方法與屬性的調(diào)用。
我覺(jué)得后期綁定某種程度上,像是靜態(tài)方法的重載。這里貼出 php 文檔中的例子來(lái)進(jìn)行一下講述
<?php class A { public static function who() { echo __CLASS__; } public static function test() { self::who(); static::who();// 后期靜態(tài)綁定 } } class B extends A { public static function who() { echo __CLASS__; } } B::test();
如果是 self::who()
調(diào)用,會(huì)輸出:A。如果是 static::who()
會(huì)輸出 B
這樣來(lái)看,是不是相當(dāng)于 class B重寫(xiě)了父類(lèi) A 的 who()
方法?那么如果靈活使用這個(gè)特性,可以讓 static 具備更強(qiáng)的靈活性。充分發(fā)揮其性能優(yōu)勢(shì),又能解決擴(kuò)展性差的問(wèn)題。當(dāng)然還是一樣,要從面向?qū)ο蟮慕嵌瘸霭l(fā),一切適可而止。
PHP 中 yield 的使用場(chǎng)景
說(shuō)實(shí)話(huà),很長(zhǎng)一段時(shí)間我并不知道 php 還有這么個(gè)語(yǔ)法。直到有一天我在 js 中遇到了這個(gè)關(guān)鍵字,感覺(jué)這么不明覺(jué)厲的東西,世界上最好的語(yǔ)言怎么沒(méi)有?回頭看文檔,真有,不愧為世界上最好的語(yǔ)言。
那么 yield 的使用場(chǎng)景是什么?剛好最近有人 sg 上面問(wèn)道我,借此整理一下。希望大家能夠?qū)⑺嗟慕Y(jié)合自己的業(yè)務(wù)進(jìn)行使用。這里不會(huì)進(jìn)行 yield 與 Iterator 的比較。相信看完后,你能夠明了二者的誰(shuí)更簡(jiǎn)介。
先說(shuō)它的使用場(chǎng)景,還是得先回顧歷史,在沒(méi)有 yield 之前,我們要生成一個(gè)數(shù)組,只能一次性把所有內(nèi)容全部讀入內(nèi)存(當(dāng)然也可以通過(guò)實(shí)現(xiàn) Iterator接口實(shí)現(xiàn)一個(gè)迭代)。有了 yield 之后,我們可以通過(guò)一個(gè)簡(jiǎn)單的 yield 關(guān)鍵字,完成一個(gè)數(shù)組的生成,并且是用到的時(shí)候才會(huì)產(chǎn)生值,相對(duì)而言?xún)?nèi)存占用肯定會(huì)下降??湛跓o(wú)憑,咱們下面通過(guò)代碼實(shí)際檢驗(yàn)一下上面的結(jié)論。
先來(lái)看普通模式
<?php function generateData($max) { $arr = []; for ($i = 0; $i <= $max; $i++) { $arr[] = $i; } } echo '開(kāi)始前內(nèi)存占用:' . memory_get_usage() . PHP_EOL; $data = generateData(100000); echo '生成完數(shù)組后內(nèi)存占用:' . memory_get_usage() . PHP_EOL; unset($data); echo '釋放后的內(nèi)存占用:' . memory_get_usage() . PHP_EOL;
運(yùn)行得到結(jié)果:
開(kāi)始前內(nèi)存占用:231528 生成完數(shù)組后內(nèi)存占用:231712 釋放后的內(nèi)存占用:231576
前后的差值是:184
使用yield后的效果
function generateData($max) { for ($i = 0; $i <= $max; $i++) { yield $i; } } echo '開(kāi)始前內(nèi)存占用:' . memory_get_usage() . PHP_EOL; $data = generateData(100000);// 這里實(shí)際上得到的是一個(gè)迭代器 echo '生成完數(shù)組后內(nèi)存占用:' . memory_get_usage() . PHP_EOL; unset($data); echo '釋放后的內(nèi)存占用:' . memory_get_usage() . PHP_EOL;
運(yùn)行結(jié)果:
開(kāi)始前內(nèi)存占用:228968 生成完數(shù)組后內(nèi)存占用:229824 釋放后的內(nèi)存占用:229016
前后的差值是:856
奇怪,使用了 yield 后,內(nèi)存占用反而上升了,這是什么鬼?別急。上面我們參數(shù)傳入的是 1,000,00,我現(xiàn)在將傳入?yún)?shù)改成改成 1,000,000試試。
第一個(gè)方法得到的結(jié)果是:
開(kāi)始前內(nèi)存占用:231528 Fatal error: Allowed memory size of 134217728 bytes exhausted (tried to allocate 32 bytes) in /test/yield.php on line 6
看了吧,一百萬(wàn)次的循環(huán)時(shí),一次性載入內(nèi)存,超出了限制。那么再來(lái)看 yield 的執(zhí)行結(jié)果:
開(kāi)始前內(nèi)存占用:228968 生成完數(shù)組后內(nèi)存占用:229824 釋放后的內(nèi)存占用:229016
前后的差值依然是:856
好了到這里,應(yīng)該看出來(lái)了,yield無(wú)論數(shù)組大小,占用均是 856 ,這是因?yàn)樗陨?,它在你進(jìn)行迭代的時(shí)候才會(huì)產(chǎn)生真實(shí)數(shù)據(jù)。
所以如果你的數(shù)據(jù)來(lái)源非常大,那么用 yield 吧。如果數(shù)據(jù)來(lái)源很小,當(dāng)然選擇一次載入內(nèi)存。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
- PHP yield關(guān)鍵字功能與用法分析
- PHP5.5新特性之yield理解與用法實(shí)例分析
- php和C#的yield迭代器實(shí)現(xiàn)方法對(duì)比分析
- 分享五個(gè)PHP7性能優(yōu)化提升技巧
- PHP開(kāi)啟opcache提升代碼性能
- 逐步提升php框架的性能
- PHP 7安裝使用體驗(yàn)之性能大提升,兼容性強(qiáng),擴(kuò)展支持不夠(升級(jí)PHP要謹(jǐn)慎)
- 提高PHP編程效率 引入緩存機(jī)制提升性能
- 提升PHP性能的21種方法介紹
- 使用Huagepage和PGO來(lái)提升PHP7的執(zhí)行性能
- php使用yield對(duì)性能提升的測(cè)試實(shí)例分析
相關(guān)文章
php5與php7的區(qū)別點(diǎn)總結(jié)
在本篇文章里小編給大家整理的是關(guān)于php5與php7的區(qū)別是什么的相關(guān)知識(shí)點(diǎn)內(nèi)容,有需要的朋友們學(xué)習(xí)下。2019-10-10PHP pthreads v3在centos7平臺(tái)下的安裝與配置操作方法
這篇文章主要介紹了PHP pthreads v3在centos7平臺(tái)下的安裝與配置操作方法,結(jié)合圖文與實(shí)例形式分析了PHP pthreads v3在centos7平臺(tái)下的安裝與配置操作具體步驟、相關(guān)命令與注意事項(xiàng),需要的朋友可以參考下2020-02-02PHP實(shí)現(xiàn)Snowflake生成分布式唯一ID的方法示例
這篇文章主要給大家介紹了關(guān)于PHP實(shí)現(xiàn)Snowflake生成分布式唯一ID的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08php的mail函數(shù)發(fā)送UTF-8編碼中文郵件時(shí)標(biāo)題亂碼的解決辦法
這篇文章主要介紹了php的mail函數(shù)發(fā)送UTF-8編碼中文郵件時(shí)標(biāo)題亂碼的解決辦法,需要的朋友可以參考下2015-10-10有關(guān) PHP 和 MySQL 時(shí)區(qū)的一點(diǎn)總結(jié)
由于暫時(shí)使用國(guó)外的空間,在我發(fā)布 Blog 的時(shí)候發(fā)現(xiàn)時(shí)間總是不對(duì)。依據(jù)我以前編寫(xiě)程序的經(jīng)驗(yàn),這是時(shí)區(qū)的問(wèn)題。這個(gè)問(wèn)題解決起來(lái)并不難,寫(xiě)下我的解決途徑以便日后參考。2008-03-03