CodeIgniter前端大牛程序员

CodeIgniter 高级技巧

2018-03-22  本文已影响40人  lip2up

CodeIgniter 是个很传统的 PHP 框架,小巧玲珑,尽管与 Laravel 等新兴框架相比,缺乏优雅,但它简单、容易上手、易掌控

下面记录一下,我在用 CodeIgniter(以下简称 CI)过程中,摸索或查阅到的一些技巧

先做一些约定:

一、改造 Controller 方法名,加上 Http Method 前缀

该方法为本人原创,受 YII、Laravel 等框架启发

CI 支持自定义一些类,重定义其本身的行为,创建 dirApp/core/MY_Router.php,重写 set_method 方法,如下:

<?php
class MY_Router extends CI_Router
{
    public function set_method($method)
    {
        $verb = isset($_SERVER['REQUEST_METHOD'])
            ? strtolower($_SERVER['REQUEST_METHOD'])
            : '';
        $this->method = $verb . ucfirst($method);
    }
}

改造之后,对于 Http 请求:

而改造之前,这些方法就没有区分,都会执行 controllers/Vendor::list 方法。很不幸,list 是 php 的关键字,你还不能用 list 作为方法名

二、修复 CI Route 不能带 Url 参数的行为

该方法思路来自网络,后经本人优化,原文链接太久忘了,就不放了

假设我定义一个这样的 route

$route['error/(:num)'] = 'site/error/index?code=$1';

CI 竟然不支持带 Url 参数,会把整个 index?code=$1 识别为方法名,从而找不到正确的方法

为了纠正这一行为,仍然重写 dirApp/core/MY_Router.phpset_method 的方法,代码如下:

<?php
class MY_Router extends CI_Router
{
    public function set_method($method)
    {
        $parts = explode('?', $method, 2);
        if (count($parts) == 2) {
            $t = $_SERVER['QUERY_STRING'];
            $q = $parts[1] . ($t != '' ? '&' : '') . $t;
            parse_str($q, $_GET);
        }
        $this->method = $parts[0];
    }
}

代码原理很简单,就是通过 explode 将 Url 参数从 $method 分离出来,将前面的部分赋值给 $this->method,并用 parse_str 将 Url 参数,赋值到 $_GET

一、二都是重写 set_method,若要具备两者的功能,代码则应改为:

<?php
class MY_Router extends CI_Router
{
    public function set_method($method)
    {
        $parts = explode('?', $method, 2);
        if (count($parts) == 2) {
            $t = $_SERVER['QUERY_STRING'];
            $q = $parts[1] . ($t != '' ? '&' : '') . $t;
            parse_str($q, $_GET);
        }
        $verb = isset($_SERVER['REQUEST_METHOD'])
            ? strtolower($_SERVER['REQUEST_METHOD'])
            : '';
        $this->method = $verb . $parts[0];
    }
}

三、设置应用代码目录名称

CI 默认的应用代码目录名为 application,这个名字老长了,不喜,改为 web,打开入口文件 index.php,搜索 application_folder =,并改写如下:

$application_folder = 'web';

后文用 web 目录,指代 application 目录

四、给 CI 插上 Composer 的翅膀

Composer 是现代化 PHP 框架的标配,而 CI 很传统,默认没有带 Composer,也没有命名空间,这很不方便,其实加上 Composer 与命名空间也很简单

1、首先需要安装 Composer,这步不赘述,考虑到 GFW 的影响,最好按照 https://pkg.phpcomposer.com/ 上的说明,设置中国镜像

2、在根目录,创建 composer.json,写入:

{
    "autoload": {
        "psr-4": {
            "web\\": "web/"
        }
    }
}

应根据需要修改 psr-4 中的内容,我这里用的是 web 作为 application 的名称

3、然后在根目录执行 composer dump

4、在 index.php 末尾部分,最后一句 require_once BASEPATH.'core/CodeIgniter.php'; 之前添加:

require_once 'vendor/autoload.php';

5、除了 Controller 与 core/MY_*,应用程序目录 web 里的其他文件,开头应带上 namespace 声明,例如文件 web\models\User.php,开头这样声明:

<?php
namespace web\models;

引用该 Model 也很简单,请用 PHP 的机制,不要再用 CI 的 $this->load->model,如下:

<?php
use web\models\User;
// 或者
use web\models\User as UserModel;

同样,应该废弃 CI 的 helpers,在 web 目录下,创建 util 子目录,并用标准的 PHP 命名空间/面向对象机制规划工具类

实际上,引入 Composer 后,几乎可以废弃所有的 $this->load->xxx 加载方法,改用标准的 PHP 类加载机制

6、每次上线前,应执行一次 composer dump -o ,以便优化 Composer 的执行速度

五、为 Model 定义基类

为了继承 CI 的遗产,可以为 Model 定义一个基类,放在 web\core\BaseModel.php 里:

<?php
namespace web\core;

// load_class 已做了缓存,不会重复加载
load_class('Model', 'core');

class BaseModel extends \CI_Model
{
    use \web\util\CI;
    use \web\util\Db;
    use \web\util\Instance;
    ...
}

web\util\CI 是一个很有用的 trait,用来在 model 里方便地获取 CI 实例,这样定义的:

<?php
namespace web\util;

/**
 * 提供获取 CI 实例的方法
 */
trait CI
{
    protected static function ci()
    {
        $ci = &get_instance();
        return $ci;
    }
}

web\util\Db 用来方便地操纵数据库,定义如下:

<?php
namespace web\util;

/**
 * 提供一组访问数据库的方法
 */
trait Db
{
    protected static function db($name = 'default')
    {
        if (!isset(self::$_ciDbCache[$name])) {
            $ci = &get_instance();
            log_message('debug', "load db $name");
            self::$_ciDbCache[$name] = $ci->load->database($name, true);
        }
        return self::$_ciDbCache[$name];
    }
    private static $_ciDbCache = [];

    protected static function executeSql($sql, $data = null)
    {
        return self::db()->query($sql, $data);
    }
}

web\util\Instance 用来提供单例模式,定义如下:

<?php
namespace web\util;

/**
 * 实现单例模式
 */
trait Instance
{
    final public static function instance()
    {
        $className = get_called_class();
        if (!isset(self::$_instanceList[$className])) {
            self::$_instanceList[$className] = new static;
        }
        return self::$_instanceList[$className];
    }
    private static $_instanceList = [];
}

六、为 Controller 定义基类

Controller 也需要有一个共同的基类,以便定义一些公用的行为,大致代码如下(省略了一些本司商业逻辑,以免泄密 😆,只讨论技术):

<?php
namespace web\core;

...

class BaseController extends \CI_Controller
{

    /**
     * 加载 Twig View
     * @data    传递给 View 的数据
     * @view    可选的 View 路径,一般不需传,会自动获取
     */
    protected function view($data = null, $view = null)
    {
        View::render($data, $view);
    }

    /**
     * 输出 { s: , data: } 格式的 json 数据
     * @s       数字状态码,默认 200,表示成功
     * @data    附加数据,应传递关联数组,不宜使用数值数组
     * @jsonp   可选,设置该参数,则返回 JSONP 数据;
     *          支持 true 或字符串,传 true 时,将自动提取
     *          callback 参数(jquery 的默认方式)作为 callback
     */
    protected function json($s = 200, $data = null, $jsonp = false)
    {
        ...
    }

    /**
     * 带可选子域名的重定向
     * @uri             重定向的目的地址
     * @keepReferrer    是否保留 document.referrer
     * @domain          可选子域名,例如传递 www,则重定向到 www.yourdomain.com/$uri
     */
    protected function redirect($uri, $keepReferrer = false, $domain = null)
    {
        $url = $domain != null
            ? '//' . preg_replace('/^[^\.]*\./', $domain . '.', $_SERVER['HTTP_HOST']) . $uri
            : $uri;
        $keepReferrer
            ? die("<script>location = '$url'</script>")
            : header("Location:$url");
    }

    /**
     * 跳转到错误页 xxx
     */
    protected function goError($code) {
        $this->redirect('/error' . $code, true);
    }

    /**
     * 跳转到 404 页
     */
    protected function go404()
    {
        $this->goError(404);
    }

    /**
     * 跳转首页
     */
    protected function goHome()
    {
        $this->redirect('xxx', true);
    }

    /**
     * 跳转到登录页
     */
    protected function goLogin()
    {
        $loginUrl = 'xxx';
        $this->redirect($loginUrl, true);
    }

    /**
     * 是否为 ajax 请求
     */
    protected function isAjax()
    {
        return 'XMLHttpRequest' == @$_SERVER['HTTP_X_REQUESTED_WITH'];
    }

    /**
     * 是否为 post 请求
     */
    protected function isPost()
    {
        return 'POST' == @$_SERVER['REQUEST_METHOD'];
    }

    /**
     * 当前登录的用户 ID
     */
    protected $uid  = 0;

    /**
     * 当前登录的用户实例
     */
    protected $user = null;

    /**
     * 需要登录
     */
    protected function requireLogin()
    {
        ...
    }
}

再定义一个 web\core\RequireLoginController 类,所有需要登录的页面,应继承自该类:

<?php
namespace web\core;

/**
 * 所有需要登录的页面,继承自该类
 */
class RequireLoginController extends BaseController
{
    public function __construct()
    {
        parent::__construct();
        $this->requireLogin();
    }
}

本来,requireLogin 方法,是放在 RequireLoginController 类的,但后来,考虑到一些继承自 BaseController 的某个页面,也可能需要登录

七、引入优秀 ORM 库 Eloquent

该方法是一名喜欢 Laravel 的前同事总结的

Eloquent 是非常优秀的 ORM 库,将它引入 CI 也非常简单,github 也有现成的库,但这里手写代码,也不复杂

以下示例,使用 mysql 数据库,其他请根据实际情况进行修改

1、根目录执行 composer require illuminate/database

2、创建 web/libraries/Eloquent.php,写入代码:

<?php
use Illuminate\Database\Capsule\Manager as Capsule;

$runtimeDb = APPPATH . 'config/' . ENVIRONMENT . '/database.php';
$defaultDb = APPPATH . 'config/database.php';

if (is_file($runtimeDb)) {
    require_once $runtimeDb;
} else {
    if (is_file($defaultDb)) {
        require_once $defaultDb;
    } else {
        exit('No database config file be found');
    }
}

$capsule = new Capsule;

$ciToEloquentKeyMap = [
    'hostname' => 'host',
    'username' => 'username',
    'password' => 'password',
    'database' => 'database',
    'dbdriver' => 'driver',
    'dbprefix' => 'prefix',
    'char_set' => 'charset',
    'dbcollat' => 'collation',
    'stricton' => 'strict',
];

foreach ($db as $k => $v) {
    $t = [];
    if (!isset($v['char_set']) or $v['char_set'] != 'utf8') {
        $v['char_set'] = 'utf8';
    }
    foreach ($v as $mm => $nn) {
        if (isset($ciToEloquentKeyMap[$mm])) {
            $t[$ciToEloquentKeyMap[$mm]] = $nn;
        } else {
            $t[$mm] = $nn;
        }
    }
    $t['driver'] = 'mysql';
    $capsule->addConnection($t, $k);
}

$capsule->bootEloquent();

3、打开 index.php,在 require_once 'vendor/autoload.php';require_once BASEPATH.'core/CodeIgniter.php'; 之间,插入:

require_once 'web/libraries/Eloquent.php';

4、定义一个基类 web\core\EloquentModel

<?php
namespace web\core;

/**
 * Eloquent Model 基类
 */
class EloquentModel extends \Illuminate\Database\Eloquent\Model
{
    use \web\util\CI;
    use \web\util\Instance;

    protected $guarded = ['id'];

    // 如果表中没有 created_at updated_at 字段,子类需要加
    // public      $timestamps = false;
}

更多用法,请参考 Eloquent 官方文档

整篇完。欢迎转载,转载请注明出处:
简书作者:lip2up
微信公众号:前端大牛

上一篇 下一篇

猜你喜欢

热点阅读