您现在的位置是:首页 > cms教程 > Discuz教程Discuz教程
discuz代码解析初始化应用的过程
夏彤2025-07-15Discuz教程已有人查阅
导读流程结构(随便找个地址来分析下,比如 设置 -> 个人资料 -> 联系方式;地址是:home.php?mod=spacecp&ac=profile&op=contact)(一)首先打开home.php文件 看到第17行
流程结构(随便找个地址来分析下,比如 设置 -> 个人资料 -> 联系方式;地址是:home.php?mod=spacecp&ac=profile&op=contact)
(一)首先打开home.php文件 看到第17行
require_once './source/class/class_core.php';
我同时打开了多个入口的文件都有这么一个引入,因此可以肯定这是个入口配置文件,负责文件引入及初始化需要的组件,下面来看下这个文件
第10行 error_reporting(E_ALL); 报告所有错误
第12-15行四个常量定义
1
C::creatapp();
这个C是core类的映射,证据是第208行
1
class C extends core {}
来看静态方法creatapp()
来看刚才执行的静态方法 discuz_application::instance() 的返回值是个啥?
public function __construct() {
$this->_init_env(); //274-316行;初始化环境变量。定义了些常量,引入了公共函数库,在./source/function/function_core.php,设置小内存128M。并设置了允许的全局变量(_GET,_POST,_REQUEST,_COOKIE,_SERVER,_ENV,_FILES),定义了一个全局变量$_G,变将这个全局变量按址传递给了$this->var,看来在模板中可以直接使用这些变量的。
$this->_init_config(); //274-316行;初始化配置变量。引入了配置,在./config/config_global.php,将配置变量全部压入到$this->var['config']中,在全局都可以使用配置啦
$this->_init_input();//224-272行;初始化输入。过滤了下全局变量($_GET['GLOBALS'],$_POST['GLOBALS']……),如果开启了get_magic_quotes_gpc(),就给$_GET,$_POST,$_COOKIE变量stripslashes一下,设置合法的cookie的键名必须 $this->config['cookie']['cookiepre']这个开头的,在配置文件中有。这样做应该是防止非法cookie变量,243行把$_POST的值合并到$_GET变量中去。把rawurlencode($_GET['page'])编码下,过滤了下$_GET['handlekey'],过滤$_GET变量,foreach($_GET as $k => $v) {$this->var['gp_'.$k] = daddslashes($v);}使用的是stripslashes函数。另外设置了几个全局变量 $this->var['XXX']自己瞅瞅。
$this->_init_output();//318-345行;初始化输出。过滤URL$this->_xss_check();<"CONTENT-TRANSFER-ENCODING都是非法的。如果控制器为'seccode', 'secqaa', 'swfupload'则进行这个过滤,有时间看看./source/include/misc/misc_security.php;开启ob_start(),设置charset;
}
好了,现在我们明白了,我们现在得到了一个对象C,并且它有一个属性叫_app,这个属性是discuz_application对象,再在回到入口配置文件class_core.php,下面执行到了第209行,我们看到一个映射
class DB extends discuz_database {}
DB代表了discuz_database对象,我们去看看这个对象。从上面的自动引入类机制中我们得到了这个类位于 ./source/class/discdz/discdz_application.php。打开看看,原来是数据库操作的;静态方法 init($driver, $config)应该是初始化这个类的方法,从字面意思看,传两个参数进来,第一个参数是数据库驱动,第二个参数是数据库连接参数,以后的数据库操作应该都是DB::update() DB::delete这样的方法了。
好了,跟着程序运行顺序走,我们现在返回到入口文件home.php第18行
require_once './source/function/function_home.php';
进去看看,原来是一组函数,这是个函数库,瞄了两眼,现在回到home.php第20行
$discuz = C::app();
原来是把discuz_application对象赋值给变量$discuz。
到第22-24行
核心代码第二句
462行;!empty($this->cachelist) && loadcache($this->cachelist);装载缓存, 使用缓存可以参照我另一篇文章中关于缓存的应用
477行;$this->var['member']['lastvisit'] = TIMESTAMP - 3600;dsetcookie('lastvisit', TIMESTAMP - 3600, 86400 * 30);设置之后浏览的时间
464-490行,设置了cookie、全局变量、增加var属性变量
核心代码第四句 $this->_init_session();discuz_application.php 386-426行
设置session类,更新用户状态
大概看了下,都是设置手机访问的一些变量,全局变量,cookie,session,跳转,常量等,有空细看下
核心代码第六句$this->_init_cron();discuz_application.php 508-515行
计划任务初始化 。配置文件中 $_config['remote']['cron'] = 1;设置后;远程调用: 开启外部 cron 任务. 系统内部不再执行cron, cron任务由外部程序激活
(一)首先打开home.php文件 看到第17行
require_once './source/class/class_core.php';
我同时打开了多个入口的文件都有这么一个引入,因此可以肯定这是个入口配置文件,负责文件引入及初始化需要的组件,下面来看下这个文件
第10行 error_reporting(E_ALL); 报告所有错误
第12-15行四个常量定义
define('IN_DISCUZ', true);
define('DISCUZ_ROOT', substr(dirname(__FILE__), 0, -12));
define('DISCUZ_CORE_DEBUG', false);
define('DISCUZ_TABLE_EXTENDABLE', TRUE);
第17-22行设置异常处理函数
set_exception_handler(array('core', 'handleException'));
if(DISCUZ_CORE_DEBUG) {
set_error_handler(array('core', 'handleError'));
register_shutdown_function(array('core', 'handleShutdown'));
}
第24-30行设置类自动引入的处理函数(这里看仔细,很重要)
if(function_exists('spl_autoload_register')) {
spl_autoload_register(array('core', 'autoload'));
} else {
function __autoload($class) {
return core::autoload($class);
}
}
第32行 初始化core类1
C::creatapp();
这个C是core类的映射,证据是第208行
1
class C extends core {}
来看静态方法creatapp()
public static function creatapp() {
if(!is_object(self::$_app)) {
self::$_app = discuz_application::instance();
}
return self::$_app;
}
我们看到是返回了一个属性,这个属性_app 是discuz_application::instance()的返回值,我们来看discuz_application类,从上面的自动引入中我们看到这个类的地址是 ./source/class/discdz/discdz_application.php,来看刚才执行的静态方法 discuz_application::instance() 的返回值是个啥?
static function &instance() {
static $object;
if(empty($object)) {
$object = new self();
}
return $object;
}
原来是按址引用的,并且是实理化自身了,这个类继承自抽象类discuz_base,这个应该是个基类,来看看构造函数。public function __construct() {
$this->_init_env(); //274-316行;初始化环境变量。定义了些常量,引入了公共函数库,在./source/function/function_core.php,设置小内存128M。并设置了允许的全局变量(_GET,_POST,_REQUEST,_COOKIE,_SERVER,_ENV,_FILES),定义了一个全局变量$_G,变将这个全局变量按址传递给了$this->var,看来在模板中可以直接使用这些变量的。
$this->_init_config(); //274-316行;初始化配置变量。引入了配置,在./config/config_global.php,将配置变量全部压入到$this->var['config']中,在全局都可以使用配置啦
$this->_init_input();//224-272行;初始化输入。过滤了下全局变量($_GET['GLOBALS'],$_POST['GLOBALS']……),如果开启了get_magic_quotes_gpc(),就给$_GET,$_POST,$_COOKIE变量stripslashes一下,设置合法的cookie的键名必须 $this->config['cookie']['cookiepre']这个开头的,在配置文件中有。这样做应该是防止非法cookie变量,243行把$_POST的值合并到$_GET变量中去。把rawurlencode($_GET['page'])编码下,过滤了下$_GET['handlekey'],过滤$_GET变量,foreach($_GET as $k => $v) {$this->var['gp_'.$k] = daddslashes($v);}使用的是stripslashes函数。另外设置了几个全局变量 $this->var['XXX']自己瞅瞅。
$this->_init_output();//318-345行;初始化输出。过滤URL$this->_xss_check();<"CONTENT-TRANSFER-ENCODING都是非法的。如果控制器为'seccode', 'secqaa', 'swfupload'则进行这个过滤,有时间看看./source/include/misc/misc_security.php;开启ob_start(),设置charset;
}
好了,现在我们明白了,我们现在得到了一个对象C,并且它有一个属性叫_app,这个属性是discuz_application对象,再在回到入口配置文件class_core.php,下面执行到了第209行,我们看到一个映射
class DB extends discuz_database {}
DB代表了discuz_database对象,我们去看看这个对象。从上面的自动引入类机制中我们得到了这个类位于 ./source/class/discdz/discdz_application.php。打开看看,原来是数据库操作的;静态方法 init($driver, $config)应该是初始化这个类的方法,从字面意思看,传两个参数进来,第一个参数是数据库驱动,第二个参数是数据库连接参数,以后的数据库操作应该都是DB::update() DB::delete这样的方法了。
好了,跟着程序运行顺序走,我们现在返回到入口文件home.php第18行
require_once './source/function/function_home.php';
进去看看,原来是一组函数,这是个函数库,瞄了两眼,现在回到home.php第20行
$discuz = C::app();
原来是把discuz_application对象赋值给变量$discuz。
到第22-24行
$cachelist = array('magic','userapp','usergroups', 'diytemplatenamehome');
$discuz->cachelist = $cachelist;
$discuz->init();
哇靠,$discuz->init();这一行才是核心中的核心,具体功能是初始化整个discuz应用。discuz_application类是整个discuz的应用初始化类,相当于织梦的 /include/common.inc.php的功能
public function init() {
if(!$this->initated) {
$this->_init_db();//初始化数据库
$this->_init_setting();//系统设置初始化
$this->_init_user();//用户信息初始化
$this->_init_session();//session操作初始化
$this->_init_mobile();//手机功能初始化
$this->_init_cron();//计划任务初始化
$this->_init_misc();//其他功能初始化
}
$this->initated = true;//设置完成标志
}
核心代码第一句
$this->_init_db();
private function _init_db() {
if($this->init_db) {
$driver = 'db_driver_mysql';
if(getglobal('config/db/slave')) {//在设置里修改slave可以修改数据库驱动类,默认为db_dirver_mysql
$driver = 'db_driver_mysql_slave';
}
DB::init($driver, $this->config['db']);//DB类位于:./source/class/discuz/discuz_database.php
}
}
DB::init方法是在discuz_database类中实例化了数据库驱动类db_driver_mysql驱动类,并将其赋值给DB::db属性,根据自动引入规则驱动类的位置在:./source/class/db/db_driver_mysql.php;核心代码第二句
$this->_init_setting(),
private function _init_setting() {
if($this->init_setting) {
if(empty($this->var['setting'])) {
$this->cachelist[] = 'setting';
}
if(empty($this->var['style'])) {
$this->cachelist[] = 'style_default';
}
if(!isset($this->var['cache']['cronnextrun'])) {
$this->cachelist[] = 'cronnextrun';
}
}
!empty($this->cachelist) && loadcache($this->cachelist);//主要是这句,加载缓存,实现的过程可以看
function_core.php中的第687-716行,实际是把缓存赋值给$_G变量($_G['setting'],$_G['cache'],$_G['grouplevels']),处理缓存的其实是在第700行 C::t('common_syscache')->fetch_all($caches);,这块是重点。可以研究一下,DZ的缓存部分,这里我略讲一下,用C::t('common_syscache')->fetch_all($caches) 实例化了一个对象common_syscache 位置:./source/class/table/table_common_syscache.php,挺绕得,我看了半天代码才找出来,其实这个完整的对象是discuz_container。common_syscache对象只是discuz_container的一个属性obj;在class_core.php的第76-90行可以看出这种关系来,table_common_syscache.php第38行,$data = memory('get', $cachenames);这个是之后得到的缓存数据,我们来追下memory函数,第一个参数是get,那么会执行function_core.php中的第1687行 case 'get': return C::memory()->get($key, $value); 其中C::memory()是实例化了discuz_memory对象,在./source/class/discuz/discuz_memory.php ;看get方法在第70-102行,它有两个参数,get($key, $prefix = '')。get方法第97行$ret = $this->memory->get($this->_key($key));使用了memory_driver_eaccelerator类的get方法,位置./source/class/memory/memory_driver_eaccelerator.php第22行,return eaccelerator_get($key);,缓存就是依靠eaccelerator_get函数获取的
if(!is_array($this->var['setting'])) {
$this->var['setting'] = array();
}
}
核心代码第三句
$this->_init_user();discuz_application.php 428-493行
$auth = getglobal('auth', 'cookie')
得到cookie中的auth的值,加密过的值
$auth = daddslashes(explode("\t", authcode($auth, 'DECODE'))); 利用authcode函数解密,得到一个数组
list($discuz_pw, $discuz_uid) = empty($auth) || count($auth) < 2 ? array('', '') : $auth; 从这句看数组第一个元素为密码,第二个元素是会员登陆的ID。存在common_member表中。
$user = getuserbyuid($discuz_uid, 1);利用getuserbyuid得到会员信息,第二个参数如果是1并且在common_member中没有该会员就去common_member_archive表中去寻找,否则直接去common_member表中查找
if(!empty($user) && $user['password'] == $discuz_pw) {如果查出的密码是对的
if(isset($user['_inarchive'])) {
C::t('common_member_archive')->move_to_master($discuz_uid);如果数组中有_inarchive元素就将当前查到的用户信息插入到用户表common_member表中
}
$this->var['member'] = $user;将用户信息压入模板变量中
} else {
$user = array();
$this->_init_guest();//设置没登陆的访客的信息
}
if($user && $user['groupexpiry'] > 0 && $user['groupexpiry'] < TIMESTAMP && (getgpc('mod') != 'spacecp'|| CURSCRIPT != 'home')) {
dheader('location: home.php?mod=spacecp&ac=usergroup&do=expiry');//如果登陆时间过期了,或者访问的是home.php下的内容时跳转到登陆页面
}
这里作用是检查登陆了没有。462行;!empty($this->cachelist) && loadcache($this->cachelist);装载缓存, 使用缓存可以参照我另一篇文章中关于缓存的应用
477行;$this->var['member']['lastvisit'] = TIMESTAMP - 3600;dsetcookie('lastvisit', TIMESTAMP - 3600, 86400 * 30);设置之后浏览的时间
464-490行,设置了cookie、全局变量、增加var属性变量
核心代码第四句 $this->_init_session();discuz_application.php 386-426行
设置session类,更新用户状态
<?php
$sessionclose = !empty($this->var['setting']['sessionclose']);//设置中sessionclose是否为真
$this->session = $sessionclose ? new discuz_session_close() : newdiscuz_session();//如果为真就实例化discuz_session_close类 应该是关闭session后的解决方案,否则实例化discuz_session类,位置在source/class/discuz/discuz_session.php
if($this->session->get('groupid') == 6)判断session中的groupid合法性
$this->session->set('lastactivity', TIMESTAMP);设置之后访问时间为当前时间
dsetcookie('lip', $this->var['member']['lastip'].','.$this->var['member']['lastvisit']);第一次访问的话设置访问IP的session
C::t('common_member_status')->update($this->var['uid'], array('lastip' => $this->var['clientip'], 'lastvisit' => TIMESTAMP));//更新用户的状态
?>
核心代码第五句 $this->_init_mobile();discuz_application.php 680-789行大概看了下,都是设置手机访问的一些变量,全局变量,cookie,session,跳转,常量等,有空细看下
核心代码第六句$this->_init_cron();discuz_application.php 508-515行
计划任务初始化 。配置文件中 $_config['remote']['cron'] = 1;设置后;远程调用: 开启外部 cron 任务. 系统内部不再执行cron, cron任务由外部程序激活
$ext= empty($this->config['remote']['on']) || empty($this->config['remote']['cron']) || APPTYPEID == 200;
if($this->init_cron && $this->init_setting && $ext) {//如果远程调用: 开启外部 cron 任务后调用discuz_cron::run();类
if($this->var['cache']['cronnextrun'] <= TIMESTAMP) {
discuz_cron::run();
}
}
至此我们看到了一个完整的初始化过程
require_once'./source/class/class_core.php'; //引入核心类文件,作用为:自动引入类规则,错误和异常处理,单例创建discuz_application类实例,引入默认函数库function.core.php
require_once './source/function/function_home.php';//项目函数库
$discuz = C::app();//实例化desiuz_application类
$cachelist = array('magic','userapp','usergroups', 'diytemplatenamehome');
$discuz->cachelist = $cachelist;//设置缓存列表
$discuz->init();//初始化整个应用
本文标签:
很赞哦! ()
上一篇:discuz nt 1.0分析
下一篇:discuz中常用函数
相关教程
图文教程
phpsso与Discuz无法同步登陆的解决方法
经过测试仍然有问题,phpcms登陆,论坛可以实现同步登陆,但论坛登陆phpcms无法同步登陆。但总体来说也能使用了。如果有更好的方法欢迎补充。
Discuz!NT各个模块功能说明
Discuz中有24个Project,每个项目各自作为项目架构中的不同的功能。下面我们来各自的分析一下这些项目。(1)DNT中包含的项目(1)Discuz.Web该项目是DNT最终生成网站的项目。
Discuz中SESSION机制讲解
在Discuz!X中一如继往的,SESSION并没有使用PHP自带的SESSION机制,而是系统的一套自带的机制。在数据库中可以看到有两个SESSION表:
discuz session机制介绍
这是discuz session机制的详细页面,介绍了和php,php, session机制 discuz session机制有关的知识、技巧、经验,和一些php源码等。
相关源码
-
(自适应)五金配件机械加工设备pbootcms模板免费下载这款基于PbootCMS开发的网站模板为五金配件和机械加工行业设计,采用简洁有力的设计风格,突出工业产品的专业性和可靠性。模板结构清晰,功能完善,能够有效展示各类工业产品的技术参数和应用场景。查看源码 -
(自适应)宽屏大气红色机械设备pbootcms模板源码下载为机械设备制造企业设计的响应式网站模板,采用PbootCMS内核开发。宽屏布局突出设备展示效果,红色工业风格贴合机械行业属性,支持PC端与手机端自动适配查看源码 -
(自适应响应式)投资理财金融机构财务管理pbootcms模板本模板基于PbootCMS系统开发,为投资理财、金融机构等行业设计。采用专业严谨的布局风格,突出金融服务行业特色,适合展示各类理财产品、投资服务和金融资讯。查看源码 -
帝国CMS7.5漫画网站模板带手机端源码免费下载本模板为漫画内容平台设计开发,采用帝国CMS7.5内核构建,深度优化漫画作品展示结构与章节管理模式。前端采用响应式布局设计,适配各类漫画阅读场景,提供作品分类、连载追踪、热度排行等垂直领域功能模块。查看源码 -
(自适应响应式)家电维修清晰服务网站pbootcms模板免费下载本模板基于PbootCMS内核开发,为维修服务类企业打造,特别适合家电维修、设备维护等行业使用。通过简洁直观的界面设计,帮助企业快速搭建专业级服务平台,实现线上业务高效管理。查看源码 -
帝国cms7.5游戏门户H5小游戏网站模板源码免费下载基于帝国CMS7.5开发的H5游戏门户模板,为在线游戏平台设计。支持PC端与移动端同步生成,打造完整的游戏娱乐体验平台。查看源码
| 分享笔记 (共有 篇笔记) |
