几道坑人的PHP面试题 试试看看你会不会也中招

这篇文章主要介绍了几道坑人的PHP面试题,试试看看你会不会也中招,这些题目都用了一些障眼法,需要你有一双火眼金睛哦,需要的朋友可以参考下
这几道题是在德问上看到的,感觉挺有意思,拿来给大家分享其中的陷阱,看看你会不会掉入其中。


第一题
代码如下:
$arr = array(0=>1,"aa"=>2, 3, 4); 
foreach($arr as $key=>$val){
    print($key == "aa" ? 5 : $val);
}
输出结果是多少?如果的答案是1534就掉入陷阱了。
先看看这个数组最终形成的结构:
 代码如下:
Array
(
    [0] => 1
    [aa] => 2
    [1] => 3
    [2] => 4
)

然后遍历每一个元素的key看等不等于aa,等于就用5替代。当我告诉你答案是5534的时候,你会不会有点惊讶!难道0等于"aa"吗?是的,0就等于"aa",这道题重点就考你这个。在PHP中两个值进行逻辑判断时,如果两个值的类型不一致PHP会自动把右边的值转换到左边的类型,然后再进行判断。因此"aa"转换整形等于0,自然也就等于左边的0了。你可以使用全等于避免这种该情况,也就是如果你写成:
复制代码 代码如下:
print($key === "aa" ? 5 : $val);
那么答案就是1534了。


第二题
代码如下:
$i='11';
printf("%d\n",printf("%d",printf("%d",$i)));
输出结果是多少?如果你回答是11,或者111111就掉入陷阱了。
先了解printf这个函数,printf不仅是打印函数,它还有返回值。重点就在这
复制代码 代码如下:
var_dump(printf("%d",$i));
你猜猜上面的结果是啥?先是printf打印变量本身11,然后printf会返回一个变量字符串长度的值,11有两个字符,于是返回2,于是上面语句的执行结果等于:
11int(2) 
清楚了这一点以后,再回过来看上面的试题,按照优先级,限制性深度printf函数,打印11,返回2。接着到第二级printf函数,打印2,返回1。最后到第三层,直接打印1,所以执行结果是 1121。

 


第三题
代码如下:
$a = 3;
$b = 5;
if($a = 5 || $b = 7) {
    $a++;
    $b++;
}
echo $a . " " . $b;
执行结果是多少?如果你回答 6 8 or 4 6 or  6 6,那你就掉入陷阱了。
第一个陷阱,认为答案等于 4 6 。估计你粗心把  $a = 5 ||  $b = 7  看成 $a == 5 ||  $b == 7 ,这是新手常犯的错误。
第二个陷阱,认为答案等于 6 8。 你识破了  $a = 5 ||  $b = 7 这个骗局,但你没有注意到,逻辑或里只要依次执行直到某个表达式结果为true,表达式后边的就不再执行,$a = 5 返回true,后边的$b=7就不执行了。
第三个陷阱,认为答案等于 6 6。 OK,你识破了 逻辑或的规则,于是$a=5执行,$b=7不执行,但是你没有考虑到这里是逻辑表达式,返回给$a的值是要转换为布尔值的。这样看。
所以经过以上三个陷阱,你应该知道答案是多少了,其实 $a等于true以后,echo $a 输出就是1 ,$b值不变,结果就是 1 6 。

 

第四题
代码如下:
$count = 5;
function get_count() {
    static $count = 0;
    return $count++;
}
++$count;
get_count();
echo get_count();
执行结果是多少?如果你回答 2 ,恭喜,你掉入陷阱了。
其实这道题主要考两点,第一点是static静态类型。这种的值永远都是静态的,第一次调用声明等于0,并且自增等于1。第二次调用,1再自增就等于2。但其实这里还有一道陷阱,那就是++a与a++的区别,前++是先自增,后++是先返回值再自增,所以结果等于 1。


第五题
代码如下:
$a = count ("567")  + count(null) + count(false);
echo $a;
如果你回答 3 or 1,恭喜,掉入陷阱了。
因为count(null)等于0,false也算一个值。所以count(false)等于1。

 

 

第一个问题关于弱类型

$str1 = 'yabadabadoo';
$str2 = 'yaba';
if (strpos($str1,$str2)) { 
  echo "/"" . $str1 . "/" contains /"" . $str2 . "/"";
} else {
  echo "/"" . $str1 . "/" does not contain /"" . $str2 . "/"";
}

正确运行的输出结果:

"yabadabadoo" does not contain "yaba"

strpos是返回字符串str2在str1的位置,没有找到则返回false然而实际上这次返回了0而在if语句中0也被当作false,所以我们需要对false做类型判断,正确的代码如下:

$str1 = 'yabadabadoo';
$str2 = 'yaba';
if (strpos($str1,$str2) !== false) { 
  echo "/"" . $str1 . "/" contains /"" . $str2 . "/"";
} else {
  echo "/"" . $str1 . "/" does not contain /"" . $str2 . "/"";
}

需要注意的是我们使用了!==,在php 和 JS中= !相对== 更为严格需要要求数据类型一致。

Q2

下面的输出结果会是怎样?

$x = 5;
echo $x; 
echo "
"; echo $x+++$x++; echo "
"; echo $x; echo "
"; echo $x---$x--; echo "
"; echo $x;

实际运行结果是

5
11
7
1
5
关于 $x++ 和 $x–这个问题其实非常容易遇见,我们只需记住$x++使用最近的值,然后才自增。

运算符的优先级,++ 是明显高于 +,因此先执行++ 再执行 + 。关于运算符的优先级,有的时候我们真的可以通过括号来让我们的程序更让人直观的了解,毕竟代码不光是用于执行的,有的时候或许团队的可读性也是提高效率的一种。

Q3

关于变量的引用;

$a = '1';
$b = &$a;
$b = "2$b";

请问 $a 和 $b的值各位多少

部分第一时间会想到 $a='1′ $b='21′,仔细一看 $b=&$a,这里$b是变量$a的引用而不是直接 赋值。

Q4

下面是true还是false

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

下面的代码有什么问题吗?输出会是什么,怎样修复它

$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);

实际输出如下:

array(2) { [0]=> int(1) [1]=> int(2) }
NULL
NULL

运行的时候你或许还能看到下面的警告

Warning: array_merge(): Argument #2 is not an array
Warning: array_merge(): Argument #1 is not an array

array_merge需要传入的参数都是数组,如果不是,则会返回null。 你可以这样修改

$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应该是输出什么?

$x = true and false;
var_dump($x);

部分同学或许会第一时间想到false,实际上这里依旧是强调运算符的优先级,= 会比 and级别高点,因此等同下面的代码

$x = true;
true and false

答案显而易见。

 

Q7

经过下面的运算 $x的值应该是多少?

$x = 3 + "15%" + "$25"

答案是18,PHP是会根据上下文实现类型的自动转换

上面的代码我们可以这样理解,如果我们在与字符串进行数学运算,实际php会尽可能将字符串中的数组进行转换,如果是数字开头的话则转换成改数字比如”15%”会变成15,如果不是数字开头则会变成0; 上面的运算类似下面 :

$x = 3 + 15 + 0

 

Q8

运行下面的代码,$text 的值是多少?strlen($text)又会返回什么结果?

$text = 'John ';
$text[10] = 'Doe';

上面代码执行完毕后 $text = “John D”(John后面会有连续的5个空格) strlen($text)会返回11

$text[10] = “Doe”给某个字符串具体的某个位置具体字符时候,实际只会把D赋给$text. 虽然$text才开始只有5个自负长度,但是php会默认填充空格。这和别的语言有些差别。

 

Q9

下面的输出结果会是什么

$v = 1;
$m = 2;
$l = 3;
 
if( $l > $m > $v){ 
  echo "yes";
}else{
  echo "no";
}

实际的输出是”no”,只要仔细分析就不难得出

$l>$m 会转换成1 ,则这个时候再和$m比较。

 

Q10

执行下面代码$x会变成什么值呢?

$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

 

原文:

https://m.jb51.net/article/53969.htm

https://m.jb51.net/article/97256.htm

你可能感兴趣的:(php)