解读 thinkphp5 源码(二):异常处理和请求生命周期

   2016-09-29 0
核心提示:上节 解读 thinkphp5 源码(一):自动加载 看完了自动加载部分,根据代码执行顺序,base.php的59-64行 // 注册错误和异常处理机制\think\Error::register();// 加载惯例配置文件\think\Config::set(include THINK_PATH . 'convention' . EXT);下面的加载配置

上节 解读 thinkphp5 源码(一):自动加载 看完了自动加载部分,根据代码执行顺序,base.php的59-64行

// 注册错误和异常处理机制
\think\Error::register();

// 加载惯例配置文件
\think\Config::set(include THINK_PATH . 'convention' . EXT);

下面的加载配置文件不用说,现在重点看一下异常处理。

打开 library/think/Error.php ,register函数。

public static function register(){
        error_reporting(E_ALL);
        set_error_handler([__CLASS__, 'appError']);
        set_exception_handler([__CLASS__, 'appException']);
        register_shutdown_function([__CLASS__, 'appShutdown']);
    }

通过 error_reporting() 来这是php的报错级别。E_ALL为显示所有报错信息,所以在运行时NOTICE级别的警告也会显示,如果不想显示NOTICE信息,可以在项目common文件中重新设置一下报错级别: error_reporting(E_ALL ^ E_NOTICE);

set_error_handler指定appError来处理系统异常

set_exception_handler来指定appException来处理用户抛出的异常

register_shutdown_function来指定appShutdown处理超时异常

然后使用getExceptionHandler方法来获取异常处理对象的实例。

public static function getExceptionHandler()
    {
        static $handle;
        if (!$handle) {
            // 异常处理handle
            $class = Config::get('exception_handle');
            if ($class && class_exists($class) && is_subclass_of($class, "\\think\\exception\\Handle")) {
                $handle = new $class;
            } else {
                $handle = new Handle;
            }
        }
        return $handle;
    }

里面有一句 $class = Config::get('exception_handle'); 也就是说我们可以通过修改配置参数来指定新的异常处理对象。

library/think/exception/ 下是几个异常处理类,主要就是在开启debug的情况下输出异常等级,异常信息,异常代码等等。然后也对CLI模式下做了异常输出的处理。

请求生命周期

应用初始化

base.php看完之后,基本上框架的初始化工作也做完了。然后回到,start.php中。

// 执行应用
App::run()->send();

这里是一个连贯操作,我们需要先看一下App::run()返回的是什么对象

library/think/App.php,run方法

run方法可以接收一个Request对象,如果不存在,则获取该对象。

is_null($request) && $request = Request::instance();

然后通过 $config = self::initCommon(); 来初始化应用

看initCommon()中的代码可以发现主要做了以下几件事情。

首先通过init()方法来初始化应用,主要就是加载该模块或者应用运行所需的文件,配置、common文件、语言包

然后判断如果配置中关闭debug的话则通过 ini_set('display_errors', 'Off'); 来关闭异常信息的显示。

ini_set(‘display_errors’, ‘Off’)和error_reporting(0)的区别主要在于,用前者设置后页面不显示异常信息,但是可以通过日志来收集异常信息,主要用在生产环境。而error_reporting(0);则屏蔽了所有异常信息,这种做法肯定是不太科学的。

然后判断如果不是cli模式下则重新申请一个缓冲区。使用缓冲区的作用主要是两方面,一是可以在输出一些内容之后在设置header(例如cookie等),使得程序设计的逻辑性变得简单;二是可以对缓冲区里面的输出内容撤销、删除、压缩、保存到文件等操作。

这里通过ob_get_level来判断缓冲区是否为空,来处理缓冲区的历史内容。

然后注册命名空间、加载额外文件、设置时区、注册app_init的hook。

回到run方法

if (defined('BIND_MODULE')) {
            // 模块/控制器绑定
            BIND_MODULE && Route::bind(BIND_MODULE);
        } elseif ($config['auto_bind_module']) {
            // 入口自动绑定
            $name = pathinfo($request->baseFile(), PATHINFO_FILENAME);
            if ($name && 'index' != $name && is_dir(APP_PATH . $name)) {
                Route::bind($name);
            }
        }

BIND_MODULE主要是来处理默认模块的,tp5的路由规则是 /m/c/a ,如果应用只有一个模块,则可以用过设置BIND_MODULE常量来设置模块名称。

通过配置auto_bind_module参数也可以设定模块名称。

$request->filter($config['default_filter']); 注入过滤方法

开启多语言支持。

路由检测

// 获取应用调度信息
            $dispatch = self::$dispatch;
            if (empty($dispatch)) {
                // 进行URL路由检测
                $dispatch = self::routeCheck($request, $config);
            }
            // 记录当前调度信息
            $request->dispatch($dispatch);
            // 记录路由和请求信息
            if (self::$debug) {
                Log::record('[ ROUTE ] ' . var_export($dispatch, true), 'info');
                Log::record('[ HEADER ] ' . var_export($request->header(), true), 'info');
                Log::record('[ PARAM ] ' . var_export($request->param(), true), 'info');
            }

