> PHP 프레임워크 > ThinkPHP > ThinkPHP5 로딩 프로세스 분석

ThinkPHP5 로딩 프로세스 분석

Guanhui
풀어 주다: 2020-06-11 09:34:48
앞으로
2872명이 탐색했습니다.

ThinkPHP5 로딩 프로세스 분석

ThinkPHP 설치

설치 방법은 자세히 설명하지 않겠습니다. 공식 문서에서는 ThinkPHP 설치가 매우 완료되었습니다. Composer, Git을 통해 zip 패키지를 다운로드하거나 다음 사이트로 직접 이동할 수 있습니다. ThinkPHP 공식 홈페이지에서 설치했습니다. 버전은 5.0.24

Test run

다운로드 및 설치 후, 프로젝트 다운로드 디렉토리가 로컬 서버의 프로젝트 루트 디렉토리에 있다면 http 주소를 직접 입력하시면 됩니다. //localhost/thinkphp5/public/ 브라우저에서 아래 그림과 같이 ThinkPHP5의 기본 환영 페이지에 들어갈 수 있습니다. 이는 ThinkPHP5가 성공적으로 설치되었음을 의미합니다

ThinkPHP5 로딩 프로세스 분석

에서 주소를 실행하는 것 외에도 위 방법을 사용하면 Apache 또는 Nginx를 통해 가상 호스트를 구성하여 프로젝트를 구현할 수도 있습니다. 액세스를 위해 관심 있는 사용자는 온라인으로 특정 튜토리얼을 본 다음 액세스용 가상 호스트를 구성할 수 있습니다.

본론으로 들어가 ThinkPHP5의 실행 과정을 단계별로 분석해보자...

입력 파일(publicindex.php)

publicindex.php 파일을 열면 원본 코드를 볼 수 있다. 항목 파일은 다음과 같습니다

// [ 应用入口文件 ]
// 定义应用目录
define('APP_PATH', __DIR__ . '/../application/');
// 加载框架引导文件
require __DIR__ . '/../thinkphp/start.php';
로그인 후 복사

Entrance 파일 코드는 매우 간단합니다. 단 두 줄의 코드로

define('APP_PATH', __DIR__ . '/../application/');定义应用目录的常量APP_PATH
require __DIR__ . '/../thinkphp/start.php';加载框架引导文件
로그인 후 복사

함수를 사용합니다. 위의 두 함수 외에도 항목 파일에서 다음과 같은 자체 상수를 정의할 수도 있습니다. 한 줄의 코드를 추가하여 정의('PUBLIC_PATH', __DIR__ .'/../public'); 공용 디렉터리의 상수와 일부 전처리 등을 정의합니다.

프레임워크 부팅 파일 로드(thinkphpstart.php)

마찬가지로 thinkphpstart.php 파일을 입력해 보면 코드가 많지 않다는 것을 알 수 있습니다

namespace think;
// ThinkPHP 引导文件
// 1. 加载基础文件
require __DIR__ . '/base.php';
// 2. 执行应用
App::run()->send();
로그인 후 복사

이 짧은 두 줄의 코드를 보면 왼쪽과 오른쪽에 두 개의 주요 부분이 있음을 알 수 있습니다

require __DIR__ . '/base.php';加载基础文件
App::run()->send();执行应用
로그인 후 복사

다음 두 개 큰 포인트는 왼쪽과 오른쪽 두 개가 무엇을 하는지 자세히 소개하겠습니다

기본 로딩 파일(thinkphpbase.php)

thinkphpbase.php 파일을 계속 열어보니 이 파일에는 마침내 더 이상 두 줄의 코드만 남아 있지 않은 것을 발견했습니다. 앞의 두 파일처럼...

