上一次我讲到使用自动导入可以实现一个类的延迟加载,这一次我想讲一下配置文件。
不知道大家注意了没有,前几篇文章我都没有涉及到配置文件,可是在一个项目中,配置文件又是比不可少的。
现在假设将Route.php中的默认控制器和Action变为可配置的,怎么弄呢?
我们使用最简单的方式:
<?php $defaultController = 'Index'; $defaultAction = 'index';
然后在Route.php中include这个文件:
<?php include APP_PATH . '/config.php';
$controller = empty($_GET['c']) ? $defaultController : trim($_GET['c']); //设置了默认的控制器 $action = empty($_GET['a']) ? $defaultAction : trim($_GET['a']); //设置了默认的Action当然也可以使用这种方式:
<?php return array( 'defaultController' => 'Index', 'defaultAction' => 'index' );还是在Route.php中include:
<?php $config = include APP_PATH . '/config.php';
$controller = empty($_GET['c']) ? $config['defaultController'] : trim($_GET['c']); //设置了默认的控制器 $action = empty($_GET['a']) ? $config['defaultAction'] : trim($_GET['a']); //设置了默认的Action
但是由于配置文件很有可能在整个框架运行过程的各个类都有用到,而前两种方式都只能在某一个类里面的某一个方法中使用,怎么让它在各个类都可以使用呢?
我们可以可以使用一个static方法来解决,比如专门定义一个Config类来存放配置信息,Config::get来读取配置信息,Config::set来设置配置信息,具体做法读者可以自己去尝试一下。
当然我不准备采用这种方式,可能当时我看Thinkphp源码的时候被它优雅的特殊函数C迷住了吧,后面我在写Toper的时候也使用了一些特殊的函数,而C主要是负责设置和读取配置信息,个人观点,面向对象是必要的,但是不是什么都要面向对象,所以我准备定义一个function.php来存放一些公有的函数,我把它称为公有函数库。
由于有了公有函数库,我们可以将上一篇里面的自动导入(__autoload)也转移到这个文件中,因为在Route.php出现一段__autoload的代码还是挺诡异的!!
好了,怎么实现刚才我们说的通过一个函数来实现配置信息的设置和读取呢?
好了,先展示我在Toper中的实现吧:
function C($name = null,$val = null) { static $_config = array(); if(empty($name)) { return $_config; } elseif(is_string($name)) { if(empty($val)) { if(!strpos($name,'=>')) { //一维 return isset($_config[$name]) ? $_config[$name] : null; } else { //目前只支持二维 $name = explode('=>',$name); return isset($_config[$name[0]][$name[1]]) ? $_config[$name[0]][$name[1]] : null; } } else { if(!strpos($name,'=>')) { //直接设置 $_config[$name] = $val; } else { //设置二维 $name = explode('=>',$name); $_config[$name[0]][$name[1]] = $val; } } } elseif(is_array($name)) { foreach($name as $key=>$value) { $_config[$key] = $value; } return ; } else { throw new Exception('参数类型出错'); return ; } }
看着代码挺长的,实际上原理很简单,如果传递的参数只有一个,那么第二个参数就调用默认参数,即NULL,再识别是否第一个参数是否是字符串,那么这个函数就识别为读取,如果第二个参数不为空或第一个参数为数组,那么就识别为设置!!
由于我自己现在比较懒,而且我用这个函数用的函数蛮顺手的,所以在这儿我就直接用这个函数来作为例子了,更多内容可以查看Toper的/Library/Toper/function.php。
比如现在要读取defaultController,那么只要使用C('defaultController')即可,如果要设置,那么使用C('defaultController','Index')!!!
现在我们只需要在入口文件中导入这个function.php即可:
<?php defined('APP_PATH') || define('APP_PATH',dirname(__FILE__) . '/..'); defined('FRAMEWORK_PATH') || define('FRAMEWORK_PATH',APP_PATH . '/Library/Test'); defined('MODULES_PATH') || define('MODULES_PATH',APP_PATH . '/UserApps/Modules'); include FRAMEWORK_PATH . '/function.php'; include FRAMEWORK_PATH . '/Route.php'; Route::run();大家可能注意到了,C函数最开始的时候,里面没有存放任何元素,那么我们怎么样进行初始化,将配置文件的内容写入C函数呢?
之前我们将配置文件存放在项目根目录,这样实际上是不符合之前我们的约定的规范,所以现在讲这个配置文件剪切到/UserApps/Configs目录下面,为了更方便的使用这个路径,我们定义一个CONFIGS_PATH来指向配置文件的路径。
现在我们看看入口文件变成了什么样了:
<?php defined('APP_PATH') || define('APP_PATH',dirname(__FILE__) . '/..'); defined('FRAMEWORK_PATH') || define('FRAMEWORK_PATH',APP_PATH . '/Library/Test'); defined('MODULES_PATH') || define('MODULES_PATH',APP_PATH . '/UserApps/Modules'); defined('CONFIGS_PATH') || define('CONFIGS_PATH',APP_PATH . '/UserApps/Configs'); include FRAMEWORK_PATH . '/function.php'; C(include CONFIGS_PATH . '/config.php'); //写入配置信息 include FRAMEWORK_PATH . '/Route.php'; Route::run();然后我们修改一下Route.php
<?php class Route { public static function run() { $controller = empty($_GET['c']) ? C('defaultController') : trim($_GET['c']); //设置了默认的控制器 $action = empty($_GET['a']) ? C('defaultAction') : trim($_GET['a']); //设置了默认的Action $controllerBasePath = APP_PATH . '/UserApps/Modules/Controllers/'; $controllerFilePath = $controllerBasePath . $controller . 'Controller.php'; if(is_file($controllerFilePath)) { include $controllerFilePath; $controllerName = $controller . 'Controller'; if(class_exists($controllerName)) { $controllerHandler = new $controllerName(); if(method_exists($controllerHandler,$action)) { $controllerHandler->$action(); } else { echo 'the method does not exists'; } } else { echo 'the class does not exists'; } } else { echo 'controller not exists'; } } }今天的例子 点此下载