ThinkPHP5 로딩 프로세스 분석
ThinkPHP 설치
설치 방법은 자세히 설명하지 않겠습니다. 공식 문서에서는 ThinkPHP 설치가 매우 완료되었습니다. Composer, Git을 통해 zip 패키지를 다운로드하거나 다음 사이트로 직접 이동할 수 있습니다. ThinkPHP 공식 홈페이지에서 설치했습니다. 버전은 5.0.24
Test run
다운로드 및 설치 후, 프로젝트 다운로드 디렉토리가 로컬 서버의 프로젝트 루트 디렉토리에 있다면 http 주소를 직접 입력하시면 됩니다. //localhost/thinkphp5/public/ 브라우저에서 아래 그림과 같이 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줄이나 되지만, 코드의 주요 기능은 다음과 같습니다
- <를 사용하세요. code>define('', '') 함수를 사용하여 많은 시스템 상수와 두 개의 환경 상수를 정의합니다
define('', '')
函数定义了很多个系统常量,外加两个环境常量 - 引入loader类(thinkphplibrarythinkloader.php),供后续使用
- 加载环境变量配置文件(环境变量配置文件名为
.env
,这个文件不一定存在,都是在实际开发过程中根据需要加上去的) - 调用
thinkLoader::register()
注册自动加载机制- 注册系统自动加载
Composer
自动加载支持- 注册命名空间定义
- 加载类库映射文件,存在于
runtime
缓存目录下classmap.php
- 自动加载
extend
目录
- 调用
thinkError::register()
이후 사용을 위해 로더 클래스(thinkphplibrarythinkloader.php)를 소개합니다환경 변수 구성 파일(환경 변수 구성 파일 이름은
.env
이며 이 파일이 반드시 존재하는 것은 아니며 실제 개발 과정에서 모두 필요에 따라 추가됩니다)thinkLoader::register()
호출 자동 로딩 메커니즘을 등록하려면
- 등록 시스템이 자동으로 로드
- 第二步:
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을 호출합니다( ) 메서드, 애플리케이션 초기화 및 구성 정보 반환을 담당합니다 -
self::init()이 클래스의 init() 메소드를 호출하여 애플리케이션을 초기화합니다
Composer
자동 로딩 지원등록 네임스페이스 정의런타임에 존재하는 클래스 라이브러리 매핑 파일을 로드합니다.
캐시 디렉토리 classmap.php
🎜🎜자동으로 extend
디렉토리🎜 ul>🎜🎜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(&#39;BIND_MODULE&#39;)) {
BIND_MODULE && Route::bind(BIND_MODULE);
} elseif ($config[&#39;auto_bind_module&#39;]) {
// 入口自动绑定
$name = pathinfo($request->baseFile(), PATHINFO_FILENAME);
if ($name && &#39;index&#39; != $name && is_dir(APP_PATH . $name)) {
Route::bind($name);
}
}
$request->filter($config[&#39;default_filter&#39;]);
// 默认语言
Lang::range($config[&#39;default_lang&#39;]);
// 开启多语言机制 检测当前语言
$config[&#39;lang_switch_on&#39;] && Lang::detect();
$request->langset(Lang::range());
// 加载系统语言包
Lang::load([
THINK_PATH . &#39;lang&#39; . DS . $request->langset() . EXT,
APP_PATH . &#39;lang&#39; . DS . $request->langset() . EXT,
]);
// 监听 app_dispatch
Hook::listen(&#39;app_dispatch&#39;, self::$dispatch);
// 获取应用调度信息
$dispatch = self::$dispatch;
// 未设置调度信息则进行 URL 路由检测
if (empty($dispatch)) {
$dispatch = self::routeCheck($request, $config);
}
// 记录当前调度信息
$request->dispatch($dispatch);
// 记录路由和请求信息
if (self::$debug) {
Log::record(&#39;[ ROUTE ] &#39; . var_export($dispatch, true), &#39;info&#39;);
Log::record(&#39;[ HEADER ] &#39; . var_export($request->header(), true), &#39;info&#39;);
Log::record(&#39;[ PARAM ] &#39; . var_export($request->param(), true), &#39;info&#39;);
}
// 监听 app_begin
Hook::listen(&#39;app_begin&#39;, $dispatch);
// 请求缓存检查
$request->cache(
$config[&#39;request_cache&#39;],
$config[&#39;request_cache_expire&#39;],
$config[&#39;request_cache_except&#39;]
);
$data = self::exec($dispatch, $config);
} catch (HttpResponseException $exception) {
$data = $exception->getResponse();
}
// 清空类的实例化
Loader::clearInstance();
// 输出数据到客户端
if ($data instanceof Response) {
$response = $data;
} elseif (!is_null($data)) {
// 默认自动识别响应输出类型
$type = $request->isAjax() ?
Config::get(&#39;default_ajax_return&#39;) :
Config::get(&#39;default_return_type&#39;);
$response = Response::create($data, $type);
} else {
$response = Response::create();
}
// 监听 app_end
Hook::listen(&#39;app_end&#39;, $response);
return $response;
}</pre><div class="contentsignin">로그인 후 복사</div></div>🎜이 약 90줄의 코드는 정확히 무엇을 수행합니까? 댓글 분석과 결합하면 주로 다음과 같은 기능이 있습니다. 🎜<ul>
<li>1단계: 변수 <code>$request
를 처리하여 변수가 유효하고 null이 아닌지 확인합니다.$request
,保证有效有用不为null
Loader::addNamespace(self::$namespace, APP_PATH);
네임스페이스 등록다양한 구성 파일 로드동작 확장 파일 로드공용 파일 로드
🎜언어 팩 로드🎜🎜🎜🎜앱 디버깅 모드 관련 처리🎜🎜추가 로드 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 중국어 웹사이트의 기타 관련 기사를 참조하세요!

핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

AI Hentai Generator
AI Hentai를 무료로 생성하십시오.

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

뜨거운 주제











PHP 8.4는 상당한 양의 기능 중단 및 제거를 통해 몇 가지 새로운 기능, 보안 개선 및 성능 개선을 제공합니다. 이 가이드에서는 Ubuntu, Debian 또는 해당 파생 제품에서 PHP 8.4를 설치하거나 PHP 8.4로 업그레이드하는 방법을 설명합니다.

VS Code라고도 알려진 Visual Studio Code는 모든 주요 운영 체제에서 사용할 수 있는 무료 소스 코드 편집기 또는 통합 개발 환경(IDE)입니다. 다양한 프로그래밍 언어에 대한 대규모 확장 모음을 통해 VS Code는

이 튜토리얼은 PHP를 사용하여 XML 문서를 효율적으로 처리하는 방법을 보여줍니다. XML (Extensible Markup Language)은 인간의 가독성과 기계 구문 분석을 위해 설계된 다목적 텍스트 기반 마크 업 언어입니다. 일반적으로 데이터 저장 AN에 사용됩니다

JWT는 주로 신분증 인증 및 정보 교환을 위해 당사자간에 정보를 안전하게 전송하는 데 사용되는 JSON을 기반으로 한 개방형 표준입니다. 1. JWT는 헤더, 페이로드 및 서명의 세 부분으로 구성됩니다. 2. JWT의 작업 원칙에는 세 가지 단계가 포함됩니다. JWT 생성, JWT 확인 및 Parsing Payload. 3. PHP에서 인증에 JWT를 사용하면 JWT를 생성하고 확인할 수 있으며 사용자 역할 및 권한 정보가 고급 사용에 포함될 수 있습니다. 4. 일반적인 오류에는 서명 검증 실패, 토큰 만료 및 대형 페이로드가 포함됩니다. 디버깅 기술에는 디버깅 도구 및 로깅 사용이 포함됩니다. 5. 성능 최적화 및 모범 사례에는 적절한 시그니처 알고리즘 사용, 타당성 기간 설정 합리적,

문자열은 문자, 숫자 및 기호를 포함하여 일련의 문자입니다. 이 튜토리얼은 다른 방법을 사용하여 PHP의 주어진 문자열의 모음 수를 계산하는 방법을 배웁니다. 영어의 모음은 A, E, I, O, U이며 대문자 또는 소문자 일 수 있습니다. 모음이란 무엇입니까? 모음은 특정 발음을 나타내는 알파벳 문자입니다. 대문자와 소문자를 포함하여 영어에는 5 개의 모음이 있습니다. a, e, i, o, u 예 1 입력 : String = "Tutorialspoint" 출력 : 6 설명하다 문자열의 "Tutorialspoint"의 모음은 u, o, i, a, o, i입니다. 총 6 개의 위안이 있습니다

숙련된 PHP 개발자라면 이미 그런 일을 해왔다는 느낌을 받을 것입니다. 귀하는 상당한 수의 애플리케이션을 개발하고, 수백만 줄의 코드를 디버깅하고, 여러 스크립트를 수정하여 작업을 수행했습니다.

정적 바인딩 (정적 : :)는 PHP에서 늦은 정적 바인딩 (LSB)을 구현하여 클래스를 정의하는 대신 정적 컨텍스트에서 호출 클래스를 참조 할 수 있습니다. 1) 구문 분석 프로세스는 런타임에 수행됩니다. 2) 상속 관계에서 통화 클래스를 찾아보십시오. 3) 성능 오버 헤드를 가져올 수 있습니다.

PHP의 마법 방법은 무엇입니까? PHP의 마법 방법은 다음과 같습니다. 1. \ _ \ _ Construct, 객체를 초기화하는 데 사용됩니다. 2. \ _ \ _ 파괴, 자원을 정리하는 데 사용됩니다. 3. \ _ \ _ 호출, 존재하지 않는 메소드 호출을 처리하십시오. 4. \ _ \ _ get, 동적 속성 액세스를 구현하십시오. 5. \ _ \ _ Set, 동적 속성 설정을 구현하십시오. 이러한 방법은 특정 상황에서 자동으로 호출되어 코드 유연성과 효율성을 향상시킵니다.
