今天碰到性能问题,便查了一下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
方法来完成。借助includes
,Active 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))