从源码角度看 PHP 字符串类型转换

        PHP 的类型转换是比较方便的,但是越是容易使用的东西,底层的实现越是复杂,而且在使用中像我这样的新手也往往不清楚转换后的结果到底是什么。有时候,对于 Java 这种强类型的语言,使用的时候需要强制进行转换,这样多半转换后的结果是可以预料的,至少自己是想这么做的。

通过实例看转换

        写个关于 PHP 类型转换的代码测试一下吧,看看大家是否能知道输出结果。各种数据类型 都转换为 字符串类型试试。

 1 <?php
 2 
 3 $var = false;
 4 echo '$var:', $var, "\r\n";
 5 
 6 $var = null;
 7 echo '$var:', $var, "\r\n";
 8 
 9 $var = true;
10 echo '$var:', $var, "\r\n";
11 
12 $var = 1;
13 echo '$var:', $var, "\r\n";
14 
15 $var = 1.1;
16 echo '$var:', $var, "\r\n";
17 $var1 = &$var;
18 echo '$var:', $var1, "\r\n";
19 
20 $var1 = 'string111';
21 echo '$var:', $var1, "\r\n";
22 
23 $var = [1, 2, 3];
24 echo '$var:', $var, "\r\n";
25 
26 class Obj
27 {
28     private $field;
29 }
30 
31 $var = new Obj();
32 echo '$var:', $var, "\r\n";

        代码很简单,大家可以猜猜输出结果,然后再和运行结果比对一下,运行结果如下:

 1 $ php string.php
 2 $var:
 3 $var:
 4 $var:1
 5 $var:1
 6 $var:1.1
 7 $var:1.1
 8 $var:string111
 9 $var:
10 Notice: Array to string conversion in C:\Users\Administrator\Desktop\string.php
11 on line 24
12 
13 Call Stack:
14     0.0010     405568   1. {main}() C:\Users\Administrator\Desktop\string.php:0
15 
16 Array
17 $var:
18 Catchable fatal error: Object of class Obj could not be converted to string in C
19 :\Users\Administrator\Desktop\string.php on line 32
20 
21 Call Stack:
22     0.0010     405568   1. {main}() C:\Users\Administrator\Desktop\string.php:0

        不知道是否实际的运行结果和猜测的结果是否一致。

PHP 源码中各类型转换为字符串类型的处理

        对于各种类型之间的赋值,并转换为字符串,使用真的是非常的方便。但是,实际的功劳都是 PHP 底层的复杂处理。看一下关于各种类型转换为字符串类型的源代码的处理。

 1 ZEND_API zend_string* ZEND_FASTCALL _zval_get_string_func(zval *op) /* {{{ */
 2 {
 3 try_again:
 4   switch (Z_TYPE_P(op)) {
 5     case IS_UNDEF:
 6     case IS_NULL:
 7     case IS_FALSE:
 8       return ZSTR_EMPTY_ALLOC();
 9     case IS_TRUE:
10       return ZSTR_CHAR('1');
11     case IS_RESOURCE: {
12       char buf[sizeof("Resource id #") + MAX_LENGTH_OF_LONG];
13       int len;
14 
15       len = snprintf(buf, sizeof(buf), "Resource id #" ZEND_LONG_FMT, (zend_long)Z_RES_HANDLE_P(op));
16       return zend_string_init(buf, len, 0);
17     }
18     case IS_LONG: {
19       return zend_long_to_str(Z_LVAL_P(op));
20     }
21     case IS_DOUBLE: {
22       return zend_strpprintf(0, "%.*G", (int) EG(precision), Z_DVAL_P(op));
23     }
24     case IS_ARRAY:
25       zend_error(E_NOTICE, "Array to string conversion");
26       return zend_string_init("Array", sizeof("Array")-1, 0);
27     case IS_OBJECT: {
28       zval tmp;
29       if (Z_OBJ_HT_P(op)->cast_object) {
30         if (Z_OBJ_HT_P(op)->cast_object(op, &tmp, IS_STRING) == SUCCESS) {
31           return Z_STR(tmp);
32         }
33       } else if (Z_OBJ_HT_P(op)->get) {
34         zval *z = Z_OBJ_HT_P(op)->get(op, &tmp);
35         if (Z_TYPE_P(z) != IS_OBJECT) {
36           zend_string *str = zval_get_string(z);
37           zval_ptr_dtor(z);
38           return str;
39         }
40         zval_ptr_dtor(z);
41       }
42       zend_error(EG(exception) ? E_ERROR : E_RECOVERABLE_ERROR, "Object of class %s could not be converted to string", ZSTR_VAL(Z_OBJCE_P(op)->name));
43       return ZSTR_EMPTY_ALLOC();
44     }
45     case IS_REFERENCE:
46       op = Z_REFVAL_P(op);
47       goto try_again;
48     case IS_STRING:
49       return zend_string_copy(Z_STR_P(op));
50     EMPTY_SWITCH_DEFAULT_CASE()
51   }
52   return NULL;
53 }
54 /* }}} */

        从上面的源码中可以看出,对于类型的转换,PHP 底层的源码使用了 switch ... case 的结构,在进行转换的时候需要对逐个的类型进行匹配后,再进行转换。当然,从源码的角度去看待上面的 PHP 类型转换的代码就非常的清晰明了了。

 

最后

        读 PHP 的源码有一段时间了,进度比较慢,说实话,对于写 PHP 的代码没有什么提高,或者对我来说没有遇到什么让我提高的地方。单纯的出于对知识求甚解的原因吧。


我的微信公众号:“码农UP2U”

猜你喜欢

转载自www.cnblogs.com/tosser/p/11877828.html