变量的作用域问题(重点)

简单来说,有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关键字,都能实现:在局部作用域中使用全局变量。

递归函数(重点/难点)

基本含义:就是一个函数内部再调用该函数本身的一种情形,这是语法形式上的。

具体场景是:

  1. 如果要解决的“最终问题”,可以根据比该问题“小一级”的问题的答案而得到解决,
  2. 并且,该“小一级”的问题,还可以根据比其“更小一级”的问题的答案而得到解决,
  3. 以此类推,直到“最小一级”的问题。如果最小一级问题已知,则最终的问题也就解决了。

危险:如果函数在执行的过程中没有一个“不再调用”的终结机制,那么就会出现“停不下来”的现象。

原理:

递归函数思想
递归函数思想

递归调用过程的代码演示:

分析一下代码的输出结果:

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