制作自己的MVC框架(二)——启动
阅读原文时间:2020年11月02日阅读:6

自己写了个框架,取名为“PrimusPHP”,就3个文件夹,log中是日志文件,会自动生成。

app中就是控制器和视图,public中是入口文件和静态资源,library中既有框架的核心类,还有工具类,数据操作类等。

这个框架会用到命名空间,PHP版本如果不能用命名空间,将不能执行。

一、单一入口

“index.php”就是入口文件,“defined.php”设置一些通用常量,例如绝对目录,并且引入了初始化库。

define('ROOT', dirname(dirname(__FILE__)));
define('DS', DIRECTORY_SEPARATOR);

require_once LIB_PATH . DS . 'init.php';

“index.php”就是在执行初始化方法。

require_once '../defined.php';
define('APP_NAME', 'weixin');
define('CONFIG_PATH', APP_PATH . DS . APP_NAME . DS . 'conf');

InitPrimus::getInstance();

二、初始化

private function __construct() {
$this->environment();//环境配置
$this->reporting();//错误报告
self::loadFunction(self::$config['autoload']['helper']);//加载辅助函数
$this->_autoload();//自动载入
self::openLog(); //开启log
self::loadClass('\library\core\Router');//路由
}

1)环境配置

通常会有三个配置环境,本地、测试和正式。

以前曾经把三个环境的参数写在一个文件中,然后版本控制的时候忽略文件,这样做很不方便。

尤其是在发布版本的时候,如果要变里面的参数,还得记得在哪里改。

现在直接分三个文件,各自管各自的,谁也影响不到谁。

“library”中有三个配置文件,在“app”中也放在“conf”方便自行控制。

 

如何判断当前环境是本地、测试还是正式环境呢?

这里我将这个变量埋在了“.htaccess”中,如果是nginx,也可以配置。

RewriteEngine on
SetEnv ENVIRONMENT local

2)错误报告

错误提示在调试的时候很有用,能定位到具体的行数,能提高修正效率。

在网上搜索了一下,找到了一些代码,再借鉴了CI的错误代码,嫁接到这个框架里。

set_error_handler('exception_error_handler');//追踪错误栈
register_shutdown_function('catch_fatal_error');//致命错误

“exception_error_handler”和“catch_fatal_error”是两个函数,写在了“core/Common.php”中。

在“library/view”中预留了三种错误的模版页面。

在URL中输入“index/sign”错误的样子如下:

3)加载辅助函数

框架中,经常会有一些函数或类,在一开始就被载入进来,这里的这个辅助函数就是这个道理。

在配置文件中预先写好要载入的文件名。

public static function loadFunction($name) {
if(is_array($name)) {
foreach ($name as $key) {
include_once HELPER_LIB_PATH . DS . $key . '_helper.php';
}
}else {
include_once HELPER_LIB_PATH . DS . $name . '_helper.php';
}
}

4)自动载入

当“new \library\xx()”某个类的时候,会自动进入到“spl_autoload_register”函数中。

在回调函数中将“\library\xx”修改为具体路径,在“include_once”进来,就初始化好了。

private function _autoload() {
//$className是通过命名空间获取的
spl_autoload_register(function ($className) {
$dir = false;
$classNames = explode('\\', $className);//分割为数组
$autoDirs = self::$autoDirs;
foreach ($autoDirs as $path) {
$tmpInfo = pathinfo($path);
if(in_array($tmpInfo['basename'], $classNames)) {
$dir = $tmpInfo['dirname'];
break;
}
}

        if($dir) {  
            $file = $dir . DS . dir\_replace\_ds($className) . '.php';  
            include\_once($file);  
        }  
    });  

}

5)日志

平时开发的时候,我喜欢将查询语句直接打印到日志中,当将代码放到线上了,也能调试各个语句的结果。

最近在做微信开发,有很多请求微信的服务器,这些请求的结果只能通过打印日志看到。

日志的类放在了“util/Log.php”中。

public function write($txt) {
if(empty($txt) || !DEVELOPMENT_ENVIRONMENT) {
return;
}
$name = date('Y-m-d') . '.log';
if(!file_exists(LOG_PATH)) {
mkdir(LOG_PATH, 0777, true);
}
$path = LOG_PATH . DS . $name;
$content = date('Y-m-d H:i:s') . ' ' . $txt . "\n";
file_put_contents($path, $content, FILE_APPEND);
}

三、路由

1)路由选择

路由支持两种,一种就是“index/index”,另外一种是“v1/index/index”。

第二种可以用于客户端接口,当客户端更新版本的时候,可能就需要新的接口,而老的接口又不能丢,就可以新开一个模块文件夹。

$params = explode('/', $uri);
if(count($params) == 2) {
$controller = $params[0];//控制器
$action = $params[1];//操作方法
}elseif(count($params) == 3) {
$module = $params[0]; //模块
$controller = $params[1];//控制器
$action = $params[2];//操作方法
}else {
$controller = 'index';
$action = 'index';
}

2)路由重定向

以前没注意到重定向,最近在做微信开发的配置文件的时候就发现这个功能。

访问“bridge.js”其实是访问了某个控制器的“action”,怪不得我在想那些微信token都是通过服务器获取的,这个js文件是怎么搞到的。

重定向规则可以用正则表达式。

$init_conf['rewrite'] = array(
'/new_bridge.js/' => 'index/config',
'/new_bridge2.js/' => 'index/config2',
'/p\/(\d+)/' => 'index/ticket?p=$1'
);

下面是正则匹配:

public function rewrite() {
$config = \InitPrimus::getConfig('rewrite');
foreach ($config as $src => $redirect) {
//$pattern = '@^\/?' . $src . '\/?$@';
if (\preg_match($src, $this->uri)) {
$this->uri = \preg_replace($src, $redirect, $this->uri);
break;
}
}
}

框架地址:

https://github.com/pwstrick/PrimusPHP

手机扫一扫

移动阅读更方便

阿里云服务器
腾讯云服务器
七牛云服务器

你可能感兴趣的文章