Ruby。每个效率

时间:2012-12-04 18:37:36

标签: ruby-on-rails ruby performance activerecord

我(在我看来):

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

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

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

Here是我要测试的ruby文件的粘贴框。 编辑修改了粘贴箱。

这让我想起了Java中的代码

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

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

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

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

1 个答案:

答案 0 :(得分:10)

  

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

不。

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

这里的代码将运行2个查询。第一个将通过其id找到用户,返回单行。第二个将运行查询,询问所有具有匹配myUser的用户ID的帖子。然后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方法,所以我们可以处理每个项目。获取集合并迭代它是两个完全独立的步骤。


从你的pastebin:

# 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循环在ruby中也会发生同样的事情。

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