php 构造方法,详解PHP对构造方法的识别

众所周知,由于历史原因,PHP之前是使用类名作为构造函数,在PHP 5中引入的新的构造函数__construct。为了实现向后兼容性,如果PHP 5在类中找不到 __construct() 函数,它就会尝试寻找旧式的构造函数,也就是和类同名的函数。

因此唯一会产生兼容性问题的情况是:类中已有一个名为 __construct() 的方法,但它却又不是构造函数。有如下一段代码:

}privatefunction__construct() {

}

}newFoo();die();

此时,输出为:

Fatal error:Call toprivateFoo::__construct() from invalid context

此时,PHP识别出来的构造函数是__construct,因为是private,于是在外部调用出错。好吧,我们从PHP的C源码中查找一下原因吧。从SQL的扩展类中直接查找类的定义开始:

spl_iterators.c 3228行 REGISTER_SPL_STD_CLASS_EX(IteratorIterator,spl_dual_it_new,spl_funcs_IteratorIterator);///spl_functions.h 31行#define REGISTER_SPL_STD_CLASS_EX(class_name, obj_ctor, funcs) \spl_register_std_class(&spl_ce_## class_name, # class_name, obj_ctor, funcs TSRMLS_CC);

//spl_functions.c 41行PHPAPI void spl_register_std_class(zend_class_entry**ppce,char*class_name,void*obj_ctor,constzend_function_entry*function_list TSRMLS_DC)//spl_functions.c 2235行ZEND_API zend_class_entry*zend_register_internal_class(

zend_class_entry*orig_class_entry TSRMLS_DC)/*{{{*///调用do_register_internal_class函数

//zend_API.c 2169行staticzend_class_entry*do_register_internal_class(

zend_class_entry*orig_class_entry,zend_uint ce_flags TSRMLS_DC)/*{{{*///调用zend_register_functions(class_entry,class_entry->builtin_functions,&class_entry->function_table,MODULE_PERSISTENT TSRMLS_CC);//zend_API.c 1795行/*Look for ctor, dtor, clone

* If it's an old-style constructor, store it only if we don't have

* a constructor already.*/if((fname_len==class_name_len)&&!memcmp(

lowercase_name,lc_class_name,class_name_len+1)&&!ctor) {

ctor=reg_function;

}elseif((fname_len==sizeof(ZEND_CONSTRUCTOR_FUNC_NAME)-1)&&!memcmp(lowercase_name,ZEND_CONSTRUCTOR_FUNC_NAME,sizeof(ZEND_CONSTRUCTOR_FUNC_NAME))) {

ctor=reg_function;

}

scope->constructor=ctor;//在1961行 确认构造函数

以上代码为PHP 5.3.0版本

从以上跟踪流程来看,程序在注册所有函数时,如果存在__construct(即ZEND_CONSTRUCTOR_FUNC_NAME)时,会覆盖class_name(类名)的构造函数,使其作为常规的成员函数存在。如下所示代码:

1.<?php2.classFoo {3.4.publicfunctionFoo() {5.echo'Foo';6.}7.8.publicfunction__construct() {9.echo'__construct';10.}11.}12.13.$foo=newFoo();14.$foo->Foo();

对于在前面的示例中的报错,我们可以在zend/zend_object_handlers.c 1057行ZEND_API union _zend_function *zend_std_get_constructor(zval *object TSRMLS_DC)找到出处。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值