Ruby on rails开发从头来(五十九)- ActiveRecord基础(预加载子记录)

61 篇文章 0 订阅
61 篇文章 0 订阅
  预加载子记录讨论的问题和“延迟加载”是相同的。 通常 ActiveRecord 会推迟从数据库中加载子记录,直到你需要他们,例如,通过 Rdoc 中的例子,我们假定博客程序有一个 Model ,像下面这样:

class Post < ActiveRecord::Base

belongs_to:author

has_many:comments, :order => 'created_on DESC'

end

如果我们遍历所有的post,访问作者和评论属性,我们使用一个Sql查询来返回posts表中的多条记录,并且对于每条post记录都要执行一次Sql来访问authors表和comments表,总共是2n+1次。

for post in Post.find(:all)

puts"Post: #{post.title}"

puts"Written by: #{post.author.name}"

puts"Last comment on:#{post.comments.first.created_on}"

end

上面的代码存在着严重的性能问题,我们可以通过find方法的:include参数来修正它,在find方法被执行时,会列出需要延迟加载的关联。Active Record会很聪明地使用一条SQL一次加载数据,如果有100post,和上面的代码相比,下面的代码将会消除100次数据库查询:

for post in Post.find(:all, :include=> :author)

puts"Post: #{post.title}"

puts"Written by: #{post.author.name}"

puts"Last comment on:#{post.comments.first.created_on}"

end

这个例子还可以更简化,只使用一条SQL

for post in Post.find(:all, :include=> [:author, :comments])

puts"Post: #{post.title}"

puts"Written by: #{post.author.name}"

puts"Last comment on: #{post.comments.first.created_on}"

end

预加载子记录并不能保证改善性能(事实上,如果你的数据库不支持左连接,那么你就不能使用延迟加载,如果你使用的是oracle8,就必须要升级到oracle9才可以),因为预加载在Sql中加入了所有的表,这样就会有很多记录被加载后转换成Model类的对象,这样就有可能导致内存使用的问题,特别是当一条记录有很多条子记录时,这种情况就更明显,与一条条地延迟加载子记录相比,预加载会占用更多的服务器内存。

如果你使用了:include,你需要使用find方法的参数消除列名的歧义,在列名前添加表名来区分,在下面的例子中,title列需要添加表名前缀:

for post in Post.find(:all,:conditions => "posts.title like '%ruby%'",

:include=> [:author, :comment])

# ...

end


  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值