从ThinkPHP源码讲解_initialize()与__construct()

517 查看

最近在学习ThinkPHP框架,看到_initialize()函数,也正好做个小总结

简单看了谷歌,百度上的教程,感觉噼里啪啦说了好多,全在进行测试,都没说到点子上~

实验版本:ThinkPHP 3.2.3,PHP5.6。

_initialize()函数的出现是为了我们可以在子类中同时调用父类与子类的构造函数

想要搞明白最简单方法的就是打开TP的源码,最靠谱。

路径:ThinkPHP3.2.3/ThinkPHP/Library/Think/Controller.class.php.(△Controller是一个抽象类△)

我们可以看到:

在此处Controller这个抽象类重写了__construct()方法,重点看看三条红线,这里就是_initialize()的全部,其实压根没有特别声明一个_initialize()方法然后赋予它特别的功能。

可以看出,_initialize()具备构造函数的功能纯粹是因为它正好在__construct()里而已(当实例化类时,构造函数__construct运行,如果当前类存在_initialize()方法,顺带执行了_initialize())

我们看下在原生PHP中如何完成同时调用子类父类构造函数需求的。

那么在ThinkPHP中呢?(不要在意格式)

原生代码想要完成需求需要在子类中调用运行parent::__construct()。

而ThinkPHP在父类经过特别处理后,_initialize()本身就就可以完成这个功能了,其实我认为ThinkPHP的_initialize函数本意就是用来在需要同时调用父类与子类构造函数时用的。

总结一下在TP中_initialize()和__construct()使用的注意事项

  • 如果_initialize()和__construct()同时出现的话,那么_initialize()将会无效,因为此时__construct()已经重写,不再调用_initialize()。

  • 如果想要父子类构造函数同时调用,一定要在父类的__construct()中进行处理:

    if(method_exists($this,’_initialize’)){
    
    $this -> _initialize();
    
    }
    
  • 两者都不能被子类覆盖重写,否则父子类同时调用的功能将会失效

就目前而言(ThinkPHP3.2.3)中,_initialize()是有漏洞的,毕竟还需要我们在手动进行一次处理,这样的话_initialize()的作用就剩下命名约束了,因为只要愿意的话完全可以将_initialize改成其他命名,相信官方在后期版本会改进这个函数。