ThinkPHP 설치
설치 방법에 대해서는 자세히 설명하지 않겠습니다. 공식 문서인 ThinkPHP 설치는 매우 포괄적입니다. Composer, Git을 사용하거나 ThinkPHP 공식 홈페이지에서 직접 zip 패키지를 다운로드할 수 있습니다. 제가 설치한 버전은 5.0.24
Test run
다운로드 및 설치 후 프로젝트 다운로드 디렉토리가 프로젝트 루트 디렉토리에 있다면 로컬 서버인 경우 브라우저 http://localhost/thinkphp5/public/
에 주소를 직접 입력할 수 있습니다. 아래 그림과 같이 ThinkPHP5의 기본 시작 페이지에 입력할 수 있습니다. ThinkPHP5가 성공적으로 설치되었습니다http://localhost/thinkphp5/public/
,就可以进入到ThinkPHP5的默认欢迎页,如下图所示,这就说明ThinkPHP5已经安装成功
除了上面的这个方式的地址运行,我们也可以通过Apache或者Nginx配置虚拟主机实现项目的访问,有兴趣的可以网上查看具体教程,然后配置虚拟主机进行访问。
下面进入正题,我们来逐步分析ThinkPHP5的执行流程……
入口文件(publicindex.php)
打开publicindex.php
文件后,我们可以看到,入口文件原始代码如下
// [ 应用入口文件 ] // 定义应用目录 define('APP_PATH', __DIR__ . '/../application/'); // 加载框架引导文件 require __DIR__ . '/../thinkphp/start.php';
入口文件代码很简洁,就两行代码,作用分别为
-
define('APP_PATH', __DIR__ . '/../application/');
定义应用目录的常量APP_PATH -
require __DIR__ . '/../thinkphp/start.php';
加载框架引导文件
除了上面的这两个作用外,我们还可以额外在入口文件中,定义我们自己的常量,例如添加一行代码define('PUBLIC_PATH', __DIR__ .'/../public');
定义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);
仔细一看,发现代码虽然有六十多行,但是,代码的作用却显而易见,作用主要有以下六点
- 使用
define('', '')
函数定义了很多个系统常量,外加两个环境常量 - 引入loader类(thinkphplibrarythinkloader.php),供后续使用
- 加载环境变量配置文件(环境变量配置文件名为
.env
,这个文件不一定存在,都是在实际开发过程中根据需要加上去的) -
调用
thinkLoader::register()
注册自动加载机制- 注册系统自动加载
-
Composer
自动加载支持 - 注册命名空间定义
- 加载类库映射文件,存在于
runtime
缓存目录下classmap.php
- 自动加载
extend
目录
- 调用
thinkError::register()
注册异常和错误处理机制 - 加载惯例配置文件(thinkphpconvention.php)
执行应用(thinkphplibrarythinkApp.php)下的run方法
为了方便,这个run方法的代码虽然有点长,但是我还是选择把整个方法贴出来,别打我哈
/** * 执行应用程序 * @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('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); } } $request->filter($config['default_filter']); // 默认语言 Lang::range($config['default_lang']); // 开启多语言机制 检测当前语言 $config['lang_switch_on'] && Lang::detect(); $request->langset(Lang::range()); // 加载系统语言包 Lang::load([ THINK_PATH . 'lang' . DS . $request->langset() . EXT, APP_PATH . 'lang' . DS . $request->langset() . EXT, ]); // 监听 app_dispatch Hook::listen('app_dispatch', self::$dispatch); // 获取应用调度信息 $dispatch = self::$dispatch; // 未设置调度信息则进行 URL 路由检测 if (empty($dispatch)) { $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'); } // 监听 app_begin Hook::listen('app_begin', $dispatch); // 请求缓存检查 $request->cache( $config['request_cache'], $config['request_cache_expire'], $config['request_cache_except'] ); $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('default_ajax_return') : Config::get('default_return_type'); $response = Response::create($data, $type); } else { $response = Response::create(); } // 监听 app_end Hook::listen('app_end', $response); return $response; }
这大概90行的代码,具体做了什么呢,结合注释分析,主要有以下几步的功能
- 第一步:处理变量
$request
,保证有效有用不为null -
第二步:
self::initCommon()
调用当前控制器中的initCommon()方法,负责初始化应用,并返回配置信息-
Loader::addNamespace(self::$namespace, APP_PATH);
注册命名空间 -
🎜🎜위의 주소 연산 방법 외에도 Apache 또는 Nginx를 통해 가상 호스트를 구성하여 액세스할 수도 있습니다. 관심 있는 사람들은 온라인으로 특정 튜토리얼을 보고 액세스할 가상 호스트를 구성할 수 있습니다. 🎜🎜본론으로 ThinkPHP5의 실행 과정을 단계별로 분석해보겠습니다...🎜🎜입력 파일(publicindex.php)🎜🎜self::init()
publicindex.php
파일을 열면, 항목 파일의 원본 코드 다음 🎜rrreee🎜 항목 파일 코드는 매우 간결합니다. 단 두 줄의 코드로, 함수는 🎜-
define('APP_PATH', __DIR__ . '/.. /application/');
애플리케이션 디렉토리의 상수 APP_PATH를 정의합니다. -
require __DIR__ '/../thinkphp/start.php';
프레임워크를 로드합니다. 부팅 파일
define('PUBLIC_PATH) 코드 줄을 추가합니다. ', __DIR__ .'/../public');
공용 디렉토리의 상수와 일부 전처리 등을 정의합니다. 🎜🎜프레임워크 부팅 파일(thinkphpstart.php)을 로드합니다. 🎜🎜마찬가지로 thinkphpstart.php 파일을 보면 코드가 많지 않다는 것을 알 수 있습니다 🎜rrreee 🎜이 짧은 두 줄의 코드에서 왼쪽과 오른쪽에 두 개의 주요 🎜- require __DIR__ . '/base.php';기본 파일 로드
-
App::run()->send();
응용 프로그램 실행
thinkphpbase.php
파일을 엽니다. 그리고 마침내 이 파일에는 이전 두 파일처럼 코드가 두 줄밖에 없다는 것을 알게 되었습니다...🎜rrreee 🎜주의깊게 살펴보면 코드가 60줄이 넘지만 코드의 기능이 분명하다는 것을 알았습니다. 🎜-
define('', '') 사용
이 함수는 많은 시스템 상수와 두 개의 환경 상수를 정의합니다. - 로더 클래스 소개( thinkphplibrarythinkloader.php) 이후 사용을 위해
- 환경 변수 구성 파일(
.env
라는 이름의 환경 변수 구성 파일 로드, 이 파일은 반드시 존재할 필요는 없으며 실제 실행 중에 필요에 따라 추가됩니다. 개발 과정) - 🎜Call
thinkLoader::register()
자동 로딩 메커니즘 등록 🎜- 시스템 자동 로딩 등록
-
Composer
자동 로딩 지원 - 네임스페이스 정의 등록
-
런타임
캐시 디렉토리classmap에 존재하는 클래스 라이브러리 매핑 파일 로드 .php
-
extend
목차 자동 로드
-
thinkError::register( )
예외 및 오류 처리 메커니즘을 등록합니다. - 규칙 구성 파일 로드(thinkphpconvention.php)
- 첫 번째 단계: 변수
$request
처리, 유효하고 유용하며 null이 아님을 보장 - 🎜 2단계:
self::initCommon()
애플리케이션 초기화 및 구성 정보 반환을 담당하는 현재 컨트롤러에서 initCommon() 메서드를 호출합니다. 🎜-
로더: :addNamespace(self::$namespace, APP_PATH);
네임스페이스 등록 - 🎜
self::init() code>초기화하려면 이 클래스의 init() 메서드를 호출하세요. 애플리케이션🎜<ul> <li>다양한 구성 파일 로드</li> <li>동작 확장 파일 로드</li> <li>공용 파일 로드</li> <li>언어 팩 로드</li> </ul>
- 디버그 모드 관련 처리 적용
- 구성 항목
extra_file_list
를 통해 추가 파일 로드 관련 파일 로드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()
date_default_timezone_set($config['default_timezone']);
시스템 시간대 설정
Hook::listen('app_init');
을 호출하여 듣기 app_init 라벨 동작3단계: 모듈 또는 컨트롤러를 바인딩할지 결정
4단계: 시스템 언어 설정 및 로딩
-
-