PHP各版本中函數(shù)的類(lèi)型聲明詳解
PHP7開(kāi)始支持標(biāo)量類(lèi)型聲明,強(qiáng)類(lèi)型語(yǔ)言的味道比較濃。使用這個(gè)特性的過(guò)程中踩過(guò)兩次坑:一次是聲明boolean,最近是聲明double。為避免以后繼續(xù)犯類(lèi)似錯(cuò)誤,就把官方文檔翻了一次。本文是看完后對(duì)PHP函數(shù)的類(lèi)型聲明使用做的一次總結(jié)。
從語(yǔ)法上,PHP的函數(shù)定義經(jīng)過(guò)了幾個(gè)時(shí)期:
遠(yuǎn)古時(shí)代(PHP 4)
定義一個(gè)函數(shù)非常的簡(jiǎn)單,使用 function name(args) {body} 的語(yǔ)法聲明。不能指定參數(shù)和返回值類(lèi)型,參數(shù)和返回值類(lèi)型有無(wú)限種可能。這是到目前為止最常見(jiàn)的函數(shù)聲明方式。
數(shù)組和引用類(lèi)型參數(shù)值聲明(PHP 5)
數(shù)組(array)、類(lèi)(class)、接口(interface)、函數(shù)(callable)可以用在函數(shù)聲明中。從5.6開(kāi)始,支持常量(包括類(lèi)常量)為默認(rèn)參數(shù),以及參數(shù)數(shù)組(以省略號(hào)…為前綴)。例如:
function sum(...$numbers) { $sum = 0; foreach ($numbers as $number) { $sum += $number; } return $sum; }
注意:如果參數(shù)的值可能為null,null必須為參數(shù)的默認(rèn)值,否則調(diào)用時(shí)會(huì)出錯(cuò)。例如:
function foo(array $arr = null) { ... }
標(biāo)量類(lèi)型和返回值聲明(PHP 7)
函數(shù)正式支持標(biāo)量類(lèi)型(int, bool, float等)和返回值類(lèi)型(可聲明類(lèi)型同參數(shù))聲明。從這個(gè)版本開(kāi)始,寫(xiě)PHP有像寫(xiě)java的感覺(jué)。
遺憾是如果函數(shù)返回值有可能是null,就不能指定返回值類(lèi)型。例如:
function getModel() : Foo { if ($this->_model === null) { $this->_model = xxxx; // get from db or otherelse } return $this->_model; // 如果$this->_model仍是null,運(yùn)行出錯(cuò) }
參數(shù)和返回值可為null以及void返回類(lèi)型聲明(PHP 7.1)
當(dāng)參數(shù)和返回值類(lèi)型有可能是null時(shí),類(lèi)型前以問(wèn)號(hào)(?)修飾,可以解決null值問(wèn)題(與默認(rèn)參數(shù)不沖突);類(lèi)型聲明新增iterable,同時(shí)還支持void類(lèi)型返回值。例如:
function getModel(?int $id) : ?Foo { if ($id !== null) { $this->_model = xxxx; } else { $this->_model = yyyy; } return $this->_model; } // 調(diào)用 $foo->getModel(null); $foo->getModel(100); // 函數(shù)聲明了參數(shù)并且沒(méi)有提供默認(rèn)參數(shù),調(diào)用時(shí)不傳入?yún)?shù)會(huì)引發(fā)錯(cuò)誤 // 將函數(shù)聲明改成 getModel(?int $id = 100) {},可以不傳參數(shù) $foo->getModel();
當(dāng)函數(shù)返回值為void時(shí),函數(shù)體的return后不能接任何類(lèi)型,或者不出現(xiàn)return語(yǔ)句。
function test(array $arr) : void { if (!count($arr) { return; } array_walk($arr, function ($elem) {xxxx}); }
回顧完以上歷史,可以看出到PHP 7.1,函數(shù)類(lèi)型聲明已經(jīng)十分完善(雖然實(shí)踐中用的不多)。
再說(shuō)說(shuō)實(shí)踐中踩到的坑。參數(shù)和返回值類(lèi)型聲明可用的類(lèi)型有:
- 類(lèi)/接口
- self,只能用在自身的方法上
- array
- bool
- callable
- int
- float
- string
- iterable
注意列表中并沒(méi)有boolean和double類(lèi)型!除非你定義了這兩個(gè)類(lèi)型,否則用在參數(shù)和返回值中就是錯(cuò)誤的!
這也是PHP有點(diǎn)蛋疼的地方。平常使用時(shí)的double和float兩個(gè)關(guān)鍵字幾乎等同,例如doubleval是floatval的別名,is_double是is_float的別名,轉(zhuǎn)換時(shí)用(double)和(float)效果相同。但是到了類(lèi)型聲明這里就不行,同樣的情況出現(xiàn)在bool和boolean身上。
總結(jié)
目前PHP 7.2穩(wěn)定版已經(jīng)發(fā)布,建議在新項(xiàng)目中盡量使用PHP 7.1及后續(xù)版本。為了寫(xiě)出清晰和可維護(hù)的代碼,推薦聲明類(lèi)型。建議引用類(lèi)型或者string才使用null值,int/float等標(biāo)量類(lèi)型的參數(shù)盡量不要用null。func_get_argc等函數(shù),如非必要,盡量不使用。
相關(guān)文章
php利用ffmpeg提取視頻中音頻與視頻畫(huà)面的方法詳解
想要提取視頻中的音頻信息,首選的技術(shù)是ffmpeg,ffmpeg是一個(gè)非常有用的命令行程序,它可以用來(lái)轉(zhuǎn)碼媒體文件。這篇文章主要給大家介紹了PHP利用ffmpeg提取視頻中音頻與視頻畫(huà)面的相關(guān)資料,需要的朋友可以參考下。2017-06-06php實(shí)現(xiàn)驗(yàn)證郵箱格式的代碼實(shí)例
在本篇文章里小編給大家整理的是關(guān)于php實(shí)現(xiàn)驗(yàn)證郵箱格式的代碼實(shí)例以及相關(guān)知識(shí)點(diǎn),需要的朋友們參考下。2020-01-01PHP函數(shù)strip_tags的一個(gè)bug淺析
PHP 函數(shù) strip_tags 提供了從字符串中去除 HTML 和 PHP 標(biāo)記的功能,該函數(shù)嘗試返回給定的字符串 str 去除空字符、HTML 和 PHP 標(biāo)記后的結(jié)果。2014-05-05php及codeigniter使用session-cookie的方法(詳解)
下面小編就為大家?guī)?lái)一篇php及codeigniter使用session-cookie的方法(詳解)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04PHP session會(huì)話(huà)操作技巧小結(jié)
這篇文章主要介紹了PHP session會(huì)話(huà)操作技巧,結(jié)合實(shí)例形式詳細(xì)總結(jié)分析了php中session會(huì)話(huà)操作的原理、配置方法、使用技巧與相關(guān)注意事項(xiàng),需要的朋友可以參考下2016-09-09php中字符集轉(zhuǎn)換iconv函數(shù)使用總結(jié)
這篇文章主要介紹了php中字符集轉(zhuǎn)換iconv函數(shù)使用總結(jié),本文同時(shí)介紹了mb_convert_encoding函數(shù),需要的朋友可以參考下2014-10-10