唯一的办法就是 monkey patch了,自己写吧.
第一个版本
require 'active_record/associations/belongs_to_association' require 'active_record/associations/has_many_association' module ActiveRecord class Base public def self.has_one_cached *params has_one *params # call the standard association assoc_name = params[0] define_method ("#{assoc_name}_with_cache") do assoc = self.class.reflections[assoc_name.to_sym] instance_values[assoc_name.to_s] || (assoc.klass.get_cache(instance_values["#{assoc_name}_cached_id"]) if instance_values["#{assoc_name}_cached_id"] ) || (a = HasOneAssociation.new(self, assoc) assoc.klass.get_cache(a.id) self.instance_variable_set("@#{assoc_name}_cached_id", a.id) self.instance_variable_set("@#{assoc_name}", a) ) end # switch out the method. has_one -> has_one_with_cache alias_method_chain assoc_name, :cache end end end
代码解释下,捡重要的说吧
1-7行,可以忽略
8 增加一个 has_one_cache的方法,使用起来等同于 has_one
17-23行的三个 "||"
第一个,就是读取AR中的缓存数据,这个没啥说的
第二个"||",是根据instance中保存的instance_values["#{assoc_name}_cached_id"]从memcache中获取对应的数据
第三个"||",就是第一次运行的时候,保存一个cache_id的变量和值,和对AR对象中缓存进行赋值
所以这里实际上是三级的缓存读取方式,先对象缓存,再memcache,如果还没有就直接读取数据库.
不过这里有个很明显的缺陷.要求instance长期存在,这个代码才能起作用.不过rails中对象的生命周期仅限于action,超出这个就没有啥用了.所以上面的代码,到头来只能在偶尔的情况下才能真正的用到memcache
解决办法是有,不过看各位是否喜欢的问题,就是把id也保存到memcache中.呵呵,很土的没有办法的办法.
于是上面的方法就改成了
define_method ("#{assoc_name}_with_cache") do assoc = self.class.reflections[assoc_name.to_sym] instance_values[assoc_name.to_s] || (assoc.klass.get_cache(instance_values["#{assoc_name}_cached_id"]) if instance_values["#{assoc_name}_cached_id"] ) || (id = assoc.klass.get_cache_ex("#{self.class.to_s}:#{self.id}:#{assoc_name}") assoc.klass.get_cache(id) if id)|| (a = HasOneAssociation.new(self, assoc) assoc.klass.get_cache(a.id) assoc.klass.set_to_cache("#{self.class.to_s}:#{self.id}:#{assoc_name}",a.id,assoc.klass.ttl) self.instance_variable_set("@#{assoc_name}_cached_id", a.id) self.instance_variable_set("@#{assoc_name}", a) ) end
增加了一个"||"
(id = assoc.klass.get_cache_ex("#{self.class.to_s}:#{self.id}:#{assoc_name}") assoc.klass.get_cache(id) if id)
先用memcache中获取id数据,在获取对应的纪录,很囧吧.
不过要看应用场合的,如果你需要访问的数据属于基础数据,永远不变的,或者长期不变的数据,这样的办法能够减轻数据库的访问负担,但是要注意的是"这样的方法不会比直接访问数据库要快"
这里重申下memcache的作用:
"memcache不会使得你的网站访问速度更快,只是为了减轻数据库的访问负担"
另外顺便提下,cache_fu中没有get_cache_ex的函数.这个函数是在robbin的提示下加进来,专门用于非id key的memcache数据读写的.
代码如下
def get_cache_ex(key, timeout = 60 * 30) reset = cache_reset data = get_from_cache(key) unless reset return data unless data.nil? if block_given? if data = yield test = true if data.instance_of? Array test = false if data.size ==0 end set_to_cache(key, data, timeout) if test end return data end end
还有,我这里的代码是参考http://groups.google.com/group/acts_as_cached/browse_thread/thread/5fefb2d2355a5048/25ea72ecd0d5a0c9
这个thread写的.
如果有has_many, belongs_to 的需求,还是参考google groups上的帖子来改写吧.
最后感谢下robbin的自动保存功能,真是救命功能阿