PHP之深入學習Yii2緩存Cache組件詳細講解

什麼是緩存組件Cache

緩存是提升 Web 應用性能簡便有效的方式。 通過將相對靜態的數據存儲到緩存並在收到請求時取回緩存, 應用程序便節省瞭每次重新生成這些數據所需的時間。

在這裡插入圖片描述

定義緩存組件

Yii2的緩存是通過組件Component實現的,在項目的配置文件中,配置components->cache實現對緩存組件的定義。

項目配置文件的路徑為config/web.php。

在這裡插入圖片描述

頁面緩存PageCache

作為網站來講,Yii2的頁面緩存非常便捷地將已經渲染完全的網頁結果保存起來,並在一個緩存周期內不需要再次處理頁面內部的控制器動作邏輯。

配置頁面緩存

頁面緩存的配置方式為,在控制器層Controller中配置行為behaviors,通過調用過濾器filters的方式,在進入具體頁面路徑action的之前,對當前key進行計算,並判斷緩存是否啟用enabled緩存有效期duration。

基礎配置代碼如下所示

        return [
            'pageCache' => [
                'class' => 'yii\filters\PageCache',
                'only' => ['index'],
                'variations' => [
                    '/',
                    Yii::$app->request->isAjax,
                ],
                'enabled'=>true,
                'duration' => Yii::$app->params['pageCacheDuration'],
            ],
        ];

過濾器是Yii2中的一個概念,他可以在控制器初始化的時候加載並執行,我們可以用這個特點去做一些對控制器的數據的限制,比如控制緩存、用戶權限控制。

這裡我們將行為名稱定義為pageCache,顯然名字不重要,因為有的案例中,因為不同的頁面緩存規則不一樣,我會定義兩個頁面緩存的行為。

其中only為過濾器調用action的參數,用於限制哪些路徑是啟用action的。

頁面緩存PageCache是緩存組件Cache的一種應用

頁面緩存的根本邏輯為

  1. 配置緩存組件的實現比如文件緩存yii\caching\FileCache
  2. 頁面緩存封裝一層Cache組件,再去調用存取邏輯

我們可以通過查看頁面緩存源碼vendor/yiisoft/yii2/filters/PageCache.php,我們可以在文件的第162行發現,這裡調用的cache,就是對於緩存的實現。

$this->cache = Instance::ensure($this->cache, 'yii\caching\CacheInterface');

自定義頁面緩存過濾器

為什麼我們需要自定義緩存組件呢,我歸納原因存在以下幾種

  • 緩存判斷邏輯過於簡單或復雜,不如自己重寫痛快地多
  • 緩存key生成方式不滿足業務需求

那麼如何自定義呢?我個人推薦最簡單粗暴的方式,繼承。

use yii\filters\PageCache;

class PageCacheCtInfo extends PageCache
{
	這裡是內部邏輯,不需要重寫的方法可以不寫。
	public $checkUser = true; //可以自定義變量
}

調用方式也是跟默認的頁面緩存一樣,隻要換上對應的類即可。

            'pageCacheInfo' => [
                'class' => 'common\components\PageCacheCtInfo',
                'only' => ['info'],
                'enabled'=>Yii::$app->params['pageCacheEnabled'],
                'variations' => [
                    'ct/'.Yii::$app->request->pathInfo,
                    Yii::$app->request->isAjax
                ],
                'duration' => Yii::$app->params['pageCacheInfo'],
                'checkUser' = false,
            ],

頁面緩存key的計算

根據上一個步驟,我們可以重寫計算key的方式,那麼之前的key計算方式是什麼樣的呢?

文件位置vendor/yiisoft/yii2/filters/PageCache.php。

    /**
     * @return array the key used to cache response properties.
     * @since 2.0.3
     */
    protected function calculateCacheKey()
    {
        $key = [__CLASS__];
        if ($this->varyByRoute) {
            $key[] = Yii::$app->requestedRoute;
        }
        return array_merge($key, (array)$this->variations);
    }

這裡的緩存key是一個數組,數組內的元素依次是

  • 當前類名
  • varyByRoute 一般為true
  • variations 驗證,這個也是配置中獲取的,根據上面的配置,則是頁面路徑和是否為ajax

如果是項目的首頁,緩存的key則為

