Maison > Article > développement back-end > use语句必须放在函数外面的理由是什么?
下面为何不可以
<code>function done() { require_once 'vendor/autoload.php'; use Qiniu\Auth; use Qiniu\Storage\UploadManager; #省略 }</code>
?>
必须写成
<code>require_once 'vendor/autoload.php'; use Qiniu\Auth; use Qiniu\Storage\UploadManager; function done() { #省略 }</code>
?>
下面为何不可以
<code>function done() { require_once 'vendor/autoload.php'; use Qiniu\Auth; use Qiniu\Storage\UploadManager; #省略 }</code>
?>
必须写成
<code>require_once 'vendor/autoload.php'; use Qiniu\Auth; use Qiniu\Storage\UploadManager; function done() { #省略 }</code>
?>
语言结构使然,如果你了解PHP命名空间相关性质我想你是不会这么问的。
use
的作用仅仅是用短名称替代长名称,或者是用别名替代本名,是一个没有语义和实效的“语法糖”。
所以消灭use
的运行时开销是一个非常合理的选择。因此php规定use
在解析阶段(parse)就被处理。
和运行时才现场执行(相当于语句)的echo
、require
等不一样,use
语言结构是在解析(parse)阶段预先扫描、提早处理的。以上是前提。
而解析操作本身,非常的单纯,仅仅是从头推到尾,识别一个个的语言关键字,并确保语法规则不被违反。我们可以做一个简单的试验:
<code><?php echo '可爱的紫妹纸是永远的 17 岁!'; die(); { 我就是叫紫妈怎么了?有本事突然从我背后出现,把我的脸按在键盘上5rt4n7tojd87tg2435t4q34 }</code></code>
我不知道紫妈会不会用我的脸滚键盘,但我知道php肯定不会让我过解析——
你说第 5 行永远都不会执行?解析器根本不知道,也不关心。
但对于一个花括号括住的作用域(scope)而言,事情就变得复杂了。因为一个小作用域的执行顺序很可能是乱的——可以回头、可以通过调用来乱跳等等。例如:
<code>namespace NS1; class ClassName { } function f() { return new ClassName(); } for ($i=0; $i</code>
如果我们认为use
会影响它后边的所有内容,那么此时$a
和$b
的赋值语句到底在不在use
的后边?
按照语义,第1次循环不在,第2次循环在,也就是说同一行会产生两种不同的语义。
但解析器不可能理解,也不可能维护得了这种逻辑。实现这种逻辑,必然产生一个运行时的开销(因为要介入程序运行当时才能确定的状态),而这是use
的设计本意要避免的。
所以use
只能摆在文件的最外层作用域中。只有这个作用域的范围是一线平推,不可能回退,也不可能出现跳转。
试分析以下use
真正的作用范围,就可以看到逻辑中,处处都是为了方便解析器处理而设计的:
从use
出现的行开始(简单的开始规则)
见到namespace
结束(简单的终止规则)
见到文件尾结束(解析器的运行不能跨文件)
事实上和严谨设计、环环相扣的语言特性不同,很多的语法糖都并没有太多的道理可讲。
能像use
这样,从最初的设计目的,从而推导出其设计必然限制的语法糖,其实挺少的。
对于语法糖,死记、活用、理解原理但别想太多,这才是我们作为语言使用者的营生之道。