PHP很早就支持嵌套函数了。并是不PHP5.3有闭包时才有的。然而,它却不是象JS,AS那样的闭包嵌套。即它的嵌套函数根本无闭包模式的逃脱。

PHP嵌套函数有一些特别之处。最特别的是,当外部函数被调用时,内部函数就会自动进入全局域中,成为新的定义函数。

所以,当外部函数确保是被调用一次,不会被调用二次,那么,可以写嵌套函数在其中。否则,就会引发致命错误。

但若我们仍想在一个可被调用多次的函数中定义一个内部函数,那么,该如何处理?

我们象在全局定义函数一样使用:

if (!function_exists('你的函数名')){

}

因此,全局函数的使用,常常用于一些特别的目的。同时要清楚,这样的函数,实际就是定义的全局函数。因此,它没有类对它封装,更没有命名空间。

看一下PHP手册中是如何说的:

  1. <?php
  2. function foo()
  3. {
  4. function bar()
  5. {
  6. echo "I don't exist until foo() is called.\n";
  7. }
  8. }
  9. /* 现在还不能调用bar()函数,因为它还不存在 */
  10. foo();
  11. /* 现在可以调用bar()函数了,因为foo()函数
  12. 的执行使得bar()函数变为已定义的函数 */
  13. bar();
  14. ?>
<?php
function foo()
{
function bar()
{
echo "I don't exist until foo() is called.\n";
}
} /* 现在还不能调用bar()函数,因为它还不存在 */ foo(); /* 现在可以调用bar()函数了,因为foo()函数
的执行使得bar()函数变为已定义的函数 */ bar(); ?>

PHP 中的所有函数和类都具有全局作用域,可以在内部定义外部调用,反之亦然。

我们不妨先看一下函数:

  1. function outer( $msg ) {
  2. function inner( $msg ) {
  3. echo 'inner: '.$msg.' ';
  4. }
  5. echo 'outer: '.$msg.' ';
  6. inner( $msg );
  7. }
  8. inner( 'test1' );  // Fatal error:  Call to undefined function inner()
  9. //上面出错,是因为外部函数还没有调用,所以出错。
  10. outer( 'test2' );  // outer: test2 inner: test2
  11. inner( 'test3' );  // inner: test3
  12. outer( 'test4' );  // Fatal error:  Cannot redeclare inner()
  13. //上面出错,是因为,外部函数被调用时,内部函数被重定义了。
function outer( $msg ) {
function inner( $msg ) {
echo 'inner: '.$msg.' ';
}
echo 'outer: '.$msg.' ';
inner( $msg );
} inner( 'test1' ); // Fatal error: Call to undefined function inner()
//上面出错,是因为外部函数还没有调用,所以出错。
outer( 'test2' ); // outer: test2 inner: test2
inner( 'test3' ); // inner: test3
outer( 'test4' ); // Fatal error: Cannot redeclare inner()
//上面出错,是因为,外部函数被调用时,内部函数被重定义了。

这里,我们再看一下,一个自动加载类,其中的做法

  1. static public function initAutoload(){
  2. //初始化Autoload Callable List
  3. self::setAutoloadCallableList();
  4. //初始化 $classList
  5. self::$classList = uxAutoloadConfig::getClassList();
  6. //如果有spl_autoload_register,则直接设置
  7. if (function_exists('spl_autoload_register')){
  8. ini_set('unserialize_callback_func', 'spl_autoload_call');
  9. spl_autoload_register(array('uxAutoload', 'splSimpleAutoload'));
  10. }elseif (!function_exists('__autoload')){  //否则要使用__autoload函数。
  11. ini_set('unserialize_callback_func', '__autoload');
  12. //因为没有spl_autoload, 所以, 这里要定义一个__autoload函数.
  13. function __autoload($class){
  14. if( self::splSimpleAutoload($class)== true)
  15. return true;
  16. //因为没有spl_autoload_register,所以在类未加载成功时,要处理Callable List
  17. foreach(self::$autoloadCallables as $key => $callable ){
  18. if (class_exists($class, false)){
  19. $classObj=self::$autoloadObjectList[$callable[0]];
  20. }else{
  21. $className=$callable[0];
  22. $classObj = new $className();
  23. self::$autoloadObjectList[$class] = &$classObj;
  24. }
  25. if (method_exists($classObj,$callable[1])){
  26. $method=$callable[1];
  27. if ($classObj->$method($class)==true)
  28. return true;
  29. }else{
  30. trigger_error('Autoload method '.$method.' not found in class '.$className.'!', E_USER_ERROR);
  31. return false;
  32. }
  33. }
  34. }
  35. }
  36. }
static public function initAutoload(){
//初始化Autoload Callable List
self::setAutoloadCallableList();
//初始化 $classList
self::$classList = uxAutoloadConfig::getClassList(); //如果有spl_autoload_register,则直接设置
if (function_exists('spl_autoload_register')){
ini_set('unserialize_callback_func', 'spl_autoload_call');
spl_autoload_register(array('uxAutoload', 'splSimpleAutoload'));
}elseif (!function_exists('__autoload')){ //否则要使用__autoload函数。
ini_set('unserialize_callback_func', '__autoload'); //因为没有spl_autoload, 所以, 这里要定义一个__autoload函数.
function __autoload($class){
if( self::splSimpleAutoload($class)== true)
return true;
//因为没有spl_autoload_register,所以在类未加载成功时,要处理Callable List
foreach(self::$autoloadCallables as $key => $callable ){
if (class_exists($class, false)){
$classObj=self::$autoloadObjectList[$callable[0]];
}else{
$className=$callable[0];
$classObj = new $className();
self::$autoloadObjectList[$class] = &$classObj;
}
if (method_exists($classObj,$callable[1])){
$method=$callable[1];
if ($classObj->$method($class)==true)
return true;
}else{
trigger_error('Autoload method '.$method.' not found in class '.$className.'!', E_USER_ERROR);
return false;
}
}
}
}
}

很明显,它是定义了一个内部函数function __autoload($class),以防没有'spl_autoload_register'。而这个类的这个函数,任一request请求中,只运行一次。

但是,如果要运行多次,比如,以下函数中,定义了一个全局的TRACE函数。这个函数的目的是在用户使用标准MVC方式时,才提供此TRACE函数给用户使用。引导用户走正确的方向。实际上,也可以看出,如果用户用不到此类,很可能,TRACE函数就不是这么几行代码。由此,这一做法确实精简了相当多的代码。

  1. static public function getInstance($config = 0 ,$className=NULL){
  2. if (!function_exists('trace')){ //specially for ajax debug!!
  3. function trace($var){
  4. $string=print_r($var,true);
  5. require_once(UXERHDIR.'../uxLogger/uxLogger.class.php');
  6. uxLogger::getInstance()->logg('INFO',
  7. '/*************************** BEGIN INFO BY TRACE: ***************************\r\n '
  8. .$string
  9. .'/***************************  END INFO BY TRACE   ***************************\r\n' );
  10. }
  11. }
  12. if (!isset(self::$instance)){
  13. if (is_array($config)){
  14. $options=$config;
  15. }else{
  16. if ($config == NULL)
  17. $config = 0;
  18. $options=uxErrorHandlerConfig::get($config);
  19. }
  20. $class=($className==NULL)?'uxErrorHandler':$className;
  21. self::$instance = new $class($options);
  22. }
  23. return self::$instance;
  24. }
    static public function getInstance($config = 0 ,$className=NULL){
if (!function_exists('trace')){ //specially for ajax debug!!
function trace($var){
$string=print_r($var,true);
require_once(UXERHDIR.'../uxLogger/uxLogger.class.php');
uxLogger::getInstance()->logg('INFO',
'/*************************** BEGIN INFO BY TRACE: ***************************\r\n '
.$string
.'/*************************** END INFO BY TRACE ***************************\r\n' );
}
}
if (!isset(self::$instance)){
if (is_array($config)){
$options=$config;
}else{
if ($config == NULL)
$config = 0;
$options=uxErrorHandlerConfig::get($config);
}
$class=($className==NULL)?'uxErrorHandler':$className;
self::$instance = new $class($options);
}
return self::$instance;
}

可以看出,嵌套函数,是一种有条件全局函数,你可以控制,在什么情况下提供这样的全局函数给用户使用。但也需要注意,过多的全局函数则会产生“全局污染”,所以,不可多用。

戏说PHP的嵌套函数的更多相关文章

  1. python嵌套函数、闭包与decorator

    1 一段代码的执行结果不光取决与代码中的符号,更多地是取决于代码中符号的意义,而运行时的意义是由名字空间决定的.名字空间是在运行时由python虚拟机动态维护的,但是有时候我们希望能将名字空间静态化. ...

  2. swift 闭包+嵌套函数+extension+单例+嵌套函数+??

    //: Playground - noun: a place where people can play import UIKit //*******************嵌套函数********* ...

  3. python之6-3嵌套函数

    1. 嵌套函数 子函数可以继承父函数的变量 父函数返回子函数 子函数返回结果 看例子如下:结果是一个字符串fun1+fun2 #!/usr/bin/env python # coding=utf-8 ...

  4. Python函数学习——作用域与嵌套函数

    全局与局部变量 在函数中定义的变量称为局部变量,在程序的一开始定义的变量称为全局变量. 全局变量作用域是整个程序,局部变量作用域是定义该变量的函数. 当全局变量与局部变量同名时,在定义局部变量的函数内 ...

  5. 嵌套函数变量修改nonlocal & 全局变量修改global

    前几天在做一个简单的界面,单击Radiobutton保存字符串,在一个嵌套函数里面修改外部函数.一直不知道怎么修改,上网查了一下,搜关键字“嵌套函数修改变量”,找了好久,才得以解决. 对于python ...

  6. PHP中嵌套函数被调用时出现报错的问题

    对于初入门的PHP新手来说,在学习关于PHP函数嵌套的知识点时可能会有一定的难度.比如有的朋友在练习PHP函数嵌套相关问题时,会遇到调用内部函数时就会出现报错的情况等. 那么本篇文章就为大家详细得分析 ...

  7. Python3学习之路~3.2 递归、函数式编程、高阶函数、匿名函数、嵌套函数

    1 递归 在函数内部,可以调用其他函数.如果一个函数在内部调用自身本身,这个函数就是递归函数. def calc(n): print(n) if int(n / 2) == 0: return n r ...

  8. PHP:第三章——PHP中嵌套函数和条件函数

    PHP中的嵌套函数: <?php header("Content-Type:text/html;charset=utf-8"); function A(){ echo &qu ...

  9. node.js中的匿名函数, 回调函数和嵌套函数

    定义一个函数相信大家已经很熟悉了, 在javascript里的函数也是非常重要的, 使用率非常高, 有几种函数不是很好理解 一, 匿名函数 var remove = function(num1) { ...

随机推荐

  1. js DOM的几个常用方法

    <div id="div1">这是个测试</div> <p </p> <p </p> //js DOM的几个常用方法 / ...

  2. 【Linux】基于Linux的buffer和cache学习

    缓存(cached)是把读取过的数据保存起来,重新读取时若命中(找到需要的数据)就不要去读硬盘了,若没有命中就读硬盘.其中的数据会根据读取频率进行组织,把最频繁读取的内容放在最容易找到的位置,把不再读 ...

  3. verilog中符号位的扩展问题

    以下内容转自 艾米电子 - 使用有符号数,Verilog(http://www.cnblogs.com/yuphone/archive/2010/12/12/1903647.html) Verilog ...

  4. Windows 10 LNK File分析

    前情提要:警方接获线报,黑道份子阿强涉及制造与贩卖毒品,警方在其住处扣得笔记本电脑及数个U盘,送往实验室进行取证分析. 取证人员对证物进行证物镜像制作,并进行证物处理(Evidence Process ...

  5. Cookie禁用了,Session还能用吗?

    Cookie与Session,一般认为是两个独立的东西,Session采用的是在服务器端保持状态的方案,而Cookie采用的是在客户端保持状态的方案.Cookie分为两种,一种可以叫做session ...

  6. iOS Architecture Patterns

    By Bohdan Orlov on 21 Mar 2016 - 0 Comments iOS FYI: Slides from my presentation at NSLondon are ava ...

  7. C#解析JSON字符串总结

    JSON文件读取到内存中就是字符串,.NET操作JSON就是生成与解析JSON字符串. 操作JSON通常有以下几种方式: 1. 原始方式:按照JSON字符串自己来解析. 2. 通用方式[★★★★★]: ...

  8. webform网站相关数据控件和其他

    一.asp:Repeater <div class="bd"> <ul> <asp:Repeater ID="rept_slide" ...

  9. mac brew install redis

    在mac 下安装redis 执行brew install redis ==> Downloading http://download.redis.io/releases/redis-2.8.19 ...

  10. Incorrect integer value: '' for column 'id' at row 1

    最近在写个查询 插入语句的时候 我是这么写的 insert into test values('',$row[contentid],'".$tn."'); 结果搞死没插入进去  然 ...