2012-12-04 196 views
5

我这样做(在我看来):红宝石。每个效率

# myUser is a User in ActiveRecord with :has_many :posts 
myUser.posts.each do |post| 
end 

如果用户有10个职位,这会是做一个数据库调用十倍?如果这些循环是这样(非常少)?:

myPosts = myUser.posts 
myPosts.each do |post| 
end 

Here的是,我做了测试红宝石文件的糊纸盒。 编辑修改了粘贴文件夹。

这让我想起的代码在Java中

for (int i = 0; i < someExpensiveFunction(); i++) 

应(除非修改该数组)

for (int i = 0, len=someExpensiveFunction(); i < len; i++) 

我缺少的东西?我看到一些轨道示例,其中人们通过对象的某个has_many字段进行循环。这对我来说很重要,因为我试图优化我的应用程序的性能。

回答

10

如果用户有10个帖子,这在技术上会做数据库调用10次?

没有。

myUser = User.find 123 
myUser.posts.each do |post| 
    puts post.title 
end 

此处的代码将运行2个查询。第一个将通过它的id找到一个用户,返回一行。第二个将运行一个查询,询问具有与myUser匹配的用户标识的所有帖子。然后each将使用该结果迭代。


如果你看在日志中的发展模式下,它会告诉你它的运行查询,你应该会看到一个查询返回所有那些帖子。

Rails使用关联代理对象来保存您的查询,当您需要记录时执行查询,然后缓存结果。这是一个非常有用的代码,大部分它为你处理这样的事情。

这是Rails的一个特性,不是红宝石。


在ruby级别,each只是作用于集合。

def get_letters 
    puts 'executing "query"' 
    sleep(3) 
    ["a","b","c"] 
end 

get_letters.each do |item| 
    puts item 
end 

这应该打印出来。

executing "query" 
a 
b 
c 

get_letters方法执行,它会返回一个数组,一个数组已满的数据就是我们所说的each方法,这样我们就可以处理每个项目。获取集合并遍历它是2个完全独立的步骤。


从你的引擎收录:

# will this run forever? 
myArr = ["a","b","c"] 
myArr.each do |item| 
    myArr.push("x") 
end 

呀会,但不是因为阵列被过度过“牵强”。

这相当于像这样的javascript,这更好地说明了这一点。

myArr = ["a","b","c"]; 
for (var i = 0; i < myArr.length; i++) { 
    myArr.push('x'); 
} 

原来的阵列,在循环检查的每个迭代,看看它是否没有完成。但是因为阵列每次获得一个项目的时间会增加一个,所以i < myArr.length将始终为true,因为它们在每次迭代时都会增加1。出于同样的原因,您的红宝石与您的each循环相同。

它永远运行,不是因为你不停地运行一些代码来重新生成数组。它永远运行,因为你是人为地改变结果数组。

+0

谢谢,但问题。在我的粘贴箱作为最后一个例子,循环永远运行......我不明白为什么会这样,如果结果会被“缓存”后,Ruby会修改缓存吗?如果我在.each帖子循环中添加帖子会发生什么? – K2xL

+0

@ K2xL请参阅我的编辑,它在基本ruby(非rails)图层上解释了更多内容。但是,你的循环,正如张贴,绝对不_永远不会“永远运行”。 3秒后,完成。 –

+0

看到我的编辑(我无意中发布了错误的pastebin url) – K2xL