• 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31

(追記) 経緯:attr_readonlyを追加しました。

はじめに、counter_cacheの使い方

例えば、Blogテーブルにarticles_countカラムを持つことで、blog.articles.sizeをキャッシュし、増減はrailsが自動でやってくれます。 こんな感じ。

   1  class Article < ActiveRecord::Base
   2    belongs_to :blog, :counter_cache => true

経緯

Rails2.0からカウンターキャッシュカラム(ex Blog#articles_count)がどうやらattr_readonlyになったようで、migration内でupdate_attributeで数が更新できなったので、困った。エラーもでなくて、データで確認すると更新できてない状態。

   1  NG for Rails2.0
   2   blog.update_attribute :articles_count, total 

本題、migrationでの注意

  • カラム名は複数形_count(ex. articles_count)がデフォルト
  • カラム名のカスタマイズは、:counter_cache => :my_custom_counter
  • カラムはdefault => 0
  • 既存のテーブルでcounterカラムを追加する場合はupdate_countersを使う

   1    def self.up
   2      add_column :blogs, :articles_count, :integer, :default => 0
   3  
   4      Blog.find(:all).each do |blog|
   5        Blog.update_counters(blog.id, :articles_count => blog.articles.count)
   6      end

reference

http://josh.the-owens.com/archives/2007/11/03/rails-edge-change-how-to-add-a-counter-cache-to-an-existing-db-table/

posted by Png satoko on Tue 15 Jan 2008 at 00:40 with 2 comments

あすなろとダブルポストです。+最後の感想ちょこっと変更しました。

まとめ

CtonrollerやActionを指定して処理を実行できます

after_contoroller_action
before_contoroller_action

ex. after_blog_save
→ BlogController#save終了後処理をする。

流れをわかるためのもがき

  • ぐぐってヒントを見つける。
    [Rails][cache] Sweeper
    http://d.hatena.ne.jp/meritdemerit/20070607/p2
    なんとなくはわかるけど、やっぱりよくわからないのでソースを見ることにする。

  • Rails Referenceで検索する。右上にファイルの場所があるので確認。
    vendor/rails/actionpack/lib/action_controller/caching.rb
    http://api.rubyonrails.org/classes/ActionController/Caching/Sweeping.html

  • ソースを読む:cache_sweeperキーワード

       1    def cache_sweeper(*sweepers)
       2      return unless perform_caching
       3      configuration = sweepers.extract_options!
       4      sweepers.each do |sweeper|
       5        ActiveRecord::Base.observers << sweeper if defined?(ActiveRecord) and defined?(ActiveRecord::Base)
       6        sweeper_instance = Object.const_get(Inflector.classify(sweeper)).instance
       7        if sweeper_instance.is_a?(Sweeper) 
       8         around_filter(sweeper_instance, :only => configuration[:only])
       9        else
      10          after_filter(sweeper_instance, :only => configuration[:only])
      11        end
      12      end
      13    end
    

  • ソースを読む:after処理のエントリポイント

       1   def after(controller)
       2      callback(:after)
       3      # Clean up, so that the controller can be collected after this request
       4      self.controller = nil
       5    end
    

  • ソースを読む:after_controller_action処理の呼び出し

       1    def callback(timing)
       2      controller_callback_method_name = "#{ timing}_#{controller.controller_name.underscore}"
       3      action_callback_method_name     = "#{controller_callback_method_name}_#{controller.action_name}"
       4      send!(controller_callback_method_name) if respond_to?(controller_callback_method_name, true)
       5      send!(action_callback_method_name)     if respond_to?(action_callback_method_name, true)
       6    end
    

最後の、"send!(action...:自分にメソッドが定義されていたら送る"ってところが実際の呼び出し。
愛が足りないながんばろ。

posted by Png satoko on Mon 14 Jan 2008 at 06:32
Contents
既存のモデル/テーブルでcounter_cacheを使う:update_counters
Sweeperを調べる:ControllerやActionを指定して処理する
Comments
KingofSmack: Here also good reads for this mobile applicatio... '14-5
satoko: stackoverflowでも同じエラーを挙げている人がいたので、1.3でアップロードしたよっ... '10-12
ujihisa: :%s/blog/glob/g '10-7
satoko: しゅが〜様 返事が遅くなって申し訳ありません。また、投稿百景ご購入ありがとうございます。 ... '09-10
しゅが~: こんにちは。投稿百景を発売日翌日から利用しています。本当にいいAppを作っていただきました。罫... '09-10
Services from s21g
twpro(ツイプロ)
Twitterプロフィールを快適検索
地価2009
土地の値段を調べてみよう
MyRestaurant
自分だけのレストラン手帳
Formula
ブログに数式を埋め込める数式コミュニティ