關於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!

推薦閱讀: