PHP 第25章 面试题 PHP 第25章 面试题

2022-07-14

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中= !相对== 更为严格需要要求数据类型一致。

2、下面的输出结果会是怎样

$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++使用最近的值,然后才自增。

运算符的优先级,++ 是明显高于 +,因此先执行++ 再执行 + 。

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

3、关于变量的引用

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

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

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

4、下面是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,很显然上面的问题已经说过了数字和字符串类型不一致。

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

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

6、$x应该是输出什么

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

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

$x = true;
true and false

答案显而易见

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

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

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

上面的代码我们可以这样理解,如果我们在与字符串进行数学运算,实际php会尽可能将字符串中的数组进行转换。

如果是数字开头的话则转换成改数字比如"15%"会变成 15,如果不是数字开头则会变成 0。上面的运算类似下面:

$x = 3 + 15 + 0

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

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

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

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

9、下面的输出结果会是什么

$v = 1;
$m = 2;
$l = 3;

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

实际的输出是"no",只要仔细分析就不难得出:$l>$m 会转换成1 ,则这个时候再和$m比较。

10、执行下面代码$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。

11、写一个函数,尽可能高效的,从一个标准 url 里取出文件的扩展名

function getExt($url){
    $arr = parse_url($url);
    $file = basename($arr['path']);
    $ext = explode(".",$file);
    return $ext[1];
}

12、通过哪一个函数,可以把错误转换为异常处理?

  • A:set_error_handler

  • B:error_reporting

  • C:error2exception

  • D:catch

正确答案:A 答案分析:set_error_handler() 可指定一个回调函数,错误发生时,会自动通过指定的回调函数处理。在回调函数中抛出新的异常即可。

13、下列哪个shell函数的描述是正确的?

  • A:shell函数可以先调用后定义

  • B:shell函数需使用关键字function定义

  • C:shell函数内的变量可以声明为局部变量

  • D:shell函数只能通过return返回值,1是成功,0是失败

正确答案:C 答案分析:shell函数必须先定义在调用;声明时,无需使用关键字;通过local可以定义函数内的局部变量;shell函数返回值,0是成功,非0是错误,其他选项正确

14、下列关于全文检索技术的说法,不对的是:

  • A: Solr是新一代的全文检索组件,它比Lucene的搜索效率高很多,还能支持HTTP的访问方式,PHP调用Solr也很方便。

  • B: MySQL中把一个字段建立FULLTEXT索引,就可以实现全文检索,目前MyISAM和InnoDB的table都支持FULLTEXT索引。

  • C: Sphinx是一个基于SQL的全文检索引擎,可以结合MySQL做全文搜索,它可以提供比数据库本身更专业的搜索功能。

  • D: Lucene附带的二元分词分析器CJKAnalyzer切词速度很快,能满足一般的全文检索需要。

正确答案:A 答案分析: Solr是新一代的全文检索组件,它基于Lucene,所以说它比Lucene快就是胡扯 :)

15、关于单例模式的说法,错误的是?

  • A:单例模式的目的是确保在全局环境中,一个类只能有一个实例存在

  • B:单利模式一般要讲构造函数设置为 private

  • C:只需要将构造函数设置为private 即可确保全局中只有一个实例

  • D:连接数据库的功能通常用单例模式实现

正确答案:C 答案分析:构造函数设置为private,仅能确保无法通过 new 创建新实例,但仍可以通过 clone、反序列化等方式创建多个实例。

16、正则的引擎表述错误的是?

  • A 正则引擎主要可以分为两大类:一种是DFA,一种是NFA。

  • B 一般而论,NFA引擎则搜索更快一些。但是DFA以表达式为主导,更容易操纵,因此一般程序员更偏爱DFA引擎!

  • C NFA表达式主导,DFA文本主导.

  • D 可以使用是否支持忽略优先量词和分组捕获来判断引擎类型:支持 NFA,不支持 DFA