['yii\filters\PageCache','','/‘,0]

如果是個詳情頁面,key為

['yii\filters\PageCach', 'xxx/info','xxx/xxx/3xxxx74.html',0 ]

那麼,這個key到底有什麼用,為什麼要單獨拿出來說呢?

因為我們需要單獨刪除某個頁面緩存。

主動清理過期緩存

根據源碼vendor/yiisoft/yii2/caching/FileCache.php

    /**
     * Stores a value identified by a key in cache.
     * This is the implementation of the method declared in the parent class.
     *
     * @param string $key the key identifying the value to be cached
     * @param string $value the value to be cached. Other types (If you have disabled [[serializer]]) unable to get is
     * correct in [[getValue()]].
     * @param int $duration the number of seconds in which the cached value will expire. 0 means never expire.
     * @return bool true if the value is successfully stored into cache, false otherwise
     */
    protected function setValue($key, $value, $duration)
    {
        $this->gc();
        $cacheFile = $this->getCacheFile($key);
        if ($this->directoryLevel > 0) {
            @FileHelper::createDirectory(dirname($cacheFile), $this->dirMode, true);
        }
        // If ownership differs the touch call will fail, so we try to
        // rebuild the file from scratch by deleting it first
        // https://github.com/yiisoft/yii2/pull/16120
        if (is_file($cacheFile) && function_exists('posix_geteuid') && fileowner($cacheFile) !== posix_geteuid()) {
            @unlink($cacheFile);
        }
        if (@file_put_contents($cacheFile, $value, LOCK_EX) !== false) {
            if ($this->fileMode !== null) {
                @chmod($cacheFile, $this->fileMode);
            }
            if ($duration <= 0) {
                $duration = 31536000; // 1 year
            }

            return @touch($cacheFile, $duration + time());
        }

        $error = error_get_last();
        Yii::warning("Unable to write cache file '{$cacheFile}': {$error['message']}", __METHOD__);
        return false;
    }

在設置緩存之前會主動調用清理緩存的方法gc()

    /**
     * Removes expired cache files.
     * @param bool $force whether to enforce the garbage collection regardless of [[gcProbability]].
     * Defaults to false, meaning the actual deletion happens with the probability as specified by [[gcProbability]].
     * @param bool $expiredOnly whether to removed expired cache files only.
     * If false, all cache files under [[cachePath]] will be removed.
     */
    public function gc($force = false, $expiredOnly = true)
    {
        if ($force || mt_rand(0, 1000000) < $this->gcProbability) {
            $this->gcRecursive($this->cachePath, $expiredOnly);
        }
    }

這裡問題就出現瞭,$gcProbability的默認值是10,也就是說,隻有0.001%的概率會在設置緩存的同時清理過期緩存。

這不就跟沒有一樣!

所以對於緩存來說,需要我們主動定期清理過期緩存,不然對應的存儲空間就會被占滿。

Yii::$app->cache->gc(true);

優化緩存配置

組件的cache在項目的配置文件中定義

'components' => ['cache' => [
    'class' => 'yii\caching\FileCache',
],],

這裡的自由度就出現瞭,現在這個配置,是文件緩存,也就是不管是數據緩存還是頁面緩存,都是保存在文件裡的

根據源碼 public $cachePath = ‘@runtime/cache’;

緩存的文件是放在runtime/cache文件夾的

那麼問題就出現瞭,磁盤的性能是有瓶頸的,文件讀寫會影響緩存性能。

目前可選的緩存有

  • yii\caching\ApcCache,APC擴展
  • yii\caching\DbCache,數據庫緩存
  • yii\caching\DummyCache,假的緩存,就是現在沒條件上緩存先把坑占上
  • yii\caching\FileCache,文件緩存
  • yii\caching\MemCache,使用 PHP memcache 和 memcached 擴展
  • yii\redis\Cache,redis
  • yii\caching\WinCache,使用 PHP WinCache 擴展
  • yii\caching\XCache,使用 PHP XCache擴展
  • yii\caching\ZendDataCache,使用Zend Data Cache

總結

我在本文中,通過漸進的方式,講瞭如何使用Yii2的緩存組件,對於一般的使用者來講,已經涵蓋瞭超過九成的坑。

如果你正在學習PHP,希望你收藏這篇文章,這會對你以後有所幫助。

在這裡插入圖片描述

到此這篇關於PHP之深入學習Yii2緩存Cache組件詳細講解的文章就介紹到這瞭,更多相關PHP之深入學習Yii2緩存Cache組件內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: