前言
PHP手冊系列文章,會挑選一些手冊中有意思的評論進(jìn)行翻譯
手冊目錄: 語言參考---類型---類型轉(zhuǎn)換的判別
參考詳情
評論
- 兩個整數(shù)想除得到的可能是通過自動轉(zhuǎn)換后的float類型結(jié)果,所以你不需要通過整數(shù)加浮點(diǎn)數(shù)來避免結(jié)果截斷取整
$dividend = 2;
$divisor = 3;
$quotient = $dividend/$divisor;
print $quotient; // 0.66666666666667
- 將對象類型轉(zhuǎn)換為數(shù)組類型永遠(yuǎn)是心里的痛
class MyClass {
private $priv = 'priv_value';
protected $prot = 'prot_value';
public $pub = 'pub_value';
public $MyClasspriv ='second_pub_value';
}
$test = new MyClass();
echo '<pre>';
print_r((array) $test);
/* 輸出
Array
(
[MyClasspriv] => priv_value
[*prot] => prot_value
[pub] => pub_value
[MyClasspriv] => second_pub_value
)
*/
結(jié)果看上去很正常,對象屬性轉(zhuǎn)換為關(guān)聯(lián)數(shù)組,key是屬性,value是值,并且通過在key前面添加*來標(biāo)示其在對象中是protected的訪問等級,但是事實遠(yuǎn)非如此
foreach ((array) $test as $key =>$value) {
$len = strlen($key);
echo "{$key} ({$len}) => {$value}<br />";
for ($i = 0;$i< $len; ++$i) {
echo ord($key[$i]) .' ';
}
echo '<hr />';
}
/*輸出
MyClasspriv (13) => priv_value
0 77 121 67 108 97 115 115 0 112 114 105 118
*prot (7) => prot_value
0 42 0 112 114 111 116
pub (3) => pub_value
112 117 98
MyClasspriv (11) => second_pub_value
77 121 67 108 97 115 115 112 114 105 118
*/
輸出的字符碼顯示 protected屬性轉(zhuǎn)變成key的時候會添加'\0*\0',同時private屬性轉(zhuǎn)變成key的時候會添加'\0'._CLASS.'\0'
- ++操作符不會將boolean類型轉(zhuǎn)換程int類型,并且如果一個變量是boolean,值為true,那么對其++之后它還是true
$a = true;
var_dump(++$a); // 輸出 bool(true)
4.object 轉(zhuǎn)換到 object是可以的,通過serialize/unserialize操作,詳情見序列化
class my{
public $m = 1;
public function test(){
echo 'he';
}
}
class yo{
public $mi = 2;
public function testi(){
echo 'hehe';
}
}
$b = new my();
$a = serialize($b);
echo $a; //輸出O:2:"my":1:{s:1:"m";i:1;},其中O表示object,O后面的2表示類名的長度,"my"表示類名,1表示變量個數(shù)(包括public,protected,private),s:1:"m"依次表示變量,變量名長度,變量名.所以我們要轉(zhuǎn)換成別的類,不僅要替換掉類名,同時也需要替換掉類名長度
$c = unserialize($a);
if($c == $b) echo 'succ'; //輸出succ,說明$c和$b是同一個類的實例,并且屬性值都相等,注意$b!==$c
$a = str_replace('my','yo',$a); //新類名長度與舊類名長度一致,所以只需要替換類名部分
echo $a; //輸出O:2:"yo":1:{s:1:"m";i:1;}
$d = unserialize($a); //新類產(chǎn)生
if($d instanceof yo)echo 'succ' //輸出succ,是yo類的實例
echo $d->m; //1
echo $d->mi; //2
echo $d->test(); //error:undefined method
echo $d->testi(); //hehe //如此可以知道轉(zhuǎn)換后的類居然保存有舊類的變量屬性,但是不具備舊類的方法屬性,很神奇有木有,其實可以手動修改串O:2:"yo":1:{s:1:"m";i:1;},想添加什么變量都可以,但是注意protected的變量轉(zhuǎn)變成字符串之后會添加'\0*\0',具體見上面的class轉(zhuǎn)換成array
思考:那么舊類中的private和protected變量會不會也傳遞給新類呢,訪問限制等級是否會發(fā)生變化呢,如果舊類中的變量的初始值是另外一個類的實例又會怎樣,如果舊類extends其他類或者implements其他接口又會是怎樣,這個可以自己去實驗,不再一一贅述
- 一些更短更快的方法進(jìn)行類型轉(zhuǎn)換
$string='12345.678';
$float=+$string;
$integer=0|$string;
$boolean=!!$string;
- 你應(yīng)該意識到不要在對某個目標(biāo)連續(xù)進(jìn)行太多次類型轉(zhuǎn)換,因為這樣很可能意外的將false變成true
if(TRUE === (boolean) (array) (int) FALSE) {
echo 'ca';
} // (boolean)array(0) 變成 true了
- 從string轉(zhuǎn)換到int或者從int轉(zhuǎn)換到string是最常見的,php可以通過+.和.=來轉(zhuǎn)換
$x = 1;
var_dump($x);// int(1)
$x .= 1;
var_dump($x);// string(2) "11"; also an empty string ("") would cast to string without changing $x
$x = "1";
var_dump($x); // string(1) "1"
$x += 1;
var_dump($x);// int(2); also a zero value (0) would cast to int without changing $x
- 提出一段很有意思的代碼
$obj = new stdClass();
$obj->{'4'} = 'id';
$arr = (array) $obj;
var_dump($arr);
foreach ($arr as $key => $value) {
var_dump($key);
var_dump($arr[$key]);
}
?>
/*
輸出:
array(1) {
["4"]=>
string(2) "id"
}
string(1) "4"
string(2) "id"
NULL
*/
發(fā)現(xiàn)怎么都讀取不到$arr[$key],應(yīng)該輸出$value的值,并且通過array()構(gòu)造$arr怎么都構(gòu)造不出下面這個數(shù)組
array(1) {
["4"]=>
string(2) "id"
}
希望高手能解釋下這個問題
- unset看起來好像用處比較單一,但是如果你想把代碼變得緊湊,你可以在同一行使用變量并且unset它
$hello = 'Hello world';
$hello = (unset) print $hello;
// 等同于
$hello = 'Hello world';
print $hello;
unset($hello);
- 僅僅是一些關(guān)于unset的經(jīng)驗之談
$var = 1;
$var_unset = (unset) $var;
$var_ref_unset &= (unset)$var; //把與&換成或|結(jié)果也一樣
var_dump($var);
var_dump($var_unset);
var_dump($var_ref_unset);
output:
int(1)
NULL
int(0)