5

我这样做(在我看来):

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

如果用户有十个帖子,这是否会进行十次数据库调用?这些循环应该像(不那么漂亮)吗?:

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

是我用来测试的 ruby​​ 文件的粘贴箱。编辑修改了粘贴箱。

这让我想起了Java中的代码

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

那应该是(除非数组被修改)

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

我错过了什么吗?我看到一堆 Rails 示例,其中人们循环遍历has_many对象的某个字段。这对我很重要,因为我正在尝试优化我的应用程序的性能。

4

1 回答 1

10

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

没有。

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

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


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

Rails 使用关联代理对象来保存您的查询,当您需要记录时执行查询,然后缓存结果。这是一段非常有用的代码,并且在大多数情况下,它会为您处理类似的事情。

这是Rails的一个特性,而不是 ruby​​。


在 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因为它们在每次迭代中都增加了一个。each出于同样的原因,你的循环在 ruby​​ 中也会发生同样的事情。

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

于 2012-12-04T18:41:50.077 回答