
PHP 的類型轉換是比較方便的,但是越是容易使用的東西,底層的實現越是復雜,而且在使用中像我這樣的新手也往往不清楚轉換后的結果到底是什么。有時候,對于 Java 這種強類型的語言,使用的時候需要強制進行轉換,這樣多半轉換后的結果是可以預料的,至少自己是想這么做的。
通過實例看轉換
寫個關于 PHP 類型轉換的代碼測試一下吧,看看大家是否能知道輸出結果。把 各種數據類型 都轉換為 字符串類型試試。
<?php
$var = false;
echo '$var:', $var, "\r\n";
$var = null;
echo '$var:', $var, "\r\n";
$var = true;
echo '$var:', $var, "\r\n";
$var = 1;
echo '$var:', $var, "\r\n";
$var = 1.1;
echo '$var:', $var, "\r\n";
$var1 = &$var;
echo '$var:', $var1, "\r\n";
$var1 = 'string111';
echo '$var:', $var1, "\r\n";
$var = [1, 2, 3];
echo '$var:', $var, "\r\n";
class Obj
{
private $field;
}
$var = new Obj();
echo '$var:', $var, "\r\n";
代碼很簡單,大家可以猜猜輸出結果,然后再和運行結果比對一下,運行結果如下:
$ php string.php
$var:
$var:
$var:1
$var:1
$var:1.1
$var:1.1
$var:string111
$var:
Notice: Array to string conversion in C:\Users\Administrator\Desktop\string.php
on line 24
Call Stack:
0.0010 405568 1. {main}() C:\Users\Administrator\Desktop\string.php:0
Array
$var:
Catchable fatal error: Object of class Obj could not be converted to string in C
:\Users\Administrator\Desktop\string.php on line 32
Call Stack:
0.0010 405568 1. {main}() C:\Users\Administrator\Desktop\string.php:0
不知道是否實際的運行結果和猜測的結果是否一致。
PHP 源碼中各類型轉換為字符串類型的處理
對于各種類型之間的賦值,并轉換為字符串,使用真的是非常的方便。但是,實際的功勞都是 PHP 底層的復雜處理。看一下關于各種類型轉換為字符串類型的源代碼的處理。
ZEND_API zend_string* ZEND_FASTCALL _zval_get_string_func(zval *op) /* {{{ */
{
try_again:
switch (Z_TYPE_P(op)) {
case IS_UNDEF:
case IS_NULL:
case IS_FALSE:
return ZSTR_EMPTY_ALLOC();
case IS_TRUE:
return ZSTR_CHAR('1');
case IS_RESOURCE: {
char buf[sizeof("Resource id #") + MAX_LENGTH_OF_LONG];
int len;
len = snprintf(buf, sizeof(buf), "Resource id #" ZEND_LONG_FMT, (zend_long)Z_RES_HANDLE_P(op));
return zend_string_init(buf, len, 0);
}
case IS_LONG: {
return zend_long_to_str(Z_LVAL_P(op));
}
case IS_DOUBLE: {
return zend_strpprintf(0, "%.*G", (int) EG(precision), Z_DVAL_P(op));
}
case IS_ARRAY:
zend_error(E_NOTICE, "Array to string conversion");
return zend_string_init("Array", sizeof("Array")-1, 0);
case IS_OBJECT: {
zval tmp;
if (Z_OBJ_HT_P(op)->cast_object) {
if (Z_OBJ_HT_P(op)->cast_object(op, &tmp, IS_STRING) == SUCCESS) {
return Z_STR(tmp);
}
} else if (Z_OBJ_HT_P(op)->get) {
zval *z = Z_OBJ_HT_P(op)->get(op, &tmp);
if (Z_TYPE_P(z) != IS_OBJECT) {
zend_string *str = zval_get_string(z);
zval_ptr_dtor(z);
return str;
}
zval_ptr_dtor(z);
}
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));
return ZSTR_EMPTY_ALLOC();
}
case IS_REFERENCE:
op = Z_REFVAL_P(op);
goto try_again;
case IS_STRING:
return zend_string_copy(Z_STR_P(op));
EMPTY_SWITCH_DEFAULT_CASE()
}
return NULL;
}
/* }}} */
從上面的源碼中可以看出,對于類型的轉換,PHP 底層的源碼使用了 switch ... case 的結構,在進行轉換的時候需要對逐個的類型進行匹配后,再進行轉換。當然,從源碼的角度去看待上面的 PHP 類型轉換的代碼就非常的清晰明了了。
最后
讀 PHP 的源碼有一段時間了,進度比較慢,說實話,對于寫 PHP 的代碼沒有什么提高,或者對我來說沒有遇到什么讓我提高的地方。單純的出于對知識求甚解的原因吧。