yii2深刻理解以內核解析

1、前言php

 

首先,yii2最爲爲數很少的PHP主流開源框架,受歡迎程度不亞於laravel和TP。我的認爲,研究這些框架底層代碼是很是有助於自身代碼編程思想的提高和代碼簡化程度和質量的提高的。css

那麼,話很少說,下面開始說一下本身對yii2底層框架架構的理解。前端

2、客戶端請求處理nginx

 

yii2對客戶請求處理,提供了url美化類urlManager,這點在最初經過composer加載,並初始化框架時,已經在config目錄下的main文件裏生成了, 只不過期註釋過的,若是須要url美化則取消掉註釋便可。其實,yii2對apache的支持是很是的好的。laravel

說到這了,那麼咱們就說一下,後端最流行的兩個服務器,apache和nginx的區別。web

a、nginx性能比apache要好不少,相比apache,但bug就會相對多。算法

b、nginx相比apache輕量級了很多,配置也相對簡單,並且,對高併發的請求事件處理,能夠是異步非阻塞的處理請求,能消耗低。數據庫

c、apache相比nginx,它的重寫能力要比nginx強不少,並且,功能模塊比nginx多不少,其性能比nginx穩定,bug較少。apache

因此說,yii2的url美化組件urlManager對apache服務器有很好的支持。編程

框架和項目配置,咱們能夠直接在common文件下的config文件下的main-local文件裏配置,數據庫的鏈接以及數據表的前綴等。web文件夾下的.htaccess文件內容以下:

 1  Header always set Access-Control-Allow-Origin "*"
 2  Header always set Access-Control-Allow-Headers: "X-Requested-With, Content-Type, Origin, Authorization, Accept, Client-Security-Token, Accept-Encoding"
 3  RewriteEngine ON
 4  RewriteCond %{REQUEST_FILENAME} !-f
 5  RewriteCond $1 !^(index\.php|robots\.txt|favicon\.ico|Sitemap\.xml|pma)
 6  RewriteRule ^(.*)$ index.php/$1 [L]
 7  php_value upload_max_filesize "50M"
 8 php_value post_max_size "50M"
 9 php_value max_execution_time "40000"
10 php_value memory_limit "128M"

代碼解析一下:

RewriteEngine On

指令含義:命令apache或nginx重寫功能開啓。

RewriteCond %{REQUEST_FILENAME} !-f

指令含義:URI匹配內容爲非文件。

RewriteCond $1 !^(index\.php|robots\.txt|favicon\.ico|Sitemap\.xml|pma)

指令含義:URI匹配內容非文件,除了這幾個文件。注:這裏用到了正則的反引用。

RewriteRule ^(.*)$ index.php/$1 [L]

指令含義:URL匹配規則,index.php/site/login

下面來看一下web目錄下的文件都有哪些:

a. assets yii2底層必備靜態文件夾,這裏放了一些,yii2靜態模板組件的必須引入的靜態文件。如:jQuery,bootstrap等。注:此文件夾不可刪除;

assets文件夾下面的文件,在你第一次訪問項目時,會自動生成靜態文件,這裏用到了yii2的另一個服務和組件assetManager,在它的裏面有一個方法,publish(),其具體用法以下:

1 <?php
2 
3 list(, $dir) = Yii::$app->getAssetManager()->publish('@backend/rbac/plugins');
4 $svg = $dir . '/image/loading.svg';

此方法須要傳入一個alias路由,這alias路由其實在common文件下的config文件夾下的bootstrap.php已經配置。代碼以下:

1 <?php
2 Yii::setAlias('@common', dirname(__DIR__));
3 Yii::setAlias('@frontend', dirname(dirname(__DIR__)) . '/frontend');
4 Yii::setAlias('@backend', dirname(dirname(__DIR__)) . '/backend');
5 Yii::setAlias('@console', dirname(dirname(__DIR__)) . '/console');
6 Yii::setAlias('@wechat', dirname(dirname(__DIR__)) . '/wechat');
7 Yii::setAlias('@api', dirname(dirname(__DIR__)) . '/api');

當傳入alias路由,此方法會解析出文件絕對路由,並在assets文件夾下生成對應的文件夾和其文件。這個方法用到了crc32()多項冗餘算法,經過文件名和文件修改時間,生成帶有符號的crc32值,並經過sprinf()轉換爲16進制字符做爲文件名。

b.css文件這裏面寫入了項目須要引入的前端靜態文件,好比:css,js,svg,圖片等;

3、index.php腳本文件運行機制


項目腳本文件,最後一行簡潔的代碼,包含了yii2架構全部運行的機制。

<?php
(new yii\web\Application($config))->run();


這裏有幾個類:

1.yii\web\Application

2.yii\base\Application

3.yii\base\Module

4.yii\di\ServiceLocator

5.yii\base\component

6.yii\base\BaseObject

首先,new yii\web\Application類,這裏的構造函數在yii\base\Application下:

1 public function __construct($config = [])
2 {
3 Yii::$app = $this;
4 static::setInstance($this);
5 
6 $this->state = self::STATE_BEGIN;
7 
8 $this->preInit($config);
9 
10 $this->registerErrorHandler($config);
11 
12 Component::__construct($config);
13 }


這裏它作了什麼?

1.單例yii\web\Application應用核心類。

2.註冊yii\web\Application模塊到註冊樹。

3.指定web應用運行狀態:初始化

4.初始化配置,拉入common和backend目錄下config目錄下bootstrap.php文件配置。

5.註冊錯誤處理程序。

6.加載核心組件、核心配置、以及自定義行爲組件等。(注:這些都是common和backend文件下config文件下main.php裏面的內容)

接下來運行run方法:

1 public function run()
2 {
3 try {
4 $this->state = self::STATE_BEFORE_REQUEST;
5 $this->trigger(self::EVENT_BEFORE_REQUEST);
6 
7 $this->state = self::STATE_HANDLING_REQUEST;
8 $response = $this->handleRequest($this->getRequest());
9 
10 $this->state = self::STATE_AFTER_REQUEST;
11 $this->trigger(self::EVENT_AFTER_REQUEST);
12 
13 $this->state = self::STATE_SENDING_RESPONSE;
14 $response->send();
15 
16 $this->state = self::STATE_END;
17 
18 return $response->exitStatus;
19 } catch (ExitException $e) {
20 $this->end($e->statusCode, isset($response) ? $response : null);
21 return $e->statusCode;
22 }
23 }

這個方法裏包含了從初始化beforeRequest事件,到afterRequest的全部程序,可謂是核心方法。

那麼我們來看一下,都有些什麼?

首先,執行beforeRequest綁定的事件,指定web應用運行狀態。

後者,處理請求url和參數。

再者,解析路由,實例化controller類、action類。

而後,執行beforeAction事件和處理action程序、afterAction事件。

最後,執行afterRequest事件,實例response指定data,響應客戶端。

好了,因爲時間關係,此次就解析這麼多。若有建議和不一樣意見,請關注博客,敬請留言。

相關文章
相關標籤/搜索