php弱类型机制,php中容易忽视的基础知识点(坑)

Q1.关于弱类型

检查下面的代码,结果会输出什么?

$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 . "\"";

}

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++,因为是先使用x再进行自增运算,所以变为5 + $x++,此时x为6;再计算x++,还是先使用x再进行自增运算,所以变为5 + 6 = 11,此时x为7; $x---$x--; 同理。

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

另一个例子:

e5e97cba5c15

image.png

Q3.关于变量的引用;

$a = '1';

$b = &$a;

$b = "2$b";

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

$a='21' $b='21'

部分第一时间会想到 $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);

输出则为:

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

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

array(5) { [0]=> int(1) [1]=> int(2) [2]=> int(3) [3]=> int(4) [4]=> int(5) }

Q6.

$x应该是输出什么?

$x = true and false;

var_dump($x);

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

$x = true;true and false

所以输出为true

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后面会有连续的6个空格) 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比较就变成if(1>$v){...}。

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。

Q11.数组排序

给下列数组排序

$arr = array(

'0' => 'z1',

'1' => 'Z10',

'2' => 'z12',

'3' => 'Z2',

'4' => 'z3',

);

得到以下数组:

$arr = array(

'0' => 'z1',

'3' => 'Z2',

'4' => 'z3',

'1' => 'Z10',

'2' => 'z12',

);

看题就能知道,这是一个数组排序的问题,可以用sort()方法,但是却是将键值已字符串的形式排序,而且忽略大小写,用到了sort的第二个可选参数,再一点需要注意的是,排序完之后原来的键值对用的键名不变,所以需要用asort()方法:

asort($arr, SORT_STRING|SORT_FLAG_CASE|SORT_NATURAL);

需要注意的是:PHP5.4.0的版本中才添加了 sort_flags(asort方法的第二个参数) 内 SORT_NATURAL 和 SORT_FLAG_CASE 的支持。

Q12.

PHP_INT_MAX 是PHP中一个内核预定义常量,表示在PHP中所支持的最大整型常量值

假设var_dump(PHP_INT_MAX) 输出 int(9223372036854775807),那么

var_dump(PHP_INT_MAX + 1)输出的值是什么?

var_dump((int)(PHP_INT_MAX + 1))输出的值是什么?

为什么?

输出分别为:

float(9.22337203685E+18)

int(-9223372036854775808)

第一个转换成了float类型,是因为PHP控制特别大的整型数值时就是将该数值转换成float类型;

第二个变成了一个负整型数,这个涉及到二进制数的存储,数值的存储在计算几种都是以二进制的形式存储的,第一位是符号位。0表示正数,1表示负数,将最大的整型添加1将改变符号位为1,则为负数。

注意: 这个问题的关键不是具体数值,而是这些数值会发生什么样的变化,为什么会有这样的变化。

Q13.常见的输出语句

echo()

可以一次输出多个值,多个值之间用逗号分隔。echo是语言结构(language construct),而并不是真正的函数,因此不能作为表达式的一部分使用。没有返回值。

print()

print()输出字符串。print() 也是一个语言结构,所以不能被可变函数调用,因此你可以不必使用圆括号来括起它的参数列表。总是返回 1。

print_r() {bool print_r ( [mixed] $expression [, bool $return ] )}

**Note**: 参数 return是在 PHP 4.3.0 的时候加上的

print_r()显示关于一个变量的易于理解的信息。如果给出的是 [string]、[integer]或 [float],将打印变量值本身。如果给出的是 [array],将会按照一定格式显示键和元素。[object]与数组类似。如果想捕捉 print_r() 的输出,可使用 return (true 或 false)参数。若此参数设为 TRUE,print_r() 将不打印结果(此为默认动作),而是返回其输出。

var_dump() {void var_dump ( [mixed]$expression [, [mixed]$... ] )}

var_dump()此函数显示关于一个或多个表达式的结构信息,包括表达式的类型与值。数组将递归展开值,通过缩进显示其结构。

Q14. header的用法

1、跳转页面:

Header("Refresh:2; URL=http://localhost//session.php");//2秒后跳转

若等待时间为0,则与header("location:")等效。

Header("Location:http://localhost//session.php");//直接跳转

2、控制网页的内容:

header('HTTP/1.1 404 Not Found'); //返回response状态码

header('content-type:text/html;charset=utf-8'); //声明content-type

// 对当前文档禁用缓存

header('Expires: Mon, 26 Jul 1997 05:00:00 GMT'); // 一个过去的时间

header("Cache-Control: no-cache, must-revalidate");

header("Pragma: no-cache");

3、附件下载:

$filename = "tupian.jpg"; //文件路径 可以绝对路径也可以相对路径

header('Content-type: application/x-jpg'); //文件的类型

header('Content-Disposition: attachment; filename="保存时的文件名.jpg"'); //下载显示的名字

readfile("$filename");

Q15.

1.include()和require()的区别:

include()引入文件时,如果遇到错误会提示错误并继续执行;require()引入文件遇到错误时也提示错误但会终止程序的运行。

require()语句一般放在php脚本页面的最前面,php在执行前,就先读入require()引入的文件,文件的内容会变成此脚本的一部分,一旦出现错误,则立即退出程序。

include()语句一般在用到时才引入,所以通常是放在流程控制的处理区段中,php脚本在执行到它时,才会将文件包含进来。

即require()是预加载机制,位置在脚本最前面,一开始就引入所有可能用到的文件;include()是即用即加载,位置灵活。

2.include()、require()和include_once()、require_once()的区别:

include()、require()执行即包含文件,不会对引入的文件进行比较判断,可能会出现重复包含的情况;而include_once()、require_once()在包含时会先判断文件是否已经包含过了,如果已包含,则不再包含文件,这样的引入文件方式即可以节省资源,又可避免重复定义的错误。

Q16.

有一个数,需要判断是奇数还是偶数,没有任何条件且不能用循环,怎样实现?

$num = 3;

$arr = array(0=>'偶数',1=>'奇数');

echo $num.'是'.$arr[$num%2];

Q17.session和cookie的区别

session:一种保存在服务端的数据结构,用于追踪用户的状态,数据存在数据库或者服务端的文件中,依赖cookie进行数据传递,相对比较安全。

cookie:一种保存客户端数据的机制,用来存储用户信息,也是实现session的一种方式。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值