CodeIgniter 常量应定义在 application/config/constants.php 中,使用 define() 函数,全局可用且不依赖加载顺序;动态值推荐用配置项而非钩子 define();URL 相关常量需明确定义,禁用磁盘路径常量拼接 URL。

CodeIgniter 的常量定义本身很轻量,但「方便」与否取决于你用在哪儿、怎么加载、什么时候访问——尤其是跨环境或动态读库时,容易掉坑里。
直接写 config/constants.php 是最稳的静态方式
这是 CI 官方唯一原生支持的常量定义入口,所有控制器、模型、视图都能无条件访问,不依赖加载顺序、不依赖实例化。
- 路径固定:
application/config/constants.php - 写法就是标准 PHP
define(),比如:defined('SITE_CREATOR') OR define('SITE_CREATOR', 'Alice');
defined('MAX_LOGIN_ATTEMPTS') OR define('MAX_LOGIN_ATTEMPTS', 5); - 在任意地方直接用:
SITE_CREATOR,不用$this->,也不用get_instance() - ⚠️ 注意:不能在
constants.php里调用 CI 类(如$this->db),因为此时框架还没启动
想从数据库动态定义常量?别在钩子里硬塞 define()
很多人试过在 post_controller_constructor 钩子中查库再 define(),结果在控制器构造函数里还是 NULL——因为常量必须在 PHP 编译期或首次执行前定义,而钩子触发时已晚。
-
pre_system或pre_controller钩子才够早,但此时$CI =& get_instance()还不可用(CI 实例未创建) - 真要动态加载,推荐改用配置项:
$this->config->load('dynamic_settings', TRUE);,然后把数据库值写进
echo $this->config->item('site_access_enabled');application/config/dynamic_settings.php文件(可定时生成) - 或者用
autoload.php加载一个自定义初始化类,在其中手动define(),但得确保它比所有控制器更早执行(需配合pre_system钩子 + 手动 require)
FCPATH、APPPATH 等系统常量不能在 View 里直接拼 URL
这些是磁盘路径常量(如 FCPATH = /var/www/html/),不是 URL,直接 echo 到 HTML 里会暴露服务器结构,还可能被 XSS 利用。
- View 中需要 URL 时,优先用
base_url()或site_url()辅助函数(先$this->load->helper('url')) - 如果非要自定义全局 URL 常量,应在
config/constants.php里明确定义:defined('ASSET_URL') OR define('ASSET_URL', 'https://cdn.example.com/assets/');,而不是拼接FCPATH - ⚠️ 特别注意:CI4 已废弃这些常量,CI3 项目升级前务必检查所有
BASEPATH/APPPATH直接拼接逻辑
真正麻烦的从来不是“怎么定义”,而是“定义后谁先看到、谁看不到、谁看到的是旧值”——尤其当你混用钩子、自动加载、缓存和多环境配置时,一个没对齐,REGISTER_ENABLED 就在生产环境悄悄变成 false 而你还在本地测得飞起。










