9/20/2006

Rails Cache 第一步 : Page Cache

Rails 分為 Page Cache ,Action Cache,Partial Cache,現在就先來講 Page Cache。

首先,將你想 cache 的 page 在 controller 裡面加入 caches_page :action_name
class AbcController < ApplicationController
caches_page :index
end
然後確定你的環境底下有 enable cache,Rails 裡面 development 環境下是 disable cache ,production 環境下是 enable cache。如果要在 development 下面 enable cache,可以在 config/environments/development.rb 下面,加入
ActionController::Base.perform_caching = true
即可。

如此以後你 click 這個頁面,他就會幫你 cache 住頁面。如果你仔細翻 log 的話,你會發現有這一段
Cached page: /abc.html (0.00062)
然後你發現到 public 底下有一個 abc.html,裡面就是你的 cache page。如果是 index action ,他就直接 cache 成 controller 的名字。如果是其他 action ,他就 cache 成 controller_name/action_name.html。
像是 ABC controller 的 index action 就 cache 成 public/abc.html
像是 ABC controller 的 go action 就 cache 成 public/abc/go.html

你可能會覺得奇怪,為何他的 cache page 跟其他的 System 的 cache 不太一樣。其他系統的 cache 檔名可能是 ajbieohhwio.86787e27.87887073 之類,為啥Rails 的這個 page cache 這個那麼的好讀?

仔細思考一下,當我們 request abc controller 的 index action 時候。大家都是打 http://www.example.com/abc 。那麼 lighttpd 接受到這個 request 的時候(這裡僅僅指 lighttpd ,其他 web server 不確定)。他會先去 document root 尋找 abc.html。結果他就發現有 abc.html ( cache 的 page ),就立刻 return回去。當中完全沒有進到 ruby cgi,完全沒有 cgi 快慢的問題,所以這是最快的 cache 方式

如果你某種情況要 expire cache。像是修改 db 之後,要將原先的 cache reload。你可以使用
     expire_page :action => 'index'
來手動 expire 掉。

但是要注意,這種 page cache 沒有設定 expire time 的方式。所以說,要嘛就是當經過某些特定的動作時 expire 掉 cache page,要嘛就是跑一個 backend process 去清掉 cache。這個網頁有批評這個 page cache 沒有 expire 機制根本就沒有用。

不過,對這個網頁的論點,我呈反對態度。

就舉個例子,樂多Blog的作法就很類似。裡面我們個人Blog每個頁面都是靜態 HTML 頁面,而非 PHP,每當我們做出一個改變的時候,管理頁面就會幫我們產生新的靜態頁面。這樣的作法當然在產生的時候比較慢,但是 show 頁面的時候可以達成比任何 CGI 還要快的效率。

結論

Cache 好不好用是看你怎麼用,以及什麼情況要用啥方式。而不是設計出一個每個情況都可以用的方式,那只會顯得臃腫沒意義。

沒有留言: