为什么戏说php,戏说PHP的嵌套函数
PHP很早就支持嵌套函数了。并是不PHP5.3有闭包时才有的。然而,它却不是象JS,AS那样的闭包嵌套。即它的嵌套函数根本无闭包模式的逃脱。
PHP嵌套函数有一些特别之处。最特别的是,当外部函数被调用时,内部函数就会自动进入全局域中,成为新的定义函数。
所以,当外部函数确保是被调用一次,不会被调用二次,那么,可以写嵌套函数在其中。否则,就会引发致命错误。
但若我们仍想在一个可被调用多次的函数中定义一个内部函数,那么,该如何处理?
我们象在全局定义函数一样使用:
if (!function_exists(‘你的函数名‘)){
}
因此,全局函数的使用,常常用于一些特别的目的。同时要清楚,这样的函数,实际就是定义的全局函数。因此,它没有类对它封装,更没有命名空间。
看一下PHP手册中是如何说的:
Php代码
functionfoo()
{
functionbar()
{
echo"I don‘t exist until foo() is called.\n";
}
}
/* 现在还不能调用bar()函数,因为它还不存在 */
foo();
/* 现在可以调用bar()函数了,因为foo()函数
的执行使得bar()函数变为已定义的函数 */
bar();
?>
function foo()
{
function bar()
{
echo "I don‘t exist until foo() is called.\n";
}
}
/* 现在还不能调用bar()函数,因为它还不存在 */
foo();
/* 现在可以调用bar()函数了,因为foo()函数
的执行使得bar()函数变为已定义的函数 */
bar();
?>
PHP 中的所有函数和类都具有全局作用域,可以在内部定义外部调用,反之亦然。
我们不妨先看一下函数:
Php代码
functionouter($msg) {
functioninner($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()
//上面出错,是因为,外部函数被调用时,内部函数被重定义了。
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()
//上面出错,是因为,外部函数被调用时,内部函数被重定义了。
这里,我们再看一下,一个自动加载类,其中的做法
Php代码
staticpublicfunctioninitAutoload(){
//初始化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)
returntrue;
//因为没有spl_autoload_register,所以在类未加载成功时,要处理Callable List
foreach(self::$autoloadCallablesas$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)
returntrue;
}else{
trigger_error(‘Autoload method ‘.$method.‘ not found in class ‘.$className.‘!‘, E_USER_ERROR);
returnfalse;
}
}
}
}
}
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函数就不是这么几行代码。由此,这一做法确实精简了相当多的代码。
Php代码
staticpublicfunctiongetInstance($config= 0 ,$className=NULL){
if(!function_exists(‘trace‘)){//specially for ajax debug!!
functiontrace($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);
}
returnself::$instance;
}
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,戏说PHP的嵌套函数相关推荐
- python函数def里面嵌套def,python菜鸟求问关于嵌套函数中作用域范围应该怎么理解?,python嵌套,直接上代码def l(l...
python菜鸟求问关于嵌套函数中作用域范围应该怎么理解?,python嵌套,直接上代码def l(l 直接上代码def l(list): def d(): return list return d ...
- python之6-3嵌套函数
1. 嵌套函数 子函数可以继承父函数的变量 父函数返回子函数 子函数返回结果 看例子如下:结果是一个字符串fun1+fun2 #!/usr/bin/env python# coding=utf-8de ...
- python之路--嵌套函数、匿名函数、高阶函数。函数的递归
嵌套函数 函数里不仅可以写代码,还可以嵌套函数 name = "小猿圈"def change():name = "小猿圈,自学编程"def change2(): ...
- Python的嵌套函数使用和闭包
文章目录 封装 - 数据隐藏 贯彻 DRY 原则 深入浅出python闭包 嵌套函数(Nested function)是在另一个函数(即:封闭函数)中定义的函数. 那么,一般在什么情况下使用嵌套函数? ...
- python用psf函数_Python 嵌套函数(高级用法)
Python 嵌套函数(高级用法) 一.嵌套函数(高级用法) 1.嵌套函数 函数的嵌套调用是在"函数调用中再调用其他函数".也就是说:函数嵌套允许在一个函数中调用另外一个函数.如下 ...
- python 避免多重条件嵌套_如何避免多层嵌套函数
我使用的python代码有4个层次的嵌套函数.您可以看到以下示例:class TestClass: def __init__(self): pass def calculate(self, x, y) ...
- Python教程:作用域与嵌套函数
一.全局变量与局部变量 在函数中定义的变量为局部变量,在程序中一开始定义的变量为全局变量: 全局变量作用域是整个程序,局部变量作用域是定义该变量的函数: 当全局变量与局部变量同名时,在定义局部变量的函 ...
- Python基础教程:嵌套函数、闭包
嵌套函数 在函数中定义另一个函数称为嵌套函数. 嵌套函数可以访问包围范围内的变量. def print_msg(msg):def printer():print(msg)printer()print_ ...
- matlab私有函数,MATLAB 嵌套函数,子函数,私有函数,重载函数
MATLAB函数嵌套 MATLAB中M文件有两种类型,脚本M文件和函数M文件.脚本M文件是将可执行程序语句放入M文件中,就像在命令窗口那样,按其语句顺序及逻辑关系执行,可以理解为一般的顺序执行程序语句 ...
最新文章
- 【leetcode】394. Decode String
- php 文件管理系统_如何编写程序实现图书管理系统的个人图书借阅查询功能
- 【ubuntu+opencv3】ubuntu16.04+qt5+opencv3.2.0编译与安装
- nodejs中使用worker_threads来创建新的线程
- 单点登录系统cas资料汇总
- MOSS中自定义WebService
- Exynos4412 内核移植(四)—— MMU 相关知识解析
- 2018年,自然语言处理很全的应用与合作
- 计算机应用技术自我分析,计算机应用*个人自我鉴定
- 信息学奥赛C++语言:求三位数的值
- 惯性导航系统--百科笔记
- python 获取macd数据_如何通过MACD函数获取MACD的5分钟对应值
- 外卖骑手,巨头的炮灰
- 爬虫(2)之re 爬取淘宝网
- 史莱姆方块java_史莱姆 - Minecraft Wiki,最详细的官方我的世界百科
- echarts-箱线图(盒须图)
- 蓝桥杯嵌入式(一)学习准备
- 小白理解k-svd算法
- 2022哈工大计算机系统大作业——程序人生
- 队列Q(Wannafly挑战赛19)
热门文章
- 咬文嚼字 | 信息化和数字化的本质区别是什么?
- c语言海盗分1000个金币,海盗分金币问题及代码实现
- 开源OA办公平台搭建教程:O2OA+Arduino实现物联网应用(五)
- 论文阅读:Faster AutoAugment: Learning Augmentation Strategies using Backpropagation
- 关键帧提取——帧差法提取关键帧(2)
- 可擦写光盘不能擦除和刻写_什么是可擦写光盘 什么时候要用到可擦洗光盘(擦除光盘)...
- 东方博宜OJ——1.整数运算题解
- DICOM:fo-dicom之C-STORE再分析‘解决System.ObjectDisposedException异常’
- python做大数据开发是做什么的_大数据开发是干什么的?
- Ubuntu16.04下NVIDIA显卡驱动安装(华硕飞行堡垒7代,1660ti)