变量的作用域问题(重点)
简单来说,有3种作用域:局部作用域,全局作用域,超全局作用域;
相对应的,有3种变量: 局部变量, 全局变量, 超全局变量;
局部作用域与局部变量:
就是函数内部范围的作用域,其中定义的变量就是局部变量(包括形参也是局部变量)。
局部变量只能在其所在的局部作用域中使用(访问)。
局部变量在函数调用结束时,会被自动销毁(可以理解为函数执行结束,该执行空间也被销毁了)。
$v1 = 1;
function f1( $p1 , $p2 ){ //此函数范围就是一个局部作用域,其中有3个局部变量:$p1, $p2, $v2
$v2 = 2;
echo $ v1; //报错!变量未定义
echo $v2;
}
f1(); //
echo $v1;
echo $v2; //报错!变量未定义
静态变量:一个特殊的局部变量
含义:在函数内部,使用static关键字修饰的变量。
形式:示例
静态变量的特点:静态变量的值不会在函数调用结束时被销毁,而是会一直保留。也就是说,当再次调用函数时,该变量(的值)还能继续使用。
对比演示:一个普通局部变量,一个静态变量,都进行简单++操作。多次调用函数,并输出他们的值。
//一个普通局部变量,一个静态变量,
//都进行简单++操作。多次调用函数,并输出他们的值。
function static_demo()
{
$p1 = 1;
static $p2 = 1; //静态变量修饰符
$p1++;
$p2++;
echo "<br>p1=$p1, p2=$p2";
}
static_demo();
static_demo();
static_demo();
全局作用域与全局变量:
就是函数外部范围的作用域,其中定义的变量就是全局变量。
全局变量只能在其所在的全局作用域中可以直接使用(访问)。
超全局作用域与超全局变量:
包括局部作用域和全局作用域的的整个作用域范围。
超全局变量可以在所有范围中使用(访问)。
实际上,只有有限的10来个系统预定义变量是超全局变量,包括:$_GET, $_POST,$_REQUEST等。
所以,系统预定义变量,也被统称为超全局变量。
PHP中的不同作用域的图示:
一个特别的超全局变量:$GLOBALS
它也是一个数组,其中存储了我们自己定义的所有全局变量。
每个全局变量的变量名,就是$GLOBALS数组的一个单元。
比如:
$v1 = 1; //这一行执行,就有了一个这个:$GLOBALS[‘v1’], 其值为1
$v2 = 'abc'; //这一行执行,就有了一个这个:$GLOBALS[‘v2’], 其值为’abc’
$v3 = true; //这一行执行,就有了一个这个:$GLOBALS[‘v3’], 其值为true
它可以让我们在局部作用域范围中,使用全局变量,方式如下:
$v1 = 10; //全局变量
function func1( ){
echo $GLOBALS['v1']; //输出10;
echo $v1; //报错:变量v1未定义
$s1 = $GLOBALS['v1'] * 5; //结果为50;
$s2 = $v1 * 5; //报错:变量v1未定义
}
一个特别的的关键字:global
作用:用于在局部作用域中,修饰一个跟全局变量同名的局部变量。此时该局部变量也可以使用全局变量的值了——实际上他们其实是类似变量引用关系。
演示:
//演示$GLOBALS
$v1 = 1;
function func1(){
//echo $v1; //报错!
//可以借助GLOBALS超全局变量来使用全局变量
echo "<br>在函数中:v1 = " . $GLOBALS['v1'];
//也可以去修改全局变量
$GLOBALS['v1']++;
}
func1();
echo "<br>在函数外:v1 = " . $v1;
echo "<hr>";
//对比关键字global
$n1 = 1;
function func2(){
//其含义是:定义一个局部变量$n1
//并且,该变量跟全局的$n1同名,并处于”引用关系“
global $n1;
echo "<br>在函数中:n1 = " . $n1;
$n1++;
}
func2();
echo "<br>在函数外:n1 = " . $n1;
总结:$GLOBALS数组(变量)和global关键字,都能实现:在局部作用域中使用全局变量。
递归函数(重点/难点)
基本含义:就是一个函数内部再调用该函数本身的一种情形,这是语法形式上的。
具体场景是:
- 如果要解决的“最终问题”,可以根据比该问题“小一级”的问题的答案而得到解决,
- 并且,该“小一级”的问题,还可以根据比其“更小一级”的问题的答案而得到解决,
- 以此类推,直到“最小一级”的问题。如果最小一级问题已知,则最终的问题也就解决了。
危险:如果函数在执行的过程中没有一个“不再调用”的终结机制,那么就会出现“停不下来”的现象。
原理:
递归调用过程的代码演示:
分析一下代码的输出结果:
function f1( $n )
{
$n++;
echo "$n ";
if($n < 5)
{
f1( $n );
}
echo "$n ";
}
f1(1);
案例1:计算5的阶层;
分析:数学上阶乘可以这样来描述:一个数n的阶乘,是n-1的阶乘,乘以n的结果!假设,我们有一个函数 jieceng(),它可以计算任意正整数n的阶乘,类似这样:
$n = 5; //或等于10, 13,等等都无所谓。
$result = jiecheng($n);
案例2:计算斐波那契数列第10项的值:1, 1, 2, 3, 5, 8, 13, 21, ......
假设有个函数,可以计算斐波那契数列的第n项:
//计算斐波那契数列第10项的值:1, 1, 2, 3, 5, 8, 13, 21, ......
//规则:从第3项开始,每一项都是其前面两项的和
//假设有个函数,可以计算斐波那契数列的第n项:
//此可以求出斐波那契的第n项的值
function feibo( $n )
{
if($n > 2) //第3项往后的项
{
$re = feibo($n-1) + feibo($n-2);
return $re;
}
return 1;
}
echo "<br>第1项的值为:" . feibo(1);
echo "<br>第2项的值为:" . feibo(2);
echo "<br>第3项的值为:" . feibo(3);
echo "<br>第4项的值为:" . feibo(4);
echo "<br>第5项的值为:" . feibo(5);
echo "<br>第10项的值为:" . feibo(10);
echo "<br>第20项的值为:" . feibo(20);