主要通过 self::routeCheck($request, $config); 来对路由进行分类,然后进行相应的操作(等后面进详细阅读)。主要有以下几种类别

路由到模块/控制器/操作;
路由到外部重定向地址;
路由到控制器方法;
路由到闭包函数;
路由到类的方法;

路由分发

根据上面得到分类,进行相应的操作,得到返回的数据(等后面进详细阅读)。

解读 thinkphp5 源码(二):异常处理和请求生命周期

请求响应

主要通过Response类来进行响应。

解读 thinkphp5 源码(二):异常处理和请求生命周期

注册app_end的hook,并返回response对象。

所以前面App::run()事实上得到的是一个经过一系列操作之后的response对象然后执行response->send();

解读 thinkphp5 源码(二):异常处理和请求生命周期

拿到缓冲区中的数据,并输出,至此,结束请求。

 
标签: ThinkPHP
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • nginx 各类网站设置 (laravel , thinkphp , nod
    基础部分设置[root@centos ~]# vim /opt/nginx/conf/nginx.confuser www www;worker_processes auto;pid logs/nginx.pid;worker_rlimit_nofile 100000;events {use epoll;multi_accept on;worker_connections 65535 ;}http {include mime.types;default_type
    02-09
  • thinkphp5 IIS7.5 隐藏index.php的方法
    ?xml version="1.0" encoding="UTF-8"? configurationsystem.webServerrewriterulesrule name="WPurls" enabled="true" stopProcessing="true"match url=".*" /conditions logicalGrouping="MatchAll"
    02-09
  • ThinkPHP 利用.htaccess文件的 Rewrite 规则隐
    通常的URL里面含有index.php,为了达到更好的SEO效果可能需要去掉URL里面的index.php ,通过URL重写的方式可以达到这种效果,通常需要服务器开启URL_REWRITE模块才能支持。 下面是Apache的配置过程,可以参考下:1、httpd.conf配置文件中加载了mod_rewrite.so
    02-09
  • 【小程序+thinkphp5】 用户登陆,返回第三方session3rd
    【小程序+thinkphp5】 用户登陆,返回第三方ses
    服务器环境: centos7   php7.0准备工作: 注册小程序,并获取 appid 、appsecret下载微信解密算法sdk : https://mp.weixin.qq.com/debug/wxadoc/dev/api/signature.htmlhttps 的域名。需要在小程序后台服务器域名那配置   代码实现 一 、 thinkphp5 
    02-09
  • 码云推荐 | 基于 ThinkPHP5 的通用后台 lyadmin_v2
    码云推荐 | 基于 ThinkPHP5 的通用后台 lyadmin
    lyadmin_v2 项目介绍 lyamdin是一套轻量级通用后台,采用Bootstrap3制作,自带权限管理,模块化开发。lyadmin_v2是基于ThinkPHP5全新开发的新版本。真正原创的Builder页面生生成器完美支持2.0并且支持多达38中表单类型的支持,包括但不限于文本、富文本、图片
  • 在 thinkphp 中使用 laravel 的全部组件
    作为一个使用 php 作为主力语言的公司,不管怎么说,在一些老的项目中,总会碰到使用 thinkphp 的。那么,热爱 laravel 的你,当你去开发新的 feature 时,大胆的去引入 laravel 的组件吧。不管是对于现在的开发效率,还是日后项目的重构,迁移都是有很大帮助
  • ThinkPHP V5.0.5 版本发布 — 祝大家新春快乐!
    ThinkPHP V5.0.5 版本发布 — 祝大家新春快乐!
    ThinkPHP V5.0.5 版本发布,祝大家新春快乐,事业有成,“鸡”祥如意^_^官方快速入门系列教程 新增了控制器从入门到精通 V5.0.5版本是年前最后一个发布版本,也是5.0系列版本的最后一个功能版本,主要改进和优化了数据库和模型部分,并修正了之前版本的一些问
  • 基于thinkphp的音乐网站源码
    基于thinkphp的音乐网站源码
    KSmusic音乐网站源码是一个以php+MySQL进行开发的免费音乐网站源码,只需进行简单的设置,即可拥有属于自己的音乐网站。 更新日志:标签增加 分页功能 大大增加数据自由性专辑 count 标签 用于读取数据总数量新增上一个,下一个标签 ,新增 网盘下载设置项新
  • 在线工单售后客户服务管理系统(thinkphp内核)
    在线工单售后客户服务管理系统(thinkphp内核)
    THINKPHP内核的工单系统,非常适合于售后客户服务,二开后对接到网站系统是非常不错的哦,可以添加客户和售后人员,支持短信和邮件提醒,只需要在后台配置好就行了,还是非常简单的。1、系统管理1.1:客户管理:可实现对职员的管理,可添加职员,并对不同角色
    01-06 ThinkPHP
  • thinkphp 模板解析的流程
    thinkphp 模板解析的流程
    ThinkPHP控制器中在每一个方法的最后都有这样的一句话 ,你确定对这句话非常了解吗?$this-display();当然你会说这句话是用来解析或显示控制器的方法所对应的模板的。有了这句,模板里的内容才可以正常显示,没有这句话就会抛出错误:模板不存在。但是这句话是
    12-23 ThinkPHP
点击排行