關於PHP方法參數的那一些事
前言
在所有的編程語言中,方法或者函數,都可以傳遞一些參數進來進行業務邏輯的處理或者計算。這沒什麼可說的,但是在PHP中,方法的參數還有許多非常有意思的能力,下面我們就來說說這方面的內容。
引用參數
涉及到值傳遞和引用傳遞的問題。在正常情況下,我們使用值傳遞的時候,變量是進行瞭拷貝,方法內外的變量不會共享內存。也就是說,在方法體中修改瞭變量的值,方法外部的變量不會產生變化。而引用傳遞則是傳遞的變量的內存地值。方法內外的變量可以看做是同一個變量,比如:
$a = 1; function test(&$arg){ $arg++; } test($a); echo $a; // 2
為參數加上&標識,就表明這個參數是引用傳遞的參數。如果沒有加這個標識,則所有的基本類型參數都會以值的方式進行傳遞。為什麼要強調基本類型呢?下面我們用類當參數來測試一下:
class A { public $a = 1; } function testA($obj) { $obj->a++; } $o = new A(); testA($o); echo $o->a; // 2
在這個例子中,我們並沒有使用&標識來表明參數$obj是引用類型的,但如果傳遞的參數是對象的話,那麼它默認就是進行的引用傳遞。如果想讓對象也是值傳遞呢?抱歉,在方法參數中是沒辦法實現的,隻能在方法體中使用clone方式對對象參數進行克隆。
class A { public $a = 1; } function testA($obj) { $o = clone $obj; $o->a++; } $o = new A(); testA($o); echo $o->a; // 1
關於值和引用的問題,可以參考設計模式中原型模式的講解:PHP設計模式之原型模式
默認參數
參數是可以有默認值的,這個我想大傢都應該很清楚瞭。但是在使用的時候也需要註意,那就是默認參數不要放在前面,否則很容易出錯,比如:
function testArgsA($a = 1, $b){ echo $a+$b; } testArgs(); // error function testArgsB($a = 1, $b = 2){ echo $a+$b; } testArgsB(); // 3 function testArgsC($a, $b = 2){ echo $a+$b; } testArgsC(1); // 3
在復雜的函數或者緊急的業務開發中,很有可能一個不小心就會漏寫參數,這時候testArgsA就會返回錯誤瞭。當然,這種粗心類的錯誤是我們應該盡量避免的。
當指定默認值的時候,我們應該根據參數的類型進行指定,比如字符串就指定為”,數字就指定為數字類型。當不確定參數是什麼類型時,建議使用NULL做為默認參數。
function testArgsD($a = NULL) { if ($a) { echo $a; } } testArgsD(1); testArgsD('a');
類型聲明
類型聲明是在PHP5之後添加的功能,就像java一樣,參數前面加上參數的類型,比如:
function testAssignA(int $a = 0) { echo $a; } testAssignA(1); testAssignA("a"); // error
如果參數的類型不對,直接就會報錯。在PHP7以前,隻支持類、數組和匿名方法的類型聲明。在PHP7之後,支持所有的普通類型,但是這裡要註意的是,隻支持普通類型的固定寫法。
- Class/interface name
- self
- array
- callable
- bool
- float
- int
- string
固定寫法是什麼意思呢?
function testAssignB(integer $a = 0) // error { echo $a; }
也就是說,int隻能寫int,不能使用integer,bool也不能使用boolean。隻能是上面列出的類型關鍵字。
類型聲明的好處是什麼呢?其實就是Java這種靜態語言和PHP這種動態語言之間的差別。動態類型語言的好處就是變量靈活,不用指定類型,方便快速開發迭代。但問題也在於靈活,為瞭靈活,動態語言往往會在比較或者計算時對變量進行自動類型轉換。如果你對變量類型轉換的理解不清晰的話,很容易就會出現各種類型的BUG。同時,靜態類型的語言一般都會有編譯打包,而動態類型則是在執行時確定變量類型,所以很少會進行編譯打包,相對來說運行效率也就不如Java之類的編譯後語言瞭。
關於PHP的類型轉換問題,可以參考此前的文章:PHP中的強制類型轉換
Tips一個小技巧,如果聲明瞭參數類型,是不能傳遞NULL值的,比如:
function testAssignC(string $a = '') { if ($a) { echo __FUNCTION__ . ':' . $a; } } testAssignC(NULL); // TypeError
這時有兩種方式可以解決,一是指定默認值=NULL,二是使用?操作符:
function testAssignD(string $a = NULL) { if ($a == NULL) { echo 'null'; } } testAssignD(NULL); // null function testAssignE(?string $a) { if ($a == NULL) { echo 'null'; } } testAssignE(NULL); // null
可變數量參數
php中的方法可以接收可變數量的參數,比如:
function testMultiArgsA($a) { var_dump(func_get_arg(2)); var_dump(func_get_args()); var_dump(func_num_args()); echo $a; } testMultiArgsA(1, 2, 3, 4);
我們隻定義瞭一個參數$a,但是傳進去瞭四個參數,這時我們可以使用三個方法來獲取所有的參數:
- func_get_arg(int $arg_num),獲取參數列表中的某個指定位置的參數
- func_get_args(),獲取參數列表
- func_num_args(),獲取參數數量
此外,php還提供瞭…操作符,用於將可變長度的參數定義到一個參數變量中,如:
function testMultiArgsB($a, ...$b) { var_dump(func_get_arg(2)); var_dump(func_get_args()); var_dump(func_num_args()); echo $a; var_dump($b); // 除$a以外的 } testMultiArgsB(1, 2, 3, 4);
和參數默認值一樣,有多個參數的情況下,…b也不要放在前面,這樣後面的參數並不會有值,所有的參數都會在b也不要放在前面,這樣後面的參數並不會有值,所有的參數都會在b中。不過PHP默認已經幫我們解決瞭這個問題,如果…參數後面還有參數的話,會直接報錯。
利用這個操作符,我們還可以很方便的解包一些數組或可迭代的對象給方法參數,例如:
function testMultiArgsC($a, $b){ echo $a, $b; } testMultiArgsC(...[1, 2]);
是不是很有意思,那麼我們利用這個特性來合並一個數組會是什麼效果呢?
$array1 = [[1],[2],[3]]; $array2 = [4]; $array3 = [[5],[6],[7]]; $result = array_merge(...$array1); // Legal, of course: $result == [1,2,3]; print_r($result); $result = array_merge($array2, ...$array1); // $result == [4,1,2,3] print_r($result); $result = array_merge(...$array1, $array2); // Fatal error: Cannot use positional argument after argument unpacking. $result = array_merge(...$array1, ...$array3); // Legal! $result == [1,2,3,5,6,7] print_r($result);
和方法聲明參數時一樣,在外部使用…操作符給方法傳遞參數時,也不能在…後面再有其他參數,所以array_merge(…array1, array1,array2)的操作會報錯。
測試代碼:
https://github.com/zhangyue0503/dev-blog/blob/master/php/201911/source/PHP%E6%96%B9%E6%B3%95%E5%8F%82%E6%95%B0%E7%9A%84%E9%82%A3%E7%82%B9%E4%BA%8B%E5%84%BF.php
參考文檔:
- https://www.php.net/manual/zh/functions.arguments.php
- https://www.php.net/manual/zh/functions.arguments.php#121579
總結
到此這篇關於關於PHP方法參數的文章就介紹到這瞭,更多相關PHP方法參數內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- PHP方法的返回值示例詳解
- PHP的runkit擴展如何使用
- php中echo、print和print_r的區別點及用法總結
- PHP如何初始化PDO及原始SQL語句操作
- PHP中強制類型轉換的示例詳解