正确答案:B

答案分析:正确的说法应该是:一般而论,DFA引擎则搜索更快一些。但是NFA以表达式为主导,更容易操纵,因此一般程序员更偏爱NFA引擎!

17、方框中的正则表达式能与以下哪些选项匹配?

/.*\*123\d/

  • A.**123

  • B.****1234

  • C.1234

  • D._*123

正确答案:B

答案分析:本题的要点是理解这个正则表达式的含义——从左往右,首先是零个或多个任意字符(.*),跟着是一个星号(*),然后是 123,最后是一个数字。因此答案是B。

18、如下关于数据库的说法,哪个是错误的?

  • A:为了效率数据库可以有多个读库

  • B:数据库可以用主从做热备

  • C:数据库不能提供多主多从架构

  • D: 数据库主从是通过日志同步的

正确答案:C

答案分析: 数据库可以提供多主多从架构。

19、下面哪个不是XSS漏洞的修复方式?

  • A:对参数进行htmlspecialchas过滤

  • B:对参数使用白名单过滤

  • C:不允许输入的内容显示到浏览器

  • D:禁止在js标签内输出用户输入的内容

正确答案:A 这类过滤可以解决尖括号类型的xss,无法解决js标签内的xss

20、下列哪一项不是PHP SAPI模式?

  • A.ISAPI

  • B.CGI

  • C.FastCGI

  • D.RESTFUL APi

正确答案:D

答案分析:A~C是最常用的模式,D是一种接口的组织方式。

21、对一个大文件进行逐行遍历,如下方法性能较高的是?

  • A:写一个实现了IteratorAggregate 接口的类,通过该类使用foreach遍历。

  • B:使用file_get_contents 将文件内容一次性载入内存,然后逐行遍历。

  • C:通过exec函数,调用shell 工具遍历

  • D:使用别人写的类库

正确答案:A

答案分析:使用 IteratorAggregate 可将文件打开后通过移动指针的方式逐行遍历,不受文件大小影响。使用 file_get_contents 处理大文件很容易导致PHP内存溢出;调用exec 会产生额外的进程,影响性能;其他人写的类库质量不一定高。

22、如下选项,哪个不是设计模式应该遵循的原则?

  •  A:组合优于继承

  • B:针对接口编程

  • C:尽可能降低耦合

  • D:尽量使用高性能的语法

正确答案:D

答案分析:设计模式的关注点在于代码的可维护性和可复用性,D选项不是设计模式关注的要点。

23、下列关于回溯的表达式错误的是?

  • A ab.lmn 匹配 abcdeflmnghijklmn 中的 abcdeflmnghijklmn

  • B ab.?lmn 匹配 abcdeflmnghijklmn 中的 abcdeflmn

  • C ab??c 匹配 abcdeflmnghijklmn 中的 abc

  • D .*lmn 匹配 abcdeflmnghijklmn 中的 abcdeflmn

正确答案:D

答案分析:D是贪婪匹配,所以应该匹配到的结果是abcdeflmnghijklmn

24、函数中如果使用了try catch finally 语法结构,return 应该写在哪儿?

  • A:finally 中

  • B:try 中

  • C:catch 中

  • D:任意位置

正确答案:A

答案分析:try 中 return 后 finally 会继续执行,如果 finally 中也有return,则最终返回值为 finally 中 return 的值。

25、以下关于NOSQL的说法,不对的是:

  • A: Redis支持字符串、哈希、列表、集合、有序集合等数据结构,目前Redis不支持事务。

  • B: MongoDB支持CAP定理中的AP,MySQL支持CAP中的CA,全部都支持不可能存在。

  • C: MongoDB不用先创建Collection的结构就可以直接插入数据,目前MongoDB不支持事务。

  • D: Memcache既支持TCP协议,也支持UDP协议,我们可以把PHP的Session存放到Memcache中。

正确答案:A

答案分析:Redis支持事务。

26、Innodb 锁机制说法错误的是?

  • A:Innodb提供了表锁与行锁两种锁机制

  • B:Innodb的表锁所会在表变更的时候触发

  • C:Innodb下update时会自动给涉及到的行加上排他锁,并创建出一个镜像副本, 此时进行select 时查询的是镜像副本的数据

  • D:Innodb行锁状态下读不受影响,写会受影响(涉及到的数据)

正确答案:A

27、下列哪个是创建一个每周三01:00~04:00每3分钟执行执行一次的crontab指令?

  • A: 1,4 3 /bin/bash /home/sijiaomao/ok.sh

  • B:/3 1,4 * 3 /bin/bash /home/sijiaomao/ok.sh

  • C:/3 1-4 * 3 /bin/bash /home/sijiaomao/ok.sh

  • D:/3 1-4 * * * /bin/bash /home/sijiaomao/ok.sh

正确答案:C

答案分析:A:每周三的1时4时每分钟执行一次 B:每周三的1时4时每3分钟执行一次 C:满足要求 D:每天的1时4时每3分钟执行一次

28、在拆分之前,系统中很多列表和详情页所需的数据是可以通过sql join来完成的。而拆分后,数据库可能是分布式在不同实例和不同的主机上,join将变得非常麻烦。下面哪种方法不能有效解决这个问题?

  • A 全局表,系统中所有模块都可能会依赖到的一些表在各个库中都保存。

  • B 字段冗余,“订单表”中保存“卖家Id”的同时,将卖家的“Name”字段也冗余,这样查询订单详情的时候就不需要再去查询“卖家用户表”。

  • C 主从复制,将数据库的读写分离。

  • D 数据同步,定时A库中的tbl_a表和B库中tbl_b关联,可以定时将指定的表做主从同步。

正确答案:C

答案分析:主从复制,将数据库的读写分离。只能扩容读并发,并不能缓解跨库join的问题。

29、关于网络IO模型,下列哪一项是正确的?

  • A.Select比Epoll更快

  • B.nginx使用的是select模型

  • C.apache支持select和epoll两种方式的切换

  • D.epoll能支持更大的并发

正确答案:D

答案分析:A epoll更快一些。B nginx使用epoll模型。C apache只支持select

30、PHP执行的时候有如下执行过程:Scanning(Lexing) - Compilation - Execution - Parsing,其含义分别为:

  • A:将PHP代码转换为语言片段(Tokens)、将Tokens转换成简单而有意义的表达式、将表达式编译成Opocdes、顺次执行Opcodes

  • B:将PHP代码转换为语言片段(Tokens)、将Tokens转换成简单而有意义的表达式、顺次执行Opcodes、将表达式编译成Opocdes

  • C:将PHP代码转换为语言片段(Tokens)、将表达式编译成Opocdes、顺次执行Opcodes、将Tokens转换成简单而有意义的表达式

  • D:将PHP代码转换为语言片段(Tokens)、将表达式编译成Opocdes、将Tokens转换成简单而有意义的表达式、顺次执行Opcodes

正确答案:C

答案分析:正确答案为C,正确的顺序为:Scanning(Lexing)、Parsing、Compilation、Execution

31、引用

首先这是一道上机题,数据部分,点击下载。你可以直接将其导入到mysql内。

题目要求

  • 操作category表(上万条数据),以层级(多级分类,3级及以上)的树状结构一次性显示全部商品分类

  • 程序运行时间尽量短。

第1点中,需要的数据结构大致如下

女装
--T恤
----长袖T恤
----七分袖T恤
----中袖T恤
------第四级别
----短袖T恤
----无袖T恤
--小衫
----长袖小衫
----中袖小衫
----短袖小衫

代码

值得体会的地方,在于generateTree函数的实现,看不懂的可以输出一下 $items 和 $tree的结构,以便理解。

32、请说明php中传值与传引用的区别?什么时候传值什么时候传引用?

  • 按值传递:函数范围内对值的任何改变在函数外部都会被忽略

  • 按引用传递:函数范围内对值的任何改变在函数外部也能反映出这些修改

  • 优缺点:按值传递时,php 必须复制值。特别是对于大型的字符串和对象来说,这将会是一个代价很大的操作。

  • 按引用传递则不需要复制值,对于性能提高很有好处。

  • 对于较大是的数据,传引用比较好,这样可以节省内存的开销。

33、交换$a、$b的值

用PHP写一段代码,实现不使用第3个变量,交换$a、$b的值,$a、$b的初始值自己定。(五种方法实现)

$a = 11;
$b = 33;
// 方法一:
list($a,$b) = array($b,$a);
echo $a.'-'.$b,'
'; // 方法二: $a = $a . '-' . $b; $a = explode('-',$a); $b = $a[0]; $a = $a[1]; echo $a.'-'.$b,'
'; // 方法三: $a = $a.$b; $b = strlen($b); $b = substr($a,0,strlen($a)-$b); $a = substr($a,strlen($b)); echo $a . '-' . $b,'
'; // 方法四: // 当两个变量都是数字的时候 $a = $a + $b; $b = $a-$b; $a = $a-$b; echo $a . '-' . $b,'
'; // 方法五 $a = 11; $b = 33; $a = $a^$b; $b = $b^$a; $a = $a^$b; echo $a . '-' . $b,'
';

注意:

  • 方法五中,两个变量的长度必须一样,

  • 方法四中,两个变量必须是数字

  • 方法三中,可能存在编码问题

  • 方法二中,注意分隔符

  • 方法一,没得说,顶上!!!

34、$a++ 和++$a

$a++ 是先返回值再加

++$a 是先加再返回

$a=1;
echo " \$a=1
"; $zong=$a++; echo " \$a++ 的结果:$zong
"; $a=1; $zong=++$a; echo " ++\$a 的结果:$zong"; 输出: $a=1 $a++ 的结果:1 ++$a 的结果:2

35、运行PHP脚本

请写出并说明如何在命令行下运行PHP脚本(写出两种方式)同时向PHP脚本传递参数

方法一:

if ($argc > 1){
    print_r($argv);
}
?>
php.exe test.php welcome Ricky Feng

方法二:

还有一种这样的,提示用户输入,然后获取输入的参数。有点像C语言

fwrite(STDOUT, "Enter your name: ");
$name = trim(fgets(STDIN));
fwrite(STDOUT, "Hello, $name!");

测试结果:

$>php io.php
Enter your name: test name
Hello, test name!

第三种:

$options = getopt("f:");
var_dump($options);

php aaa.php -f "123456"

array(1) {
  ["f"]=>
  string(5) "123456"
}

windows 和 linux

①、windows

  • window下,假设php安装目录为c:\program files\php5\,那么使用命令窗口进入到该路径下,敲入php hello.php回车,则会执行当前路径下的hello.php文件,

  • 如果要指向其他路径下php文件,可以在php 路径/hello.php ,这种形式称为CLI模式,我们平时通过浏览器看到的那种成为CGI模式,至于传递参数,php文件在cli模式下,直接通过在文件名称后面接参数,多个参数中间用空格隔开,在php文件里面是通过两个变量来获取参数的,一个是$argv,一个是$argc,前者是传递参数的数组,默认第一个为php文件的名称;后者为$argv的数组个数。

②、linux

  • linux下,一般程序安装都会安装在/usr/bin/php下面,可以通过man php查看一下,如果有信息说明可以使用,使用方法类似于window下。如果前面这步成立,那么你可以直接 php php文件 来运行php文件,如果man php没有信息,则说明当前php执行文件没有在环境路径里面,可以修改环境路径包含php路径,也可以类似于window进入php路径,在执行 php php文件。其他类似于window下。

阅读 968