PHP基础知识巩固--面试题

1. static 关键字

   class A{
        public static $num=0;
        public function __construct(){
            self::$num++; }
    }
    new A();
    new A();
    new A();
    echo A::$num;

结果为 3,因为static属性常驻内存,不会立即回收。且静态属性不能通过一个类已实例化的对象来访问。在类的内部使用 self:: n u m , 类 的 外 部 使 用 A : : num,类的外部使用A:: num使A::num 来访问。而不是A->$num

2. 继承类

<?php 
class A{ 
    public function __construct(){ 
        echo "Class A...<br/>"; 
    }
}
class B extends A{
    public function __construct(){
        echo "Class B...<br/>"; 
    }
}
    new B();
?>

输出结果为:Class B…

A 子类继承父类,子类的构造函数会覆盖父类的构造函数,子类 没有定义 构造 函数时,默认继承父类的构造方法

  1. 对象变量赋值
<?php 
class A{
    public $num=100; 
}
$a = new A();
$b = $a;
$a->num=200;
echo $b->num;
?>

输出结果为:200
在php5,一个对象变量已经不再保存整个对象的值。只是保存一个标识符来访问真正的对象内容。 当对象作为参数传递,作为结果返回,或者赋值给另外一个变量,另外一个变量跟原来的不是引用的关系,只是他们都保存着同一个标识符的拷贝,这个标识符指向同一个对象的真正内容。
对象的复制是通过引用来实现的,

$a=new A();$b=$a;
//相当于
$a=new A();$b=&$a;
  1. date()函数支持
    1.Y 4位数字完整表示年份,y两位数字表示年份
    2.a/A:表示上午或者下午,以am或者AM表示

  2. 以读写模式打开文件 "FILE.txt"的正确方法是
    fopen(“FILE.txt”,“r+”); ,该函数没有rw参数,用r+表示读写

'r'    只读方式打开,将文件指针指向文件头。
'r+'   读写方式打开,将文件指针指向文件头。
'w'    写入方式打开,将文件指针指向文件头并将文件大小截为零。如果文件不存在则尝试创建之。
'w+'   读写方式打开,将文件指针指向文件头并将文件大小截为零。如果文件不存在则尝试创建之。
'a'    写入方式打开,将文件指针指向文件末尾。如果文件不存在则尝试创建之。
'a+'   读写方式打开,将文件指针指向文件末尾。如果文件不存在则尝试创建之。
'x'    创建并以写入方式打开,将文件指针指向文件头。如果文件已存在,则 fopen() 调用失败并返回 FALSE,并生成一条 E_WARNING 级别的错误信息。如果文件不存在则尝试创建之。这和给 底层的 open(2) 系统调用指定 O_EXCL|O_CREAT 标记是等价的。
'x+'   创建并以读写方式打开,其他的行为和 'x' 一样。
  1. 以下代码执行结果
$a = 12;
$b = 012
$c = 0x12;
echo $a,"\n",$b,"\n",$c;
结果:12 10 18
0 开头代表8进制
0x 开头代表16进制
  1. 字符串转整型
echo 1+2+"3+4+5";
结果:6;
解析:一旦遇到 + 相当于将后面的字符串强转整型。
  1. unset 删除概念
$a =6;
$b = &$a;
unset($a);
echo $b 
结果:6
解析:unset 只是删除值的引用,不删除内存空间本身的值。
  1. PHP字符串定义方式及各自区别
单引号:
1.单引号不能解析变量
2.单引号不能解析转义字符,只能解析单引号和反斜线本身
3.变量和变量、变量和字符串、字符串和字符串之间可以用 . 连接
双引号:
双引号可以解析变量,变量可以使用特殊字符和{}包含
双引号可以解析所有转义字符 
变量和变量、变量和字符串、字符串和字符串之间可以用 . 连接
区别:
单引号效率高于双引号
heredoc的功能类似于双引号      nowdoc  功能类似与单引号

10.浮点类型不能用于精确的计算判断
在这里插入图片描述

浮点类型,在计算机进行计算交给cup进行计算,转成2进制,会有一定损耗。
  1. $_SERVER 详解
$_SERVER['SERVE_ADDR'] //服务端 IP 。
$_SERVER['REMOTE_ADDR'] //客户端 IP 。 
$_SERVER['REMOTE_HOST'] //当前用户主机名 
$_SERVER['REQUEST_URI'] //URL
$_SERVER['HTTP_REFERER'] //链接到当前页面的前一页面的 URL 地址
$_SERVER['HTTP_USER_AGENT'] //当前请求的 User_Agent: 头部的内容。
  1. foo()和@foo()函数的区别
结果:@foo()控制错误输出,foo()正常表达
解析:@ 符号在 PHP 中可以忽略错误报告,对于表达式有提示类错误的,又不影响语句的执行的,可以在表达式前面加 @
  1. 运算符优先级,逻辑短路,
    在这里插入图片描述
解析:先执行3>0 为true 执行 || 为true,后面不执行。
注意:
$a = true,$a++; 此时var_dump($a),$a为bool类型,echo $a ,$a = 1;
$a = true,$a=$a+1; 此时var_dump($a),$a为int类型,echo $a ,$a = 2;
  1. 变量作用域
$outer ='outer'; //全局变量
function fun(){
	//  global $outer;
	//  $outer = $GLOBALS['outer'];
    return $outer; //局部变量
}
echo fun();
解析:外部定义的全局变量要在内部使用,需要使用global关键字,或者$GLOBALS数组使用。否则取不到值。
  1. static 静态关键字
$count =5;
function get_count(){
    static $count;  //null
    return $count++; //1 - 2
}
echo $count; //5
++$count;  //6
echo get_count(); //null
echo get_count(); //1
结果:51
解析:
static关键字
1.仅初始化一次,初始化需赋值(不赋值则为null)。
2.每次执行函数后,该值都会保留。
3.static 修饰的变量是局部的,只在函数内部能够使用。
4.可以记录函数调用次数,从而可以在某些条件下终止递归。
  1. include 和 require
相同:
1.include/require 语句包含并运行指定文件。
2.如果给出路径名按照路径来找,否则从include_path中查找
3.如果include_path中也没有,则从调用脚本文件所在目录和当前工作目录下查找。
不同:
1.require一个文件存在错误的话,那么程序就会中断执行了,并显示致命错误。
include一个文件存在错误的话,那么程序不会中端,而是继续执行,并显示一个警告错误。 
include_once(require_once)和 include/require差别
include_once(require_once)需要查询一遍已加载的文件列表, 确认是否存在, 然后再加载。
  1. 抽象类,接口
接口:
interface 接口名称{ 
		//使用interface关键字声明接口
		常量成员 //接口中的成员属性只能是常量,不能是变量
		抽象方法 //接口中的所有方法必须是抽象方法,不能有非抽象的方法存在
}
		//eg:
		interface IAction {
		  public function eat($food);
		  public function swim();
		}
		//子类实现一个接口,用关键字 implements 
		//子类可以实现多个接口,用逗号来分隔多个接口的名称
		//子类中必须实现接口中定义的所有方法,否则会报一个致命错误
		class Whale implements IAction{
			public function eat($food){
				return $food;
			}
			public function swim(){
				return 1;
			}
		}
注意!!
抽象方法没有方法体,而且方法结束使用 ; 号,非抽象方法必须有方法体即{},可以不写具体内容)
函数体内不能写任何东西,连两个大括号都不能写!!!

抽象类:
abstract class AbstractClass
{
  // 强制要求子类定义这些方法
    abstract protected function getValue();
    abstract protected function prefixValue($prefix);

    // 普通方法(非抽象方法)
    public function printOut() {
        print $this->getValue() . "\n";
    }
}
//子类用 extends 继承父类
class ConcreteClass1 extends AbstractClass
{
    protected function getValue() {
        return "ConcreteClass1";
    }

    public function prefixValue($prefix) {
        return "{$prefix}ConcreteClass1";
    }
}

继承一个抽象类的时候,子类必须定义父类中的所有抽象方法;
另外,这些方法的访问控制必须和父类中一样(或者更为宽松)

抽象类和接口的区别:
1.接口没有数据成员,但是抽象类有数据成员,抽象类可以实现数据的封装。
2.接口没有构造函数,抽象类可以有构造函数。
3.接口中的方法都是public类型,而抽象类中的方法可以使用private、protected或public来修饰。
4.一个类可以同时实现多个接口,但是只能实现一个抽象类。

18.http协议和https区别

https 是基于SSL/TLS的HTTP协议,所有的http数据都是在SSL/TLS协议封装之上传输的。
https协议在http协议基础上,添加了SSL/TLS握手,以及数据加密传输,也属于应用层协议。
  1. MYSQL 数据类型表达的意义
整型:
TINYINT  带符号最大值127  最小值 -128
INT
BIGINT
整型格式:
unsigned:无符号的,表示只允许正数,所以取值范围取绝对值且最大值*2
zerofill: 填充0(如果声明了zerofill,该列会自动设为unsigned)
每个整型类型可以指定一个最小显示宽度
栗子:
int(3) 此时存入12333,是可以存入,但是显示宽度只会为123,这个宽度并不表示存储的值有多大
设置了 zerofill int(3) 此时存入12 会在前面自动补0,存储为012

定点小数 decimal:
DECIMAL数据类型用于在数据库中存储精确的数值。我们经常将DECIMAL数据类型用于保留准确精确度的列,例如会计系统中的货币数据

字符串:
char(n):固定长度,最多255个字符
varchar(n):可变长度,最多65535个字符

char是固定长度字符串,其长度范围为0~255且与编码方式无关,无论字符实际长度是多少,都会按照指定长度存储,不够的用空格补足
char 一般用于存储密码之类长度固定的字符串,char的字符串检索速度要比varchar类型的快

  1. 魔术方法,魔术常量
1。__construct()
实例化对象时被调用,当__construct和以类名为函数名的函数同时存在时,__construct将被调用,另一个不被调用。

2。__destruct()
当删除一个对象或对象操作终止时被调用。

3。__call()
对象调用某个方法,若方法存在,则直接调用;若不存在,则会去调用__call函数。

4。__get()
读取一个对象的属性时,若属性存在,则直接返回属性值;若不存在,则会调用__get函数。

5。__set()
设置一个对象的属性时,若属性存在,则直接赋值;若不存在,则会调用__set函数。

6。__toString()
打印一个对象的时被调用。如echo $obj;或print $obj;

7。__clone()
克隆对象时被调用。如:$t=new Test();$t1=clone $t;

8。__sleep()
serialize之前被调用。若对象比较大,想删减一点东东再序列化,可考虑一下此函数。

9。__wakeup()
unserialize时被调用,做些对象的初始化工作。

10。__isset()
检测一个对象的属性是否存在时被调用。如:isset($c->name)。

11。__unset()
unset一个对象的属性时被调用。如:unset($c->name)。

12。__set_state()
调用var_export时,被调用。用__set_state的返回值做为var_export的返回值。

13。__autoload()
实例化一个对象时,如果对应的类不存在,则该方法被调用。
  • SQL注入的原理是什么?如何防止SQL注入
原理:第一SQL本身有问题(这个不是主要问题)。第二你写的SQL很有问题(这是最主要的)
防范:第一,绝对不要相信用户输入的任何东西。第二,实用sql 的prepare预编译。现在的框架一般都会有SQL过滤的。
  • 数组函数
 1. array_chunk ( array $array , int $size [, bool $preserve_keys = false ] )
//将一个数组分割成多个数组,其中每个数组的单元数目由 size 决定。最后一个数组的单元数目可能会少于 size 个。 
栗子:
$input_array = array('a', 'b', 'c', 'd', 'e');
print_r(array_chunk($input_array, 2));
以上例程会输出:
array(array('a','b'),array('c','d'),array('e'))

 2.array_values($arr);  获得数组的值,并给其建立数字索引!
 栗子:
 $array = array("size" => "XL", "color" => "gold");
 print_r(array_values($array));
 以上例程会输出:
array("XL", "gold")

3.array_keys( $array , $search_value );  获得数组的键名,第二个参数如果传,则搜索数组中value等于第二个参数值的键名
栗子:
$array = array(0 => 100, "color" => "red");
print_r(array_keys($array));
输出:
array(0,"color")

$array = array("blue", "red", "green", "blue", "blue");
print_r(array_keys($array, "blue"));
输出:
array(0,3,4)

4.array_flip($arr);  数组中的值与键名互换(如果有重复前面的会被后面的覆盖)
$input = array("a" => 1, "b" => 1, "c" => 2);
$flipped = array_flip($input);
array([1] => b,[2] => c)

5.in_array("apple",$arr);  在数组中检索apple  检查数组中是否存在某个值,第三个参数默认false,当为true时严格比较
$a = array('1.10', 12.4, 1.13);
in_array('12.4', $a)  输出:true
in_array('12.4', $a, true)  输出:false

5.array_search("apple",$arr);  在数组中检索apple ,如果成功则返回首个相应的键名
$array = array(0 => 'blue', 1 => 'red', 2 => 'green', 3 => 'red');
$key = array_search('green', $array);
输出:
$key = 2;

6.array_key_exists("apple",$arr);  检索给定的键名是否存在数组中
$search_array = array('first' => 1, 'second' => 4);
array_key_exists('first', $search_array)  //输出:true

7.array_merge();合并一个或多个数组,返回作为结果的数组
//如果输入的数组中有相同的字符串键名,则该键名后面的值将覆盖前一个值
//然而,如果数组包含数字键名,后面的值将不会覆盖原来的值,而是附加到后面。 
//如果只给了一个数组并且该数组是数字索引的,则键名会以连续方式重新索引。 
$array1 = array("color" => "red", 2, 4);
$array2 = array("a", "b", "color" => "green", "shape" => "trapezoid", 4);
$result = array_merge($array1, $array2);
输出:
Array
(
    [color] => green
    [0] => 2
    [1] => 4
    [2] => a
    [3] => b
    [shape] => trapezoid
    [4] => 4
)

$array1 = array();
$array2 = array(1 => "data");
$result = array_merge($array1, $array2);
输出: 数字键名将会被重新编号
Array
(
    [0] => data
)

//两个数组相加 $array + $array  
//第一个数组的键名将会被保留。在两个数组中存在相同的键名时,
//第一个数组中的同键名的元素将会被保留,第二个数组中的元素将会被忽略 
$array1 = array(0 => 'zero_a', 2 => 'two_a', 3 => 'three_a');
$array2 = array(1 => 'one_b', 3 => 'three_b', 4 => 'four_b');
$result = $array1 + $array2;
输出 $result:
array(5) {
  [0]=>
  string(6) "zero_a"
  [2]=>
  string(5) "two_a"
  [3]=>
  string(7) "three_a"
  [1]=>
  string(5) "one_b"
  [4]=>
  string(6) "four_b"
}

8.shuffle($arr);  将数组的顺序打乱,随机数组

9.array_filter($arr,"function"); 数组过滤
 使用回调函数过滤数组中的每个元素,如果回调函数为TRUE,数组的当前元素会被包含在返回的结果数组中,数组的键名保留不变
 $array = array(6, 7, 8, 9, 10, 11, 12);
function even($var)
{
    return($var%2 == 0);
}
print_r(array_filter($array, "even"));
输出:
Array
(
    [0] => 6
    [2] => 8
    [4] => 10
    [6] => 12
)

10.array_map($callback,$arr);返回数组,是为 array 每个元素应用 callback函数之后的数组

11.array_push($arr,"apple","pear");  将一个或多个元素压入数组栈的末尾(入栈),返回入栈元素的个数

12.array_pop($arr);  将数组栈的最后一个元素弹出(出栈)

13.array_sum($arr);  对数组内部的所有元素做求和运算

24. 禁用cookie怎么实现session

禁用了cookie,session_id就不能保存,而服务器正是根据session_id来判断用户的session,导致无法判断当前用户。
实现:
1.服务端获取session_id
session_start();
$session_id = session_id();
2.可在url地址上加上session_id
3.可使用缓存,redis,memacahe来实现

25. 哪些情况下索引不会被用到

1.like 后面的字符当首位为通配符时不走索引;
2.如果条件中有or,即使其中有部分条件带索引也不会使用;
3.表的数据库小或者需要选择大部分数据,估计使用全表扫描要比使用索引快,则不使用索引;
4.索引字段 is null 不走索引;
5. where 子句里对索引列上有数学运算,用不上索引; 

26.聚簇索引和非聚簇索引 的区别

聚簇索引:
聚簇索引,实际存储的循序结构与数据存储的物理机构是一致的,所以通常来说物理顺序结构只有一种,那么一个表的聚簇索引也只能有一个。
通常默认都是主键,设置了主键,系统默认就为你加上了聚簇索引。
如果不想拿主键作为聚簇索引,其他字段作为索引,这就需要你在设置主键之前自己手动的先添加上唯一的聚簇索引,然后再设置主键。

非聚簇索引:
非聚簇索引记录的物理顺序与逻辑顺序没有必然的联系,与数据的存储物理结构没有关系。
一个表对应的非聚簇索引可以有多条,根据不同列的约束可以建立不同要求的非聚簇索引。

27.定义常量

const CONSTANT = '123'    //定义常量 
defined('CONSTANT','123') //定义常量 
defined()  //检查常量是否定义,返回一个bool 值

NOTE:
//常量定义只能为标量, 标量是字符串,整型,浮点型,布尔型数据类型。
//如果定义了两个相同的常量,前者起作用,因为常量一旦定义就不能被重新定义或者取消定义。
const 不能在函数体内定义
和使用 define() 来定义常量相反的是,使用 const 关键字定义常量必须处于最顶端的作用区域,
因为用此方法是在编译时定义的。这就意味着不能在函数内,循环内以及 if 语句之内用 const 来定义常量。
// 常用的php内置常量
__FILE__ :当前文件所在服务器路径
__LINE__:文件中的当前行号
__FUNCTION__:当前函数名
__CLASS__:当前类名
PHP_VERSION:当前php版本
PHP_OS:当前php运行操作系统

28.大小写

函数名不区分大小写,变量名区分大小写。

29.mysql锁

mysql 锁分类:共享锁,排他锁。

加锁语句:
select * from table where id=1 lock in share mode; (加共享锁)
select * from table where id=1 for update; (加排他锁)

MyISAM 和 InnoDB 锁的区别:
MyISAM中使用表锁,因为MyISAM在修改数据记录的时候会将整个表锁起来;
InnoDB使用的是行锁,是通过给索引上的索引项加锁来实现的。
InnoDB这种行锁实现特点意味着:只有通过索引条件检索数据,InnoDB才使用行级锁,否则,InnoDB将使用表锁!

表级锁:开销小,加锁快;不会出现死锁;锁定粒度大,发生锁冲突的概率最高,并发度最低。
行级锁:开销大,加锁慢;会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度也最高。

mysql死锁产生:
在InnoDB中,使用行锁机制,于是,锁通常是逐步获得的,这就决定了在InnoDB中发生死锁是可能的。
即将分享的四种死锁的锁冲突分别是:
不同表的相同记录行索引锁冲突、主键索引锁冲突、主键索引锁与非聚簇索引锁冲突、锁升级造成锁队列阻塞。

30.PHP 多个进程 同时写入一个文件成功

function writeData($filepath, $data) 
{ 
    $fp = fopen($filepath,'a');   //写入方式打开,指针在末尾
    do{ 
        usleep(100);  		 //当无法加文件锁时,等待100微秒
    }while (!flock($fp, LOCK_EX));   //LOCK_EX 取得独占锁定(写入的程序。
    $res = fwrite($fp, $data."\n");   //fwrite 写入文件
    flock($fp, LOCK_UN);    // LOCK_UN 释放锁定(无论共享或独占)。
    fclose($fp);   //关闭文件
    return $res; 
}

31.HTTP 请求状态码 详解

200 : 请求成功,请求的数据随之返回。
301 : 永久性重定向。
302 : 暂时行重定向。
400: 客户端请求的语法错误,服务器无法理解。
404 : 请求失败,请求的数据在服务器上未发现。
500 : 服务器错误。一般服务器端程序执行错误。
503 : 服务器临时维护或过载。这个状态时临时性的。

32.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值