Heim  >  Artikel  >  php教程  >  一步步编写PHP的Framework(四)

一步步编写PHP的Framework(四)

WBOY
WBOYOriginal
2016-06-21 08:50:421108Durchsuche

 

上一篇文章我提了一个问题,怎么实现自动载入一个类呢?

实际上PHP已经有相应的机制可以实现这个功能了,这个机制就是autoload,它会在试图使用尚未被定义的类时自动调用。

现在我们可以在Route.php中的开头定义autoload函数(当然,这样是不规范的,但是为了简单,先这么做)。

我们现在的自动导入需要导入两类文件,一类是框架类文件,另外一类是用户应用模块的类文件,为了简化代码,可以假设框架的所有文件都存放在/Library/Test目录下面,用户类文件都存放在/UserApps/Modules目录下面,并且只会在Controllers,Models,Helpers这三个目录下面,并且这几个目录下面没有子目录。

对于这个自动导入的函数,它需要首先尝试导入框架类文件,如果该文件不存在,说明是用户类文件,然后再尝试导入用户类文件,如果是,那么include它。

注意:

由于要频繁使用UserApps/Modules这个目录,所以我定义了一个MODULES_PATH这个常量;

01 function __autoload($className) {

02     $frameworkFileName = FRAMEWORK_PATH . '/' . $className . '.php';

03     if(is_file($frameworkFileName)) {

04         include $frameworkFileName;

05     } else {

06         //用户类文件

07         $controllerFileName = MODULES_PATH . '/Controllers/' . $className . '.php';

08         if(is_file($controllerFileName)) {

09             include $controllerFileName;

10         } else {

11             $modelFileName = MODULES_PATH . '/Models/' . $className . '.php';

12             if(is_file($modelFileName)) {

13                 include $modelFileName;

14             } else {

15                 $helperFileName = MODULES_PATH . '/Helpers/' . $className . '.php';

16                 if(is_file($helperFileName)) {

17                     include $helperFileName;

18                 } else {

19                     throw new Exception("class not found");

20                 }

21             }

22         }

23     }

24 }

           当你把这个函数写完之后,可以在之前已经写好的IndexController.php中去测试一下,比如在和IndexController同一目录下面建立一个文件Test.php,文件代码如下:

1

2 class Test {

3       public function test() {

4          echo 'Test';

5       }

6 }

然后在IndexController.php中使用如下:

1

2 class IndexController {

3     public function index() {

4         $test = new Test();

5                 $test->test();

6     }

7 }

如果出现Test,那么恭喜你,自动导入成功了!!

 

现在再考虑一下,如果针对一个很复杂的项目,使用这种方式来自动导入,有什么问题吗?

其实问题是比较严重的,首先对于框架的文件,我们不可能将所有文件都存放在一个目录下面,这样当文件多了之后就检索就麻烦了;对于用户类文件,比如控制器的文件,我们不可能将它全部存放在一个目录下面,我们需要按照模块切分目录等。如果用__autoload来实现的话,这个函数的代码量就太大了,而且如果有一个地方的修改那么就可能牵一发而动全身,这样对于项目的维护是很不利的。

那么我们怎么解决这个问题呢?

方法一:

我们定义很多辅助函数,比如导入框架文件,我们定义一个frameworkAutoloadHelper,对于用户文件定义一个userAutoloadHelper,然后将业务逻辑存放在这两个函数中,最后在__autoload中调用这两个函数即可,当用户功能需要修改的时候,代码维护性就好一些了;

方法2:

在SPL中已经定义了一个spl_autoload_register,可以使用这个来讲自动导入的功能分摊到多个类中,而且还可以将自动导入的控制权交给用户,这对于框架来说是比较重要的,所以,我推荐使用这种方式。

具体这种方式怎么实现,读者可以自己查查PHP手册。

当我们使用了spl_autoload_register之后,是不是一切就高枕无忧了?

其实不是,我们现在可以想一下是否可以去除自动导入呢,因为当业务逻辑很复杂并且自动导入设计的又不是特别好的时候,自动导入的效率就不是很高了!!

不用自动导入,难道还是使用include?No。

大家都知道include,include_once,require,require_once的区别吧!!

include和require每次调用都会包含这个文件,include_once和require_once只会导入一次,include如果包含了一个不存在的文件,只会抛出警告,程序会继续执行,而require会停止执行,所以这四个函数我比较建议使用require_once,但是很遗憾,这个函数效率很低,因为它要考虑的东西太多了!!

PS:使用这四个函数的时候,最好使用绝对路径,这样效率要高一些;

如果我们使用require,那么效率会高很多,但是如果使用require怎么实现require_once的功能呢?

我给大家提供一个思路,使用static变量:

1

2 function testRequireOnce($file) {

3       static $_config = array();

4       if(!isset($_config[$file])) {

5          require $file;

6          $_config[$file] = $file;

7       }

8 }

 

当然,还可以使用class_exists来判定。



Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn