搜尋
首頁php教程php手册面向对象中PHP构造方法的识别

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

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

<ol class="dp-c">
<li class="alt"><span><span><?php    </span></span></span></li>
<li>
<span class="keyword">class</span><span> Foo {    </span>
</li>
<li class="alt"><span>     </span></li>
<li>
<span>    </span><span class="keyword">public</span><span> </span><span class="keyword">function</span><span> Foo() {    </span>
</li>
<li class="alt"><span>     </span></li>
<li><span>    }    </span></li>
<li class="alt"><span>     </span></li>
<li>
<span>    </span><span class="keyword">private</span><span> </span><span class="keyword">function</span><span> __construct() {    </span>
</li>
<li class="alt"><span>     </span></li>
<li><span>    }    </span></li>
<li class="alt"><span>}    </span></li>
<li><span>     </span></li>
<li class="alt">
<span class="keyword">new</span><span> Foo();    </span>
</li>
<li>
<span class="keyword">die</span><span>();   </span>
</li>
</ol>

  此时,输出为:

  Fatal error: Call to private Foo::__construct() from invalid context

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

<ol class="dp-c">
<li class="alt"><span><span>spl_iterators.c 3228行 REGISTER_SPL_STD_CLASS_EX(IteratorIterator, spl_dual_it_new, spl_funcs_IteratorIterator);    </span></span></li>
<li>
<span class="comment">///spl_functions.h 31行   </span><span> </span>
</li>
<li class="alt"><span>#define REGISTER_SPL_STD_CLASS_EX(class_name, obj_ctor, funcs) \    </span></li>
<li><span> spl_register_std_class(&spl_ce_ ## class_name, # class_name, obj_ctor, funcs TSRMLS_CC);    </span></li>
<li class="alt">
<span class="comment">//spl_functions.c 41行   </span><span> </span>
</li>
<li>
<span>PHPAPI void spl_register_std_class(zend_class_entry ** ppce, char * class_name, void * obj_ctor, </span><span class="keyword">const</span><span> zend_function_entry * function_list TSRMLS_DC)    </span>
</li>
<li class="alt"><span>     </span></li>
<li>
<span class="comment">//spl_functions.c 2235行   </span><span> </span>
</li>
<li class="alt">
<span>ZEND_API zend_class_entry *zend_register_internal_class(zend_class_entry *orig_class_entry TSRMLS_DC) </span><span class="comment">/* {{{ */</span><span>    </span>
</li>
<li>
<span class="comment">//调用do_register_internal_class函数   </span><span> </span>
</li>
<li class="alt"><span>     </span></li>
<li>
<span class="comment">//zend_API.c 2169行   </span><span> </span>
</li>
<li class="alt">
<span class="keyword">static</span><span> zend_class_entry *do_register_internal_class(zend_class_entry *orig_class_entry, zend_uint ce_flags TSRMLS_DC) </span><span class="comment">/* {{{ */</span><span>    </span>
</li>
<li>
<span class="comment">//调用   </span><span> </span>
</li>
<li class="alt"><span>zend_register_functions(class_entry, class_entry->builtin_functions, &class_entry->function_table, MODULE_PERSISTENT TSRMLS_CC);    </span></li>
<li><span>     </span></li>
<li class="alt">
<span class="comment">//zend_API.c 1795行   </span><span> </span>
</li>
<li>
<span class="comment">/* Look for ctor, dtor, clone   </span> </li>
<li class="alt"><span><span class="comment">* If it's an old-style constructor, store it only if we don't have   </span> </span></li>
<li><span><span class="comment">* a constructor already.   </span> </span></li>
<li class="alt"><span><span class="comment">*/</span><span>    </span></span></li>
<li>
<span class="keyword">if</span><span> ((fname_len == class_name_len) && !memcmp(lowercase_name, lc_class_name, class_name_len+1) && !ctor) {    </span>
</li>
<li class="alt"><span> ctor = reg_function;    </span></li>
<li>
<span>} </span><span class="keyword">else</span><span> </span><span class="keyword">if</span><span> ((fname_len == sizeof(ZEND_CONSTRUCTOR_FUNC_NAME)-1) && !memcmp(lowercase_name, ZEND_CONSTRUCTOR_FUNC_NAME, sizeof(ZEND_CONSTRUCTOR_FUNC_NAME))) {    </span>
</li>
<li class="alt"><span> ctor = reg_function;    </span></li>
<li><span>}     </span></li>
<li class="alt"><span>     </span></li>
<li>
<span>scope->constructor = ctor; </span><span class="comment">//在1961行 确认构造函数 </span><span> </span>
</li>
</ol>

  以上代码为PHP 5.3.0版本

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

<ol class="dp-xml">
<li class="alt"><span><span class="tag"></span><span class="tag-name">php</span><span>   </span></span></li>
<li><span>class Foo {    </span></li>
<li class="alt"><span>     </span></li>
<li><span>    public function Foo() {    </span></li>
<li class="alt"><span>        echo 'Foo';    </span></li>
<li><span>    }    </span></li>
<li class="alt"><span>     </span></li>
<li><span>    public function __construct() {    </span></li>
<li class="alt"><span>        echo '__construct';    </span></li>
<li><span>    }    </span></li>
<li class="alt"><span>}    </span></li>
<li><span>     </span></li>
<li class="alt">
<span>$</span><span class="attribute">foo</span><span> = </span><span class="attribute-value">new</span><span> Foo();    </span>
</li>
<li>
<span>$foo-</span><span class="tag">></span><span>Foo();  </span>
</li>
</ol>

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




陳述
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

SublimeText3 英文版

SublimeText3 英文版

推薦:為Win版本,支援程式碼提示!

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

mPDF

mPDF

mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境