define('THINK_VERSION', '5.0.24');
define('THINK_START_TIME', microtime(true));
define('THINK_START_MEM', memory_get_usage());
define('EXT', '.php');
define('DS', DIRECTORY_SEPARATOR);
defined('THINK_PATH') or define('THINK_PATH', __DIR__ . DS);
define('LIB_PATH', THINK_PATH . 'library' . DS);
define('CORE_PATH', LIB_PATH . 'think' . DS);
define('TRAIT_PATH', LIB_PATH . 'traits' . DS);
defined('APP_PATH') or define('APP_PATH', dirname($_SERVER['SCRIPT_FILENAME']) . DS);
defined('ROOT_PATH') or define('ROOT_PATH', dirname(realpath(APP_PATH)) . DS);
defined('EXTEND_PATH') or define('EXTEND_PATH', ROOT_PATH . 'extend' . DS);
defined('VENDOR_PATH') or define('VENDOR_PATH', ROOT_PATH . 'vendor' . DS);
defined('RUNTIME_PATH') or define('RUNTIME_PATH', ROOT_PATH . 'runtime' . DS);
defined('LOG_PATH') or define('LOG_PATH', RUNTIME_PATH . 'log' . DS);
defined('CACHE_PATH') or define('CACHE_PATH', RUNTIME_PATH . 'cache' . DS);
defined('TEMP_PATH') or define('TEMP_PATH', RUNTIME_PATH . 'temp' . DS);
defined('CONF_PATH') or define('CONF_PATH', APP_PATH); // 配置文件目录
defined('CONF_EXT') or define('CONF_EXT', EXT); // 配置文件后缀
defined('ENV_PREFIX') or define('ENV_PREFIX', 'PHP_'); // 环境变量的配置前缀
// 环境常量
define('IS_CLI', PHP_SAPI == 'cli' ? true : false);
define('IS_WIN', strpos(PHP_OS, 'WIN') !== false);
// 载入Loader类
require CORE_PATH . 'Loader.php';
// 加载环境变量配置文件
if (is_file(ROOT_PATH . '.env')) {
    $env = parse_ini_file(ROOT_PATH . '.env', true);
    foreach ($env as $key => $val) {
        $name = ENV_PREFIX . strtoupper($key);
        if (is_array($val)) {
            foreach ($val as $k => $v) {
                $item = $name . '_' . strtoupper($k);
                putenv("$item=$v");
            }
        } else {
            putenv("$name=$val");
        }
    }
}
// 注册自动加载
\think\Loader::register();
// 注册错误和异常处理机制
\think\Error::register();
// 加载惯例配置文件
\think\Config::set(include THINK_PATH . 'convention' . EXT);
로그인 후 복사

