N + 1 查询问题

今天碰到性能问题,便查了一下rails的文档。

考虑下面的代码,找出10个client并打印相关的postcode:

clients = Client.limit(10)
 
clients.each do |client|
  puts client.address.postcode
end

第一眼看上去是没啥问题的。但是问题存在于总共执行的查询的数量。上面的代码总共执行了1(为了找10个client)+10(每个client载入address一次)=11次查询。

解决N+1查询

ActiveRecord允许我们指定所有将会被载入的关联。这个可以通过在Model.find调用时指定includes方法来完成。借助includesActive Record确保所有的指定的关联已经通过最少可能的查询载入。

重新改写上述代码,我们可以把Client.limit(10)重写为载入地址:

clients = Client.includes(:address).limit(10)

clients.each do |client|
  puts client.address.postcode
end

上面的代码只会执行两次查询,比之前11次查询要好很多:

SELECT * FROM clients LIMIT 10
SELECT addresses.* FROM addresses
  WHERE (addresses.client_id IN (1,2,3,4,5,6,7,8,9,10))

你可能感兴趣的:(N + 1 查询问题)