PHP中強制類型轉換的示例詳解

前言

學過靜態語言開發的朋友對類型轉換不會陌生,比如Java、C#、C++等。靜態語言的好處就是變量強制必須指定類型,這也是編譯的要求,所以大部分編譯型的語言都會有強制變量類型的要求。而PHP據說也會在PHP8中加入JIT實現編譯功能,並且在7.4中就會引入變量聲明時的類型指定。下面我們先看看目前PHP中的參數類型及返回值類型的使用。

1function add(int $a, float $b) : int{
2 return (int) $a + $b;
3}

上述代碼中,方法參數中定義瞭參數的類型,包括一個int類型的a和一個float類型的b。然後在方法後面定義瞭方法的返回值必須是int類型。我們知道,如果計算表達式中出現瞭float類型,那麼計算結果會變成float類型。這個方法需要返回的是一個int類型。因此我們使用瞭一個強制類型轉換(int)。在定義瞭參數類型和返回值類型後,如果傳遞或者返回的類型不一致,就會報錯。

參數類型和返回值類型最好在7以上的版本使用。基本類型如int、float等的參數類型聲明都是7以後才支持的,詳情參見文檔:

https://www.php.net/manual/zh/functions.arguments.php

我們通過(int)、(float)、(bool)等就可以實現PHP的類型強制轉換,和C基本上一樣。文檔中關於可以強制轉換的包括如下類型:

  • (int), (integer) – 轉換為整形 integer
  • (bool), (boolean) – 轉換為佈爾類型 boolean
  • (float), (double), (real) – 轉換為浮點型 float
  • (string) – 轉換為字符串 string
  • (array) – 轉換為數組 array
  • (object) – 轉換為對象 object
  • (unset) – 轉換為 NULL (PHP 5)
  • (binary) 轉換和 b 前綴轉換支持為 PHP 5.2.1 新增

(int), (integer)

如果是佈爾值,轉換結果為false變成0,true變成1

如果是float,向下取整,如7.99會轉換為7

如果是字符串,字符串從頭開始查找,開頭第一個是數字會直接變成該轉換結果,如果開頭沒有數字返回0

其他類型轉換在文檔中並沒有定義,文檔提示為“沒有定義從其它類型轉換為整型的行為。不要依賴任何現有的行為,因為它會未加通知地改變。”,但我們通過測試,可以發現對於其他類型的轉換是通過多次的類型轉換達成的,比如數組類型轉換為int類型,是根據數組是否包含內容轉換為bool類型後再轉換為int類型

// (int)(integer)
 
var_dump((int) true); // 1
var_dump((int) false); // 0
 
var_dump((int) 7.99); // 7
 
var_dump((int) "35 ok"); // 35
var_dump((int) "ok 77"); // 0
var_dump((int) "ok yes"); // 0

var_dump((int) []); // 0
var_dump((int) [3,4,5]); // 1

(bool)(boolean)

當轉換為 boolean 時,以下值被認為是 FALSE:

  • 佈爾值 FALSE 本身
  • 整型值 0(零)
  • 浮點型值 0.0(零)
  • 空字符串,以及字符串 “0”
  • 不包括任何元素的數組
  • 特殊類型 NULL(包括尚未賦值的變量)
  • 從空標記生成的 SimpleXML 對象

所有其它值都被認為是 TRUE(包括任何資源 和 NAN)

這裡需要註意的是,負數也會是TRUE,隻有0是FASLE

// (bool)(boolean)

var_dump((bool) 0); // false
var_dump((bool) 1); // true
var_dump((bool) -1); // true

var_dump((bool) 0.0); // false
var_dump((bool) 1.1); // true
var_dump((bool) -1.1); // true

var_dump((bool) ""); // false
var_dump((bool) "0"); // false
var_dump((bool) "a"); // true

var_dump((bool) []); // false
var_dump((bool) ['a']); // true

$a;
var_dump((bool) $a); // false
var_dump((bool) NULL); // false

(string)

  • 佈爾值,false轉換為空字符串””,true轉換為”1″
  • int或float類型,轉換為字符串形式的字面量,如1轉換為”1″
  • 數組和對象分別轉換為”Array”和”Object”字面量
  • 資源類型會被轉換為”Resource id #1″形式的字面量
  • NULL轉換為空字符串””

直接把 array,object 或 resource 轉換成 string 不會得到除瞭其類型之外的任何有用信息。可以使用函數 print_r() 和 var_dump() 列出這些類型的內容

註:測試結果,對象類型需要實現__tostring()魔術函數,否則報錯無法轉換為string類型

// (string)

var_dump((string) true); // "1"
var_dump((string) false); // ""

var_dump((string) 55); // "55"
var_dump((string) 12.22); // "12.22"

var_dump((string) ['a']); // "Array"
class S{
 function __tostring(){
 return "S";
 }
}
var_dump((string) new S()); // "S"

var_dump((string) NULL); // ""

(array)

對於任意 integer,float,string,boolean 和 resource 類型,如果將一個值轉換為數組,將得到一個僅有一個元素的數組,其下標為 0,該元素即為此標量的值。換句話說(array)scalarValue 與 array(scalarValue) 是完全一樣的

如果一個 object 類型轉換為 array,則結果為一個數組,其單元為該對象的屬性。鍵名將為成員變量名,不過有幾點例外:整數屬性不可訪問;私有變量前會加上類名作前綴;保護變量前會加上一個 ‘*’ 做前綴。這些前綴的前後都各有一個 NULL 字符

將 NULL 轉換為 array 會得到一個空的數組

// (array)

var_dump((array) 1);
var_dump((array) 2.2);

var_dump((array) "a");

var_dump((array) true);

class Arr
{
 public $a = 1;
 private $b = 2.2;
 protected $c = "f";
}
class ChildArr extends Arr
{
 public $a = 2;
 private $d = "g";
 private $e = 1;
}
var_dump((array) new Arr());
var_dump((array) new ChildArr());

var_dump((array) null);

(object)

如果將一個對象轉換成對象,它將不會有任何變化。如果其它任何類型的值被轉換成對象,將會創建一個內置類 stdClass 的實例。如果該值為 NULL,則新的實例為空。array 轉換成 object 將使鍵名成為屬性名並具有相對應的值

註意:使用 PHP 7.2.0 之前的版本,數字鍵隻能通過迭代訪問

// (object)

var_dump((object) 1);
var_dump((object) 1.1);
var_dump((object) "string");
var_dump((object) true);
var_dump((object) NULL);

var_dump((object) [1, 2, 3]);
var_dump((object) ["a" => 1, "b" => 2, "c" => 3]);

(unset)

使用 (unset) $var 將一個變量轉換為 null 將不會刪除該變量或 unset 其值。僅是返回 NULL 值而已

// (unset)

var_dump((unset) 1);
var_dump((unset) 1.1);
var_dump((unset) "string");
var_dump((unset) true);
var_dump((unset) null);

var_dump((unset) [1, 2, 3]);
var_dump((unset) new \stdClass());

(binary)

將所有類型轉換為二進制字符串。二進制字符串是區別於傳統常用的普通php的Unicode字符串。二進制字符串是字節字符串,沒有字符集。具體的區別就類似於數據庫中的binary和char類型及blob和text類型

在日常的開發中基本用不到,瞭解即可

// (binary)

var_dump((binary) 1);
var_dump((binary) 1.1);
var_dump((binary) "string");
var_dump((binary) true);
var_dump((binary) null);

var_dump((binary) [1, 2, 3]);
var_dump((binary) new S());

以上就是我們的強制類型轉換的所有類型,其中有一些類型的轉換中提到瞭資源類型(Resource),但是並沒有資源類型的強制轉換。因為資源類型大多是一些句柄操作,如數據庫鏈接、文件讀寫等,將其它類型強制轉換為資源類型沒有意義。

本文內容會經常出現在面試題中,而且在實際開發中的很多邏輯判斷出現的BUG也常常是由於PHP的自動類型轉換所導致的,所以這篇文章好好收藏多拿出來看看絕對會讓你有意想不到的收獲哦!!

測試代碼:

https://github.com/zhangyue0503/dev-blog/blob/master/php/201910/source/php%E4%B8%AD%E7%9A%84%E5%BC%BA%E5%88%B6%E7%B1%BB%E5%9E%8B%E8%BD%AC%E6%8D%A2.php

參考文檔:

https://www.php.net/manual/zh/language.types.type-juggling.php#language.types.typecasting

總結

到此這篇關於PHP中強制類型轉換的文章就介紹到這瞭,更多相關PHP強制類型轉換內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: