在一些公司里,将各个类放在单独的文件中是很常见的做法。
再来看图书馆的例子,假设管理应用程序需要表示图书、员工、事件和顾客的类。
根据这个项目的任务,可能要创建一个名为classes的目录来放置如下文件:Books.class.php
、Employees.class.php
、Events.class.php
和Patrons.class.php
。
虽然这确实有助于类的管理,不过它要求各个文件的脚本都能找到相应文件,这一般通过require_once()
语句来实现。
因此,如果脚本需要这4个类,就要在前面插入如下语句:
require_once ("classes/Books.class.php") ;
require_once ("classes/Employees.class.php" ) ;
require_once ("classes/Events.class.php") ;
require_once ("classes/Patrons.class.php") ;
以这种方式管理类包含(class inclusion)会变得非常麻烦,并且会在本来已经很复杂的开发过程中再增加一个额外的步骤。
为消除这个额外的任务,PHP5中引入了自动加载对象的概念。
自动加载通过定义特殊的__autoload
函数,当引用未在脚本中定义的类时会自动调用这个函数。
回到图书馆的例子,通过定义如下函数,就不必再手工包含各个类文件了:
function __autoload ($class) {
require_once ("classes/$class.class.php");
}
定义这个函数后,将不再需要那些require_once()
语句,因为当第一次调用一个类时,就会调用__autoload()
,并根据__autoload()
中定义的命令加载类。
这个函数可以放在某个全局应用程序配置文件中,它只是说该函数要在脚本中可用。
为什么要使用__autoload
在 PHP 中使用类时,我们必须在使用前加载进来,不管是通过 require 的方式还是 include 的方式,但是会有两个问题影响我们做出加载的决定。
首先是不知道这个类文件存放在什么地方,另外一个就是不知道什么时候需要用到这个文件。
特别是项目文件特别多时,不可能每个文件都在开始的部分写很长一串的 require …
在PHP5之后,我们可以通过 __autoload
来解决这个问题。
而且在PHP5.1之后,还提供了 spl_autoload_register()
来提供更完善的加载机制。
Autoload 的加载机制大概是这样的,当通过 new 来实例化一个类时,PHP会通过定义的__autoload
函数加载相应的文件,如果这个类文件使用了 extends 或者 implements 需要用到其他的类文件,php会重新运行 autoload 去进行类文件的查找和加载,如果发生了两次对同一类文件的请求,就会报错。
一般情况下,有很多种方法来解决加载时到相应位置查找文件的方法。
用的最多的就是指定特定的命名标准。
zend推荐了一种最流行的办法,在文件名中包含路径。例如下面的例子:
// Main.class
function __autoload($class_name) {
$path = str_replace('_', DIRECTORY_SEPARATOR, $class_name);
require_once $path.'.php';
}
$temp = new Main_Super_Class();
所有的下划线都会被替换成路径中的分隔符,上例中就会去 Main/Super/Class.php
文件。
这种方法的缺点是在编码过程中,我们必须明确的知道代码文件应当所处的位置,而且由于将文件路径硬编码在了类名中,如果需要修改文件夹的结构时,我们必须手工修改所有的类名。
如果是在一个开发环境中,并且对于速度不是很在意的话,使用’Include All’这个方法是非常方便的。
通过将所有类文件放在一个或几个特定文件夹中,然后通过遍历的方式查找加载。例如:
<?php
$arr = array (
'Project/Classes',
'Project/Classes/Children',
'Project/Interfaces'
);
foreach($arr as $dir) {
$dir_list = opendir($dir);
while ($file = readdir($dir_list)) {
$path = $dir.DIRECTORY_SEPARATOR.$file;
if(in_array($file, array('.', '..')) || is_dir($path))
continue;
if (strpos($file, ".class.php"))
require_once $path;
}
}
?>
另外一个方法是在类文件和他的位置之间建立关联的配置文件,例如:
// configuration.php
array_of_associations = array(
'MainSuperClass' = 'C:/Main/Super/Class.php',
'MainPoorClass' = 'C:/blablabla/gy.php'
);
调用的文件
<?php
require 'autoload_generated.php';
function __autoload($className) {
global $autoload_list;
require_once $autoload_list[$className];
}
$x = new A();
?>
当然,如果文件特别多的时候,维护起来会是一件麻烦事,但是与在类名中硬编码位置,哪个更好呢?
猜你喜欢: