CodeIgniter 高级技巧

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

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

先做一些约定:

  • dirApp 表示 application 目录

一、改造 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 请求:

  • get /vendor/list,将会执行 controllers/Vendor::getList 方法
  • post /vendor/list,将会执行 controllers/Vendor::postList 方法
  • put /vendor/list,将会执行 controllers/Vendor::putList 方法
  • 以此类推

而改造之前,这些方法就没有区分,都会执行 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
微信公众号:前端大牛

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容