1、框架目录
1.1 创建目录结构
1.2 文件分类存放
将上一讲的文件分类存放到不同的目录中
将文件存放到不同的目录以后,由于类文件地址发生了变化,所以无法完成自动加载类,那么今天的主要任务就是围绕如何实现类的自动加载展开
由于每次都请求入口文件,所以"."表示入口文件所在的目录
2、添加命名空间
通过文件目录地址做命名空间,这样获取了命名空间就能知道文件存放的地址
Model.class.php
namespace Core;
class Model {
...
MyPDO.class.php
namespace Core;
class MyPDO{
...
ProductsModel.class.php
<?php
namespace Model;
class ProductsModel extends Model{
...
ProductsController.class.php
<?php
namespace Controller\Admin;
class ProductsController {
...
3、框架类实现
3.1 定义路径常量
由于文件路径使用频率很高,而且路径比较长,所以将固定不变的路径定义为路径常量
1、getcwd():入口文件的绝对路径
2、window下默认的目录分隔符是'\'('\'和'/'都支持)
linux下默认的目录分隔符是'/'
DIRECTORY_SEPARATOR常量根据不同的操作系统返回不同的目录分隔符
代码实现
在core文件夹下创建Framework.class.php
private static function initConst(){
define('DS', DIRECTORY_SEPARATOR); //定义目录分隔符
define('ROOT_PATH', getcwd().DS); //入口文件所在的目录
define('APP_PATH', ROOT_PATH.'Application'.DS); //application目录
define('CONFIG_PATH', APP_PATH.'Config'.DS);
define('CONTROLLER_PATH', APP_PATH.'Controller'.DS);
define('MODEL_PATH', APP_PATH.'Model'.DS);
define('VIEW_PATH', APP_PATH.'View'.DS);
define('FRAMEWORK_PATH', ROOT_PATH.'Framework'.DS);
define('CORE_PATH', FRAMEWORK_PATH.'Core'.DS);
define('LIB_PATH', FRAMEWORK_PATH.'Lib'.DS);
define('TRAITS_PATH', ROOT_PATH.'Traits'.DS);
}
3.2 引入配置文件
1、在config目录下创建config.php
<?php
return array(
//数据库配置
'database'=>array(),
//应用程序配置
'app' =>array(
'dp' => 'Admin', //默认平台
'dc' => 'Products', //默认控制器
'da' => 'list' //默认方法
),
);
2、在框架类中引入配置文件
private static function initConfig(){
$GLOBALS['config']=require CONFIG_PATH.'config.php';
}
思考:配置文件为什么不保存在常量中?
答:因为7.0之前,常量不能保存数组和对象。
3.3 确定路由
p:【platform】平台
c:【controller】控制器
a:【action】方法
private static function initRoutes(){
$p=$_GET['p']??$GLOBALS['config']['app']['dp'];
$c=$_GET['c']??$GLOBALS['config']['app']['dc'];
$a=$_GET['a']??$GLOBALS['config']['app']['da'];
$p=ucfirst(strtolower($p));
$c=ucfirst(strtolower($c)); //首字母大写
$a=strtolower($a); //转成小写
define('PLATFROM_NAME', $p); //平台名常量
define('CONTROLLER_NAME', $c); //控制器名常量
define('ACTION_NAME', $a); //方法名常量
define('__URL__', CONTROLLER_PATH.$p.DS); //当前请求控制器的目录地址
define('__VIEW__',VIEW_PATH.$p.DS); //当前视图的目录地址
}
3.4 自动加载类
private static function initAutoLoad(){
spl_autoload_register(function($class_name){
$namespace= dirname($class_name); //命名空间
$class_name= basename($class_name); //类名
if(in_array($namespace, array('Core','Lib'))) //命名空间在Core和Lib下
$path= FRAMEWORK_PATH.$namespace.DS.$class_name.'.class.php';
elseif($namespace=='Model') //文件在Model下
$path=MODEL_PATH.$class_name.'.class.php';
elseif($namespace=='Traits') //文件在Traits下
$path=TRAITS_PATH.$class_name.'.class.php';
else //控制器
$path=CONTROLLER_PATH.PLATFROM_NAME.DS.$class_name.'.class.php';
if(file_exists($path) && is_file($path))
require $path;
});
}
3.5 请求分发
private static function initDispatch(){
$controller_name='\Controller\\'.PLATFROM_NAME.'\\'.CONTROLLER_NAME.'Controller'; //拼接控制器类名
$action_name=ACTION_NAME.'Action'; //拼接方法名
$obj=new $controller_name();
$obj->$action_name();
}
3.6 封装run()方法
class Framework{
//启动框架
public static function run(){
self::initConst();
self::initConfig();
self::initRoutes();
self::initAutoLoad();
self::initDispatch();
}
...
3.7 在入口中调用run()方法
<?php
require './Framework/Core/Framework.class.php';
Framework::run();
4、 运行项目
1、连接数据库的参数从配置文件中获取
class Model {
...
//连接数据库
private function initMyPDO() {
$this->mypdo= MyPDO::getInstance($GLOBALS['config']['database']);
}
}
2、更改ProductsControleller控制器
<?php
namespace Controller\Admin;
//商品模块
class ProductsController {
//获取商品列表
public function listAction() {
//实例化模型
$model=new \Model\ProductsModel();
$list=$model->getList();
//加载视图
require __VIEW__.'products_list.html';
}
//删除商品
public function delAction() {
...
$model=new \Model\ProductsModel();
...
}
}
3、更改ProductsModel类
<?php
namespace Model;
class ProductsModel extends \Core\Model{
}
4、更改MyPDO类
...
private function fetchType($type){
switch ($type){
case 'num':
return \PDO::FETCH_NUM;
case 'both':
return \PDO::FETCH_BOTH;
case 'obj':
return \PDO::FETCH_OBJ;
default:
return \PDO::FETCH_ASSOC;
}
}
...
//所有的内置类都在公共的命名空间下。
测试:成功