2021年最新PHP 面试、笔试题汇总(一)
读在前面:
面向对象编程和面向对象设计的五个基本原则「SOLID」
单一职责原则(Single Responsibility Principle)
开放封闭原则(Open Closed Principle)
里氏替换原则(Liskov Substitution Principle)
接口隔离原则(Interface Segregation Principle)
依赖反转原则(Dependency Inversion Principle)
1.单一职责原则规定一个类有且仅有一个理由使其改变。换句话说,一个类的边界和职责应当是十分狭窄且集中的。我很喜欢的一句话"在类的职责问题上,无知是福"。一个类应当做它该做的事,并且不应当被它的任何依赖的变化所影响。
2.开放封闭原则,又称开闭原则,规定代码对扩展是开放的,对修改是封闭的。
3.里氏替换原则规定对象可以被其子类的实例所替换,并且不会影响到程序的正确性。
4.接口隔离原则规定,不应该强制接口的实现依赖于它不使用的方法。
5.依赖反转原则,它规定高层次的代码不应该依赖低层级的代码。换句话说,高层次的代码应该依赖抽象接口,抽象接口就像是「中间人」一样,负责连接着高层次和低层次代码。这个原则的另一层意思是,抽象接口不应该依赖具体实现,但具体实现应该依赖抽象接口
一、秒杀(商品超卖,高并发,同一用户多次抢购)
后端:redis+队列
redis队列实现,三个队列(库存队列,排队队列,抢购结果队列)
用户先进入排队队列,先进先出,判断是否已经在抢购结果队列,如果在,则直接下一个,如果不在,将用户信息加入抢购结果队列,库存-1,等待数据库空闲时,将抢购结果写入数据库
前端:
面对高并发的抢购活动,前端常用的三板斧是【扩容】【静态化】【限流】
扩容:加机器,这是最简单的方法,通过增加前端池的整体承载量来抗峰值。
静态化:将活动页面上的所有可以静态的元素全部静态化,并尽量减少动态元素。通过CDN来抗峰值。
限流:一般都会采用IP级别的限流,即针对某一个IP,限制单位时间内发起请求数量。或者活动入口的时候增加游戏或者问题环节进行消峰操作。
有损服务:在接近前端池承载能力的水位上限的时候,随机拒绝部分请求来保护活动整体的可用性。
二、订单模块(同一订单,多家商户结算问题)
订单拆分:用户支付后,将订单拆分,生成子订单
三、用户下单
先判断有没有登录
点击下单,生成唯一订单号,状态为未支付
四、接口安全
使用HTTP的POST方式,对固定参数+附加参数进行数字签名,使用的是md5加密,比如:我想通过标题获取一个信息,在客户端使用 信息标题+日期+双方约定好的一个key通过md5加密生成一个签名(sign),然后作为参数传递到服务器端,服务器端使用同样的方法进行校验,如何接受过来的sign和我们通过算法算的值相同,证明是一个正常的接口请求,我们才会返回相应的接口数据。
五、如何处理负载、高并发
1、HTML静态化
其实大家都知道,效率最高、消耗最小的就是纯静态化的html页面,所以我们尽可能使我们的 网站上的页面采用静态页面来实现,这个最简单的方法其实也是最有效的方法。
2、图片服务器分离
把图片单独存储,尽量减少图片等大流量的开销,可以放在一些相关的平台上,如七牛云
3、数据库集群和库表散列及缓存
数据库的并发连接为100,一台数据库远远不够,可以从读写分离、主从复制,数据库集群方面来着手。另外尽量减少数据库的访问,可以使用缓存数据库如memcache、redis。
4、镜像:
尽量减少下载,可以把不同的请求分发到多个镜像端。
5、负载均衡:
Apache的最大并发连接为1500,只能增加服务器,可以从硬件上着手,如F5服务器。当然硬件的成本比较高,我们往往从软件方面着手。
负载均衡 (Load Balancing) 建立在现有网络结构之上,它提供了一种廉价有效透明的方法扩展网络设备和服务器的带宽、增加吞吐量、加强网络数据处理能力,同时能够提高网络的灵活性和可用性。目前使用最为广泛的负载均衡软件是Nginx、LVS、HAProxy。
六、修改会话的生存时间
1.在php.ini中 设置 session.gc_maxlifetime = 1440 //默认时间
2.代码实现;
$lifeTime = 24 * 3600; //保存一天
session_set_cookie_params($lifeTime);
session_start();
七、PHP的垃圾收集机制
PHP可以自动进行内存管理,清除不再需要的对象。PHP使用了引用计数(referencecounting)这种单纯的垃圾回收(garbagecollection)机制。每个对象都内含一个引用计数器,每个reference连接到对象,计数器加1。当reference离开生存空间或被设为NULL,计数器减1。当某个对象的引用计数器为零时,PHP知道你将不再需要使用这个对象,释放其所占的内存空间
八、正则的引擎
正则引擎主要可以分为两大类:一种是DFA,一种是NFA。
一般而论,DFA引擎则搜索更快一些。但是NFA以表达式为主导,更容易操纵,因此一般程序员更偏爱NFA引擎!
可以使用是否支持忽略优先量词和分组捕获来判断引擎类型:支持 NFA,不支持 DFA
九、对一个大文件进行逐行遍历,如下方法性能较高的是?
写一个实现了IteratorAggregate 接口的类,通过该类使用foreach遍历。
(使用 IteratorAggregate 可将文件打开后通过移动指针的方式逐行遍历,不受文件大小影响。使用 file_get_contents 处理大文件很容易导致PHP内存溢出;调用exec 会产生额外的进程,影响性能;其他人写的类库质量不一定高。)
十、读取文件加锁和解锁
$fp = fopen("lock.txt","w+");
if (flock($fp,LOCK_EX)) {
//获得写锁,写数据
fwrite($fp, "write something");
// 解除锁定
flock($fp, LOCK_UN);
} else {
echo "file is locking...";
}
fclose($fp);
十一、array_merge() 数组合并函数
定义:array_merge() 函数把一个或多个数组合并为一个数组。(您可以向函数输入一个或者多个数组。)
注释:如果两个或更多个数组元素有相同的键名,则最后的元素会覆盖其他元素。
如果两个数组都是索引数组,则不会覆盖
如果您仅向 array_merge() 函数输入一个数组,且键名是整数,则该函数将返回带有整数键名的新数组,其键名以 0 开始进行重新索引。
该函数与 array_merge_recursive() 函数之间的不同是在处理两个或更多个数组元素有相同的键名的情况。array_merge_recursive() 不会进行键名覆盖,而是将多个相同键名的值递归组成一个数组。
示例;
$a1=array("red","green");
$a2=array("blue","yellow");
$a3=array("CC","DD");
$b1=array("a"=>"sa","b"=>"sb");
$b2=array("a"=>"qa","b"=>"qb");
$b3=array("a"=>"wa","c"=>"ww");
print_r(array_merge($a1,$a2)); //Array ( [0] => red [1] => green [2] => blue [3] => yellow )
print_r(array_merge($a1,$a2,$a3)); //Array ( [0] => red [1] => green [2] => blue [3] => yellow [4] => CC [5] => DD )
print_r(array_merge($b1,$b2)); //Array ( [a] => qa [b] => qb )
print_r(array_merge($b1,$b2,$a3)); // Array ( [a] => qa [b] => qb [0] => CC [1] => DD )
print_r(array_merge($b1,$b2,$b3)); // Array ( [a] => wa [b] => qb [c] => ww )
十二、获取文件扩展名
//plan A
function get_ext(string $url){
//$url = 'http://www.sina.com.cn/abc/de/fg.html?id=1&ajksfg&aakzsdfj';
$a = parse_url($url); //Array ( [scheme] => http [host] => www.sina.com.cn [path] => /abc/de/fg.html [query] => id=1&ajksfg&aakzsdfj )
$file = basename($a['path']); //fg.html
$b = explode('.',$file);
return array_pop($b);
}
//plan B
function get_ext(string $url){
//$url = 'http://www.sina.com.cn/abc/de/fg.html?id=1&ajksfg&aakzsdfj';
$a = basename($url); //fg.html?id=1&ajksfg&aakzsdfj
$b = explode('?',$a);; //Array ( [0] => fg.html [1] => id=1&ajksfg&aakzsdfj )
$ext = explode('.',$b[0]);
return array_pop($ext);
}
十三、遍历一个文件夹下的所有文件和子文件夹
function my_scandir($dir){
$files = array();
if(is_dir($dir)){
if ($handle = opendir($dir)){
while (($file = readdir($handle))!= false){
if ($file != "." && $file != "..") {
if (is_dir($dir."/".$file)){
$files[$file] = my_scandir($dir."/".$file);
} else{
$files[] = $dir."/".$file;
}
}
}
closedir($handle);
return $files;
}
}
}
十四、编写一个函数,递归遍历,实现无限分类
function tree($arr,$pid=0,$level=0){
static $list = array();
foreach ($arr as $v) {
//如果是顶级分类,则将其存到$list中,并以此节点为根节点,遍历其子节点
if ($v['parent_id'] == $pid) {
$v['level'] = $level;
$list[] = $v;
tree($arr,$v['cat_id'],$level+1);
}
}
return $list;
}
十五、获取上月的最后一天
function get_last_month_last_day($date = ''){
if ($date != '') {
$time = strtotime($date);
} else {
$time = time();
}
$day = date('j',$time);//获取该日期是当前月的第几天
return date('Y-m-d',strtotime("-{$day} days",$time));
}
十六、php中WEB上传文件的原理是什么,如何限制上传文件的大小?
上传文件的表单使用post方式,并且要在form中添加enctype=‘multipart/form-data’。
一般可以加上隐藏域:,位置在file域前面。
value的值是上传文件的客户端字节限制。可以避免用户在花时间等待上传大文件之后才发现文件过大上传失败的麻烦。
使用file文件域来选择要上传的文件,当点击提交按钮之后,文件会被上传到服务器中的临时目录,在脚本运行结束时会被销毁,所以应该在脚本结束之前,将其移动到服务器上的某个目录下,可以通过函数move_uploaded_file()来移动临时文件,要获取临时文件的信息,使用$_FILES。
限制上传文件大小的因素有:
客户端的隐藏域MAX_FILE_SIZE的数值(可以被绕开)。
服务器端的upload_max_filesize,post_max_size和memory_limit。这几项不能够用脚本来设置。
自定义文件大小限制逻辑。即使服务器的限制是能自己决定,也会有需要个别考虑的情况。所以这个限制方式经常是必要的。
十七、双引号和单引号的区别
双引号解释变量,单引号不解释变量
双引号里插入单引号,其中单引号里如果有变量的话,变量解释
双引号的变量名后面必须要有一个非数字、字母、下划线的特殊字符,或者用{}讲变量括起来,否则会将变量名后面的部分当做一个整体,引起语法错误
双引号解释转义字符,单引号不解释转义字符,但是解释’\和\
能使单引号字符尽量使用单引号,单引号的效率比双引号要高(因为双引号要先遍历一遍,判断里面有没有变量,然后再进行操作,而单引号则不需要判断)
十八、常用的超全局变量
$_GET ----->get传送方式
$_POST ----->post传送方式
$_REQUEST ----->可以接收到get和post两种方式的值
$GLOBALS ----->所有的变量都放在里面
$_FILES ----->上传文件使用
$_SERVER ----->系统环境变量
$_SESSION ----->会话控制的时候会用到
$_COOKIE ----->会话控制的时候会用到
十九、echo、print_r、print、var_dump之间的区别
* echo、print是php语句,var_dump和print_r是函数
* echo 输出一个或多个字符串,中间以逗号隔开,没有返回值是语言结构而不是真正的函数,因此不能作为表达式的一部分使用
* print也是php的一个关键字,有返回值 只能打印出简单类型变量的值(如int,string),如果字符串显示成功则返回true,否则返回false
* print_r 可以打印出复杂类型变量的值(如数组、对象)以列表的形式显示,并以array、object开头,但print_r输出布尔值和NULL的结果没有意义,因为都是打印"\n",因此var_dump()函数更适合调试
* var_dump() 判断一个变量的类型和长度,并输出变量的数值
二十、对于大流量网站,采用什么方法来解决访问量的问题
确认服务器硬件是否能够支持当前的流量
数据库读写分离,优化数据表
优化SQL语句
禁止外部盗链
控制大文件的下载
使用不同主机分流主要流量
使用流量分析统计
扫描关注 进入”PHP资料“
免费获取进阶
面试、文档、视频资源