Q1
第一個問題關於弱型別
1 2 3 4 5 6 7 |
$str1 = 'yabadabadoo'; $str2 = 'yaba'; if (strpos($str1,$str2)) { echo "\"" . $str1 . "\" contains \"" . $str2 . "\""; } else { echo "\"" . $str1 . "\" does not contain \"" . $str2 . "\""; } |
正確執行的輸出結果:
1 |
"yabadabadoo" does not contain "yaba" |
strpos
是返回字串str2
在str1
的位置,沒有找到則返回false
然而實際上這次返回了0
而在if語句中0也被當作false,所以我們需要對false做型別判斷,正確的程式碼如下:
1 2 3 4 5 6 7 |
$str1 = 'yabadabadoo'; $str2 = 'yaba'; if (strpos($str1,$str2) !== false) { echo "\"" . $str1 . "\" contains \"" . $str2 . "\""; } else { echo "\"" . $str1 . "\" does not contain \"" . $str2 . "\""; } |
需要注意的是我們使用了!==
,在php 和 JS中= !相對== 更為嚴格需要要求資料型別一致。
Q2
下面的輸出結果會是怎樣?
1 2 3 4 5 6 7 8 9 10 |
$x = 5; echo $x; echo "<br />"; echo $x+++$x++; echo "<br />"; echo $x; echo "<br />"; echo $x---$x--; echo "<br />"; echo $x; |
實際執行結果是
1 2 3 4 5 |
5 11 7 1 5 |
關於 $x++
和 $x--
這個問題其實非常容易遇見,我們只需記住$x++
使用最近的值,然後才自增。
運算子的優先順序,++
是明顯高於 +
,因此先執行++ 再執行 + 。關於運算子的優先順序,有的時候我們真的可以通過括號來讓我們的程式更讓人直觀的瞭解,畢竟程式碼不光是用於執行的,有的時候或許團隊的可讀性也是提高效率的一種。
Q3
關於變數的引用;
1 2 3 |
$a = '1'; $b = &$a; $b = "2$b"; |
請問 $a
和 $b
的值各位多少
部分第一時間會想到 $a='1' $b='21'
,仔細一看 $b=&$a
,這裡$b
是變數$a
的引用而不是直接 賦值。
Q4
下面是true還是false
1 2 3 |
var_dump(0123 == 123); var_dump('0123' == 123); var_dump('0123' === 123); |
var_dump(0123 == 123);// false
,PHP會預設把0123當作8進位制來處理,實際轉化為10進位制就是83,顯然這不是相等的。
var_dump('0123' == 123);// true
這裡php會非常有趣的將’0123’轉換成一個數字而且預設去掉了前面的0也就是123==123
var_dump('0123' === 123);// false
很顯然上面的問題已經說過了數字和字串型別不一致。
Q5
下面的程式碼有什麼問題嗎?輸出會是什麼,怎樣修復它
1 2 3 4 5 6 7 8 9 10 11 12 13 |
$referenceTable = array(); $referenceTable['val1'] = array(1, 2); $referenceTable['val2'] = 3; $referenceTable['val3'] = array(4, 5); $testArray = array(); $testArray = array_merge($testArray, $referenceTable['val1']); var_dump($testArray); $testArray = array_merge($testArray, $referenceTable['val2']); var_dump($testArray); $testArray = array_merge($testArray, $referenceTable['val3']); var_dump($testArray); |
實際輸出如下:
1 2 3 |
array(2) { [0]=> int(1) [1]=> int(2) } NULL NULL |
執行的時候你或許還能看到下面的警告
1 2 |
Warning: array_merge(): Argument #2 is not an array Warning: array_merge(): Argument #1 is not an array |
array_merge
需要傳入的引數都是陣列,如果不是,則會返回null。 你可以這樣修改
1 2 3 4 5 6 |
$testArray = array_merge($testArray, (array)$referenceTable['val1']); var_dump($testArray); $testArray = array_merge($testArray, (array)$referenceTable['val2']); var_dump($testArray); $testArray = array_merge($testArray, (array)$referenceTable['val3']); var_dump($testArray); |
Q6
$x應該是輸出什麼?
1 2 |
$x = true and false; var_dump($x); |
部分同學或許會第一時間想到false,實際上這裡依舊是強調運算子的優先順序,= 會比 and級別高點,因此等同下面的程式碼
1 2 |
$x = true; true and false |
答案顯而易見。
Q7
經過下面的運算 $x的值應該是多少?
1 |
$x = 3 + "15%" + "$25" |
答案是18
,PHP是會根據上下文實現型別的自動轉換
上面的程式碼我們可以這樣理解,如果我們在與字串進行數學運算,實際php會盡可能將字串中的陣列進行轉換,如果是數字開頭的話則轉換成改數字比如”15%”會變成15,如果不是數字開頭則會變成0; 上面的運算類似下面 :
1 |
$x = 3 + 15 + 0 |
Q8
執行下面的程式碼,$text
的值是多少?strlen($text)
又會返回什麼結果?
1 2 |
$text = 'John '; $text[10] = 'Doe'; |
上面程式碼執行完畢後 $text = "John D"
(John後面會有連續的5個空格) strlen($text)
會返回11
$text[10] = "Doe"
給某個字串具體的某個位置具體字元時候,實際只會把D賦給$text
. 雖然$text
才開始只有5個自負長度,但是php會預設填充空格。這和別的語言有些差別。
Q9
下面的輸出結果會是什麼
1 2 3 4 5 6 7 8 9 |
$v = 1; $m = 2; $l = 3; if( $l > $m > $v){ echo "yes"; }else{ echo "no"; } |
實際的輸出是”no”,只要仔細分析就不難得出
$l>$m
會轉換成1 ,則這個時候再和$m比較。
Q10
執行下面程式碼$x
會變成什麼值呢?
1 2 3 4 5 |
$x = NULL; if ('0xFF' == 255) { $x = (int)'0xFF'; } |
實際的執行結果是$x=0
而不是255.
首先'oxFF' == 255
我們好判斷,會進行轉換將16進位制數字轉換成10進位制數字,0xff -> 255.
PHP使用is_numeric_string
判斷字串是否包含十六進位制數字然後進行轉換。
但是$x = (int)'0xFF';
是否也會變成255呢?顯然不是,將一個字串進行強制型別轉換實際上用的是convert_to_long
,它實際上是將字串從左向右進行轉換,遇到非數字字元則停止。因此0xFF
到x就停止了。所以$x=0