乐趣区

关于php:PHP中的强制类型转换

学过动态语言开发的敌人对类型转换不会生疏,比方 Java、C#、C++ 等。动态语言的益处就是变量强制必须指定类型,这也是编译的要求,所以大部分编译型的语言都会有强制变量类型的要求。而 PHP 据说也会在 PHP8 中退出 JIT 实现编译性能,并且在 7.4 中就会引入变量申明时的类型指定。上面咱们先看看目前 PHP 中的参数类型及返回值类型的应用。


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

上述代码中,办法参数中定义了参数的类型,包含一个 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

===========

各自媒体平台均可搜寻【硬核项目经理】

退出移动版