注意:这是在PHP中处理变量范围的参考问题。请关闭所有适合此模式的问题,以作为该问题的重复。
PHP中的“可变范围”是什么?一个.php文件中的变量可以在另一个文件中访问吗?为什么有时会出现 “未定义变量” 错误?
变量具有有限的“作用域”或“可从其访问的位置”。仅仅因为你写$foo = 'bar';一次 的地方 在你的应用程序并不意味着你可以参照$foo从 到处 其他的应用程序中。该变量$foo在一定范围内有效,并且只有相同范围内的代码才能访问该变量。
$foo = 'bar';
$foo
很简单:PHP具有 函数作用域 。那是PHP中存在的唯一一种范围分隔符。函数内部的变量仅在该函数内部可用。函数外部的变量在函数外部的任何位置都可用,但在任何函数内部均不可用。这意味着PHP中有一个特殊的作用域: 全局 作用域。在任何函数外部声明的任何变量都在此全局范围内。
<?php $foo = 'bar'; function myFunc() { $baz = 42; }
$foo在 全局 范围内,$baz在 本地 范围内myFunc。只有内部的代码myFunc可以访问$baz。只有 外部 代码才能myFunc访问$foo。双方都无法访问对方:
$baz
myFunc
<?php $foo = 'bar'; function myFunc() { $baz = 42; echo $foo; // doesn't work echo $baz; // works } echo $foo; // works echo $baz; // doesn't work
文件边界 不能分开 范围:
a.php
<?php $foo = 'bar';
b.php
<?php include 'a.php'; echo $foo; // works!
included代码与其他任何代码均适用相同的规则:仅适用于functions单独的范围。出于范围的目的,您可能会考虑包括诸如复制和粘贴代码之类的文件:
include
function
c.php
<?php function myFunc() { include 'a.php'; echo $foo; // works } myFunc(); echo $foo; // doesn't work!
在上面的示例中,a.php被包含在内部myFunc,内部的任何变量a.php仅具有局部函数作用域。仅仅因为它们 似乎 位于全局范围内a.php并不一定意味着它们存在,所以实际上取决于包含/执行代码的上下文。
每个新的function声明都引入了一个新的作用域,就是这么简单。
function foo() { $foo = 'bar'; $bar = function () { // no access to $foo $baz = 'baz'; }; // no access to $baz }
$foo = 'foo'; class Bar { public function baz() { // no access to $foo $baz = 'baz'; } } // no access to $baz
处理范围问题似乎很烦人,但是 有限的变量范围对于编写复杂的应用程序至关重要! 如果声明的每个变量在应用程序中的其他任何地方都可用,那么您将遍历所有变量,而没有真正的方法来跟踪什么更改。您只能为变量指定多个明智的名称,您可能希望$name在多个位置使用变量“ ”。如果您只能在应用程序中使用一次唯一的变量名,则必须采用真正复杂的命名方案,以确保变量是唯一的,并且不会从错误的代码段中更改错误的变量。
$name
观察:
function foo() { echo $bar; }
如果没有范围,上述功能会做什么?哪里$bar来的?它处于什么状态?它甚至被初始化了吗?每次都要检查吗?这是无法维持的。这带我们去…
$bar
function foo($bar) { echo $bar; return 42; }
变量$bar作为函数参数显式进入此范围。仅查看此函数,就可以清楚知道其使用值的来源。然后,它显式 返回 一个值。调用者有信心知道函数将使用哪些变量以及其返回值来自何处:
$baz = 'baz'; $blarg = foo($baz);
$foo = 'bar'; $baz = function () use ($foo) { echo $foo; }; $baz();
匿名函数$foo从其周围范围显式包括在内。请注意,这与 全局 范围不同。
global
如前所述,全局范围有些特殊,函数可以从中显式导入变量:
$foo = 'bar'; function baz() { global $foo; echo $foo; $foo = 'baz'; }
此函数使用并修改全局变量$foo。 不要这样做! (除非您真的真的真的知道自己在做什么,即使如此:不要!)
该函数的所有调用者看到的是:
baz(); // outputs "bar" unset($foo); baz(); // no output, WTF?! baz(); // outputs "baz", WTF?!?!!
没有迹象表明该功能有任何 副作用 ,但是确实有。由于某些函数不断修改 并需要 某些全局状态,因此这很容易成为一团糟。您希望函数是 无状态的 ,仅对它们的输入起作用并返回定义的输出,无论您多次调用它们。
您应该尽可能避免以任何方式使用全局范围;最肯定的是,您不应该将变量从全局范围“拉”到局部范围。