코드가 60줄이나 되지만, 코드의 주요 기능은 다음과 같습니다

  1. <를 사용하세요. code>define('', '') 함수를 사용하여 많은 시스템 상수와 두 개의 환경 상수를 정의합니다define(&#39;&#39;, &#39;&#39;)函数定义了很多个系统常量,外加两个环境常量
  2. 引入loader类(thinkphplibrarythinkloader.php),供后续使用
  3. 加载环境变量配置文件(环境变量配置文件名为.env,这个文件不一定存在,都是在实际开发过程中根据需要加上去的)
  4. 调用thinkLoader::register()注册自动加载机制
    • 注册系统自动加载
    • Composer自动加载支持
    • 注册命名空间定义
    • 加载类库映射文件,存在于runtime缓存目录下classmap.php
    • 自动加载extend目录
  5. 调用thinkError::register()
  6. 이후 사용을 위해 로더 클래스(thinkphplibrarythinkloader.php)를 소개합니다
  7. 환경 변수 구성 파일(환경 변수 구성 파일 이름은 .env이며 이 파일이 반드시 존재하는 것은 아니며 실제 개발 과정에서 모두 필요에 따라 추가됩니다)

thinkLoader::register() 호출 자동 로딩 메커니즘을 등록하려면

    등록 시스템이 자동으로 로드

    Composer 자동 로딩 지원

    등록 네임스페이스 정의

    런타임에 존재하는 클래스 라이브러리 매핑 파일을 로드합니다. 캐시 디렉토리 classmap.php🎜🎜자동으로 extend 디렉토리🎜🎜🎜Call thinkError::register()</code를 로드합니다. > 예외 및 오류 처리 메커니즘을 등록하려면 🎜🎜규칙 구성 파일(thinkphpconvention.php)을 로드합니다. 🎜🎜🎜🎜응용 프로그램(thinkphplibrarythinkApp.php)에서 실행 메서드를 실행합니다. 🎜🎜🎜편의를 위해 이 코드는 run 메소드가 조금 길어서 여전히 전체 메소드를 게시하기로 선택했습니다. Don't hit me🎜<div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:php;toolbar:false">/** * 执行应用程序 * @access public * @param Request $request 请求对象 * @return Response * @throws Exception */ public static function run(Request $request = null) { $request = is_null($request) ? Request::instance() : $request; try { $config = self::initCommon(); // 模块/控制器绑定 if (defined(&amp;#39;BIND_MODULE&amp;#39;)) { BIND_MODULE &amp;&amp; Route::bind(BIND_MODULE); } elseif ($config[&amp;#39;auto_bind_module&amp;#39;]) { // 入口自动绑定 $name = pathinfo($request-&gt;baseFile(), PATHINFO_FILENAME); if ($name &amp;&amp; &amp;#39;index&amp;#39; != $name &amp;&amp; is_dir(APP_PATH . $name)) { Route::bind($name); } } $request-&gt;filter($config[&amp;#39;default_filter&amp;#39;]); // 默认语言 Lang::range($config[&amp;#39;default_lang&amp;#39;]); // 开启多语言机制 检测当前语言 $config[&amp;#39;lang_switch_on&amp;#39;] &amp;&amp; Lang::detect(); $request-&gt;langset(Lang::range()); // 加载系统语言包 Lang::load([ THINK_PATH . &amp;#39;lang&amp;#39; . DS . $request-&gt;langset() . EXT, APP_PATH . &amp;#39;lang&amp;#39; . DS . $request-&gt;langset() . EXT, ]); // 监听 app_dispatch Hook::listen(&amp;#39;app_dispatch&amp;#39;, self::$dispatch); // 获取应用调度信息 $dispatch = self::$dispatch; // 未设置调度信息则进行 URL 路由检测 if (empty($dispatch)) { $dispatch = self::routeCheck($request, $config); } // 记录当前调度信息 $request-&gt;dispatch($dispatch); // 记录路由和请求信息 if (self::$debug) { Log::record(&amp;#39;[ ROUTE ] &amp;#39; . var_export($dispatch, true), &amp;#39;info&amp;#39;); Log::record(&amp;#39;[ HEADER ] &amp;#39; . var_export($request-&gt;header(), true), &amp;#39;info&amp;#39;); Log::record(&amp;#39;[ PARAM ] &amp;#39; . var_export($request-&gt;param(), true), &amp;#39;info&amp;#39;); } // 监听 app_begin Hook::listen(&amp;#39;app_begin&amp;#39;, $dispatch); // 请求缓存检查 $request-&gt;cache( $config[&amp;#39;request_cache&amp;#39;], $config[&amp;#39;request_cache_expire&amp;#39;], $config[&amp;#39;request_cache_except&amp;#39;] ); $data = self::exec($dispatch, $config); } catch (HttpResponseException $exception) { $data = $exception-&gt;getResponse(); } // 清空类的实例化 Loader::clearInstance(); // 输出数据到客户端 if ($data instanceof Response) { $response = $data; } elseif (!is_null($data)) { // 默认自动识别响应输出类型 $type = $request-&gt;isAjax() ? Config::get(&amp;#39;default_ajax_return&amp;#39;) : Config::get(&amp;#39;default_return_type&amp;#39;); $response = Response::create($data, $type); } else { $response = Response::create(); } // 监听 app_end Hook::listen(&amp;#39;app_end&amp;#39;, $response); return $response; }</pre><div class="contentsignin">로그인 후 복사</div></div>🎜이 약 90줄의 코드는 정확히 무엇을 수행합니까? 댓글 분석과 결합하면 주로 다음과 같은 기능이 있습니다. 🎜<ul> <li>1단계: 변수 <code>$request를 처리하여 변수가 유효하고 null이 아닌지 확인합니다.$request,保证有效有用不为null
  • 第二步:self::initCommon()调用当前控制器中的initCommon()方法,负责初始化应用,并返回配置信息
    • Loader::addNamespace(self::$namespace, APP_PATH);注册命名空间
    • self::init()调用本类的init()方法初始化应用
      • 加载各种配置文件
      • 加载行为扩展文件
      • 加载公共文件
      • 加载语言包
    • 应用调试模式相关处理
    • 加载额外文件,通过配置项extra_file_list的值去加载相关文件
    • date_default_timezone_set($config['default_timezone']);设置系统时区
    • 调用Hook::listen('app_init');监听app_init标签的行为
  • 第三步:判断是否进行模块或者控制器的绑定
  • 第四步:系统语言设置和加载
  • 第五步:self::routeCheck($request, $config)加载当前控制器的routeCheck()方法进行路由检测
    • 先进行路由地址配置检测,先读取缓存路由,不存在再导入路由文件配置
    • 无路由配置,直接解析模块/控制器/操作
    • 返回module模块信息(模块名、控制器名和操作方法名)
  • 第六步:开启调试模式下,记录路由和请求信息的日志
  • 第七步:self::exec($dispatch, $config)调用控制器中的exec()方法执行调用分发
    • 根据用户请求类型进行分发处理,这里是module模块类型
    • 调用self::module()
    • 2단계: self::initCommon() 현재 컨트롤러에서 initCommon을 호출합니다( ) 메서드, 애플리케이션 초기화 및 구성 정보 반환을 담당합니다
  • Loader::addNamespace(self::$namespace, APP_PATH);네임스페이스 등록
  • self::init()이 클래스의 init() 메소드를 호출하여 애플리케이션을 초기화합니다

다양한 구성 파일 로드동작 확장 파일 로드공용 파일 로드

🎜언어 팩 로드🎜🎜🎜🎜앱 디버깅 모드 관련 처리🎜🎜추가 로드 files , 구성 항목 extra_file_list🎜🎜date_default_timezone_set($config['default_timezone'])의 값을 통해 관련 파일을 로드합니다.시스템 시간대를 설정합니다🎜🎜Hook:: listening('app_init');app_init 태그의 동작을 수신합니다🎜🎜🎜🎜3단계: 모듈을 바인딩할지 컨트롤러를 바인딩할지 결정🎜🎜4단계: 시스템 언어 설정 및 로딩🎜🎜 5단계 :self::routeCheck($request, $config)경로 감지를 위해 현재 컨트롤러의 RouteCheck() 메서드를 로드합니다🎜🎜먼저 라우팅 주소 구성 감지를 수행하고, 먼저 캐시된 경로를 읽고, 라우팅 파일 구성이 없으면 가져옵니다🎜🎜라우팅 구성이 없으며 모듈/컨트롤러/작업을 직접 구문 분석합니다🎜🎜모듈 모듈 정보(모듈 이름, 컨트롤러 이름 및 작동 방법 이름) 반환🎜🎜🎜🎜6단계: 켜기 디버깅 모드, 라우팅 기록 및 요청 정보 로그🎜🎜7단계: self::exec($dispatch, $config)컨트롤러에서 exec() 메서드를 호출하여 호출 분배를 수행🎜🎜분배 처리 사용자 요청 유형을 기반으로 합니다.🎜🎜 self::module()을 호출하여 모듈을 실행하고, 모듈을 배포 및 초기화하고, 현재 컨트롤러 이름과 작업을 가져오고 설정합니다. name🎜🎜🎜🎜8단계: 인스턴스화 클래스를 지우고 해당 형식의 데이터를 클라이언트, 즉 사용자가 보는 출력 인터페이스🎜🎜🎜추천 튜토리얼: "🎜PHP🎜" "🎜ThinkPHP Tutorial🎜" 🎜

위 내용은 ThinkPHP5 로딩 프로세스 분석의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
원천:juejin.im
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