Clean way to find ActiveRecord objects by id in the order specified
I want to obtain an array of ActiveRecord objects given an array of ids.
I assumed that
Object.find([5,2,3])
Would return an array with object 5, object 2, then object 3 in that order, but instead I get an array ordered as object 2, object 3 and then object 5.
The ActiveRecord Base find method API mentions that you shouldn't expect it in the order provided (other documentation doesn't give this warning).
One potential solution was given in Find by array of ids in the same order?, but the order option doesn't seem to be valid for SQLite.
I can write some ruby code to sort the objects myself (either somewhat simple and poorly scaling or better scaling and more complex), but is there A Better Way?
ruby-on-rails sqlite activerecord find rails-activerecord
add a comment |
I want to obtain an array of ActiveRecord objects given an array of ids.
I assumed that
Object.find([5,2,3])
Would return an array with object 5, object 2, then object 3 in that order, but instead I get an array ordered as object 2, object 3 and then object 5.
The ActiveRecord Base find method API mentions that you shouldn't expect it in the order provided (other documentation doesn't give this warning).
One potential solution was given in Find by array of ids in the same order?, but the order option doesn't seem to be valid for SQLite.
I can write some ruby code to sort the objects myself (either somewhat simple and poorly scaling or better scaling and more complex), but is there A Better Way?
ruby-on-rails sqlite activerecord find rails-activerecord
1
Where are those id's coming from? If it is the UI (via a user selecting them) then scaling shouldn't be an issue, that is the user is unlikely to spend time selecting 1000's of ids). If it's the database (e.g. from a join table), could you store the order in the join table and issue the find based on that?
– pauliephonic
Apr 29 '09 at 11:13
It looks like this is no longer true in Rails 5.
– XML Slayer
Jun 6 '18 at 19:49
add a comment |
I want to obtain an array of ActiveRecord objects given an array of ids.
I assumed that
Object.find([5,2,3])
Would return an array with object 5, object 2, then object 3 in that order, but instead I get an array ordered as object 2, object 3 and then object 5.
The ActiveRecord Base find method API mentions that you shouldn't expect it in the order provided (other documentation doesn't give this warning).
One potential solution was given in Find by array of ids in the same order?, but the order option doesn't seem to be valid for SQLite.
I can write some ruby code to sort the objects myself (either somewhat simple and poorly scaling or better scaling and more complex), but is there A Better Way?
ruby-on-rails sqlite activerecord find rails-activerecord
I want to obtain an array of ActiveRecord objects given an array of ids.
I assumed that
Object.find([5,2,3])
Would return an array with object 5, object 2, then object 3 in that order, but instead I get an array ordered as object 2, object 3 and then object 5.
The ActiveRecord Base find method API mentions that you shouldn't expect it in the order provided (other documentation doesn't give this warning).
One potential solution was given in Find by array of ids in the same order?, but the order option doesn't seem to be valid for SQLite.
I can write some ruby code to sort the objects myself (either somewhat simple and poorly scaling or better scaling and more complex), but is there A Better Way?
ruby-on-rails sqlite activerecord find rails-activerecord
ruby-on-rails sqlite activerecord find rails-activerecord
edited Jun 9 '14 at 4:34
mu is too short
353k58696671
353k58696671
asked Apr 29 '09 at 10:53
Andrew GrimmAndrew Grimm
44.1k35161279
44.1k35161279
1
Where are those id's coming from? If it is the UI (via a user selecting them) then scaling shouldn't be an issue, that is the user is unlikely to spend time selecting 1000's of ids). If it's the database (e.g. from a join table), could you store the order in the join table and issue the find based on that?
– pauliephonic
Apr 29 '09 at 11:13
It looks like this is no longer true in Rails 5.
– XML Slayer
Jun 6 '18 at 19:49
add a comment |
1
Where are those id's coming from? If it is the UI (via a user selecting them) then scaling shouldn't be an issue, that is the user is unlikely to spend time selecting 1000's of ids). If it's the database (e.g. from a join table), could you store the order in the join table and issue the find based on that?
– pauliephonic
Apr 29 '09 at 11:13
It looks like this is no longer true in Rails 5.
– XML Slayer
Jun 6 '18 at 19:49
1
1
Where are those id's coming from? If it is the UI (via a user selecting them) then scaling shouldn't be an issue, that is the user is unlikely to spend time selecting 1000's of ids). If it's the database (e.g. from a join table), could you store the order in the join table and issue the find based on that?
– pauliephonic
Apr 29 '09 at 11:13
Where are those id's coming from? If it is the UI (via a user selecting them) then scaling shouldn't be an issue, that is the user is unlikely to spend time selecting 1000's of ids). If it's the database (e.g. from a join table), could you store the order in the join table and issue the find based on that?
– pauliephonic
Apr 29 '09 at 11:13
It looks like this is no longer true in Rails 5.
– XML Slayer
Jun 6 '18 at 19:49
It looks like this is no longer true in Rails 5.
– XML Slayer
Jun 6 '18 at 19:49
add a comment |
10 Answers
10
active
oldest
votes
It's not that MySQL and other DBs sort things on their own, it's that they don't sort them. When you call Model.find([5, 2, 3])
, the SQL generated is something like:
SELECT * FROM models WHERE models.id IN (5, 2, 3)
This doesn't specify an order, just the set of records you want returned. It turns out that generally MySQL will return the database rows in 'id'
order, but there's no guarantee of this.
The only way to get the database to return records in a guaranteed order is to add an order clause. If your records will always be returned in a particular order, then you can add a sort column to the db and do Model.find([5, 2, 3], :order => 'sort_column')
. If this isn't the case, you'll have to do the sorting in code:
ids = [5, 2, 3]
records = Model.find(ids)
sorted_records = ids.collect {|id| records.detect {|x| x.id == id}}
14
Another solution that is more performant than using #detect to avoid O(N) performance: records = Model.find(ids).group_by(&:id); sorted_records = ids.map { |id| records[id].first }
– Ryan LeCompte
Dec 17 '11 at 0:04
add a comment |
Based on my previous comment to Jeroen van Dijk you can do this more efficiently and in two lines using each_with_object
result_hash = Model.find(ids).each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
For reference here is the benchmark i used
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
result_hash = results.each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
end
end.real
#=> 4.45757484436035 seconds
Now the other one
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
ids.collect {|id| results.detect {|result| result.id == id}}
end
end.real
# => 6.10875988006592
Update
You can do this in most using order and case statements, here is a class method you could use.
def self.order_by_ids(ids)
order_by = ["case"]
ids.each_with_index.map do |id, index|
order_by << "WHEN id='#{id}' THEN #{index}"
end
order_by << "end"
order(order_by.join(" "))
end
# User.where(:id => [3,2,1]).order_by_ids([3,2,1]).map(&:id)
# #=> [3,2,1]
add a comment |
Apparently mySQL and other DB management system sort things on their own. I think that you can bypass that doing :
ids = [5,2,3]
@things = Object.find( ids, :order => "field(id,#{ids.join(',')})" )
That was the answer suggested in the link "Find by arrays of id in the same order?", but it doesn't seem to work for SQLite.
– Andrew Grimm
Apr 29 '09 at 23:30
3
And it doesn't seem to work for Postgres 9
– Jeroen van Dijk
Aug 24 '11 at 13:31
2
This no longer works. For more recent Rails:Object.where(id: ids).order("field(id, #{ids.join ','})")
– mahemoff
Apr 18 '15 at 8:37
Improved:Object.where(id: ids).order(ids.present? && "field(id, #{ids.join ','})")
. This prevents a SQL error if the IDs array is empty or nil.
– mahemoff
Apr 18 '15 at 9:09
add a comment |
A portable solution would be to use an SQL CASE statement in your ORDER BY. You can use pretty much any expression in an ORDER BY and a CASE can be used as an inlined lookup table. For example, the SQL you're after would look like this:
select ...
order by
case id
when 5 then 0
when 2 then 1
when 3 then 2
end
That's pretty easy to generate with a bit of Ruby:
ids = [5, 2, 3]
order = 'case id ' + (0 .. ids.length).map { |i| "when #{ids[i]} then #{i}" }.join(' ') + ' end'
The above assumes that you're working with numbers or some other safe values in ids
; if that's not the case then you'd want to use connection.quote
or one of the ActiveRecord SQL sanitizer methods to properly quote your ids
.
Then use the order
string as your ordering condition:
Object.find(ids, :order => order)
or in the modern world:
Object.where(:id => ids).order(order)
This is a bit verbose but it should work the same with any SQL database and it isn't that difficult to hide the ugliness.
5
I did it withObject.order('case id ' + ids.each_with_index.map { |id, i| "when #{id} then #{i}" }.join(' ') + ' end')
– Peter Hamilton
Feb 13 '14 at 14:00
mu's wouldn't work for me, though Peter's did. +1
– dj.
Sep 20 '14 at 2:53
1
or to sql injection escape, put this in the map block:sanitize_sql_array(["when ? then ? ", id, i])
(works within an AR model)
– nruth
May 1 '16 at 22:19
1
@nruth: There was an unstated assumption thatids
contained numbers, assumptions are a bad idea (doubly so when they're not explicit) so I've hopefully cleared that up.
– mu is too short
May 2 '16 at 6:00
1
@muistooshort it's still a good answer, and generally a reasonable assumption, especially in 2013. But with javascript front-ends and uuids in fashion it seemed worth a mention, as Google points here for solving the ordering problem.
– nruth
May 3 '16 at 19:50
add a comment |
As I answered here, I just released a gem (order_as_specified) that allows you to do native SQL ordering like this:
Object.where(id: [5, 2, 3]).order_as_specified(id: [5, 2, 3])
Just tested and it works in SQLite.
add a comment |
Justin Weiss wrote a blog article about this problem just two days ago.
It seems to be a good approach to tell the database about the preferred order and load all records sorted in that order directly from the database. Example from his blog article:
# in config/initializers/find_by_ordered_ids.rb
module FindByOrderedIdsActiveRecordExtension
extend ActiveSupport::Concern
module ClassMethods
def find_ordered(ids)
order_clause = "CASE id "
ids.each_with_index do |id, index|
order_clause << "WHEN #{id} THEN #{index} "
end
order_clause << "ELSE #{ids.length} END"
where(id: ids).order(order_clause)
end
end
end
ActiveRecord::Base.include(FindByOrderedIdsActiveRecordExtension)
That allows you to write:
Object.find_ordered([2, 1, 3]) # => [2, 1, 3]
Nice drop-in solution but only applies to theid
. Would be nice to be able to dynamically determine the attribute to order by, likeuid
or something.
– Joshua Pinter
Sep 9 '15 at 19:03
@JoshPinter: Just add a parameterfield_name
to the method and use thatfield_name
instead of"CASE id"
like this:"CASE #{field_name} "
– spickermann
Sep 9 '15 at 21:12
Yep, thanks for the follow-up. Was thinking more of allowing forfind_ordered_by_uid(uids)
but it's just building off what you said. Cheers.
– Joshua Pinter
Sep 9 '15 at 22:01
1
This code breaks on empty arrays...upvoted anyways...
– alexandrecosta
Oct 6 '15 at 18:12
Recommend changing the name of the method tofind_by_ordered_ids
to be more explicit and match the name of the module file.
– Joshua Pinter
Nov 12 '15 at 16:38
|
show 5 more comments
Here's a performant (hash-lookup, not O(n) array search as in detect!) one-liner, as a method:
def find_ordered(model, ids)
model.find(ids).map{|o| [o.id, o]}.to_h.values_at(*ids)
end
# We get:
ids = [3, 3, 2, 1, 3]
Model.find(ids).map(:id) == [1, 2, 3]
find_ordered(Model, ids).map(:id) == ids
add a comment |
Another (probably more efficient) way to do it in Ruby:
ids = [5, 2, 3]
records_by_id = Model.find(ids).inject({}) do |result, record|
result[record.id] = record
result
end
sorted_records = ids.map {|id| records_by_id[id] }
you can use each_with_object to make this code two lines FWIW. also did some benchmarking and it looks like this code is much faster if you're iterating over a large set.
– Schneems
Sep 13 '11 at 19:07
Thanks for your comment. I didn't know of the existence of #each_with_object (api.rubyonrails.org/classes/…). Is there a significant difference between the above #inject and #each_with_object approach?
– Jeroen van Dijk
Sep 14 '11 at 14:19
Nope, each_with_object is essentially the same as inject except you don't have to return the object your modifying (in your caseresult
) at the end of your block. It simplifies building hashes IMHO.
– Schneems
Sep 16 '11 at 20:14
add a comment |
Here's the simplest thing I could come up with:
ids = [200, 107, 247, 189]
results = ModelObject.find(ids).group_by(&:id)
sorted_results = ids.map {|id| results[id].first }
add a comment |
@things = [5,2,3].map{|id| Object.find(id)}
This is probably the easiest way, assuming you don't have too many objects to find, since it requires a trip to the database for each id.
1
you shouldn't do this (N+1 queries)
– Schneems
Sep 13 '11 at 19:20
It's N queries, not N+1. There are much better ways to do this, but as I said, this is the easiest. For performance, look them all up, then dump them in a hash with the ids as keys.
– jcnnghm
Sep 14 '11 at 2:43
add a comment |
Your Answer
StackExchange.ifUsing("editor", function () {
StackExchange.using("externalEditor", function () {
StackExchange.using("snippets", function () {
StackExchange.snippets.init();
});
});
}, "code-snippets");
StackExchange.ready(function() {
var channelOptions = {
tags: "".split(" "),
id: "1"
};
initTagRenderer("".split(" "), "".split(" "), channelOptions);
StackExchange.using("externalEditor", function() {
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled) {
StackExchange.using("snippets", function() {
createEditor();
});
}
else {
createEditor();
}
});
function createEditor() {
StackExchange.prepareEditor({
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: true,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: 10,
bindNavPrevention: true,
postfix: "",
imageUploader: {
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
},
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
});
}
});
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f801824%2fclean-way-to-find-activerecord-objects-by-id-in-the-order-specified%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
10 Answers
10
active
oldest
votes
10 Answers
10
active
oldest
votes
active
oldest
votes
active
oldest
votes
It's not that MySQL and other DBs sort things on their own, it's that they don't sort them. When you call Model.find([5, 2, 3])
, the SQL generated is something like:
SELECT * FROM models WHERE models.id IN (5, 2, 3)
This doesn't specify an order, just the set of records you want returned. It turns out that generally MySQL will return the database rows in 'id'
order, but there's no guarantee of this.
The only way to get the database to return records in a guaranteed order is to add an order clause. If your records will always be returned in a particular order, then you can add a sort column to the db and do Model.find([5, 2, 3], :order => 'sort_column')
. If this isn't the case, you'll have to do the sorting in code:
ids = [5, 2, 3]
records = Model.find(ids)
sorted_records = ids.collect {|id| records.detect {|x| x.id == id}}
14
Another solution that is more performant than using #detect to avoid O(N) performance: records = Model.find(ids).group_by(&:id); sorted_records = ids.map { |id| records[id].first }
– Ryan LeCompte
Dec 17 '11 at 0:04
add a comment |
It's not that MySQL and other DBs sort things on their own, it's that they don't sort them. When you call Model.find([5, 2, 3])
, the SQL generated is something like:
SELECT * FROM models WHERE models.id IN (5, 2, 3)
This doesn't specify an order, just the set of records you want returned. It turns out that generally MySQL will return the database rows in 'id'
order, but there's no guarantee of this.
The only way to get the database to return records in a guaranteed order is to add an order clause. If your records will always be returned in a particular order, then you can add a sort column to the db and do Model.find([5, 2, 3], :order => 'sort_column')
. If this isn't the case, you'll have to do the sorting in code:
ids = [5, 2, 3]
records = Model.find(ids)
sorted_records = ids.collect {|id| records.detect {|x| x.id == id}}
14
Another solution that is more performant than using #detect to avoid O(N) performance: records = Model.find(ids).group_by(&:id); sorted_records = ids.map { |id| records[id].first }
– Ryan LeCompte
Dec 17 '11 at 0:04
add a comment |
It's not that MySQL and other DBs sort things on their own, it's that they don't sort them. When you call Model.find([5, 2, 3])
, the SQL generated is something like:
SELECT * FROM models WHERE models.id IN (5, 2, 3)
This doesn't specify an order, just the set of records you want returned. It turns out that generally MySQL will return the database rows in 'id'
order, but there's no guarantee of this.
The only way to get the database to return records in a guaranteed order is to add an order clause. If your records will always be returned in a particular order, then you can add a sort column to the db and do Model.find([5, 2, 3], :order => 'sort_column')
. If this isn't the case, you'll have to do the sorting in code:
ids = [5, 2, 3]
records = Model.find(ids)
sorted_records = ids.collect {|id| records.detect {|x| x.id == id}}
It's not that MySQL and other DBs sort things on their own, it's that they don't sort them. When you call Model.find([5, 2, 3])
, the SQL generated is something like:
SELECT * FROM models WHERE models.id IN (5, 2, 3)
This doesn't specify an order, just the set of records you want returned. It turns out that generally MySQL will return the database rows in 'id'
order, but there's no guarantee of this.
The only way to get the database to return records in a guaranteed order is to add an order clause. If your records will always be returned in a particular order, then you can add a sort column to the db and do Model.find([5, 2, 3], :order => 'sort_column')
. If this isn't the case, you'll have to do the sorting in code:
ids = [5, 2, 3]
records = Model.find(ids)
sorted_records = ids.collect {|id| records.detect {|x| x.id == id}}
answered Apr 30 '09 at 7:41
tomafrotomafro
5,34022221
5,34022221
14
Another solution that is more performant than using #detect to avoid O(N) performance: records = Model.find(ids).group_by(&:id); sorted_records = ids.map { |id| records[id].first }
– Ryan LeCompte
Dec 17 '11 at 0:04
add a comment |
14
Another solution that is more performant than using #detect to avoid O(N) performance: records = Model.find(ids).group_by(&:id); sorted_records = ids.map { |id| records[id].first }
– Ryan LeCompte
Dec 17 '11 at 0:04
14
14
Another solution that is more performant than using #detect to avoid O(N) performance: records = Model.find(ids).group_by(&:id); sorted_records = ids.map { |id| records[id].first }
– Ryan LeCompte
Dec 17 '11 at 0:04
Another solution that is more performant than using #detect to avoid O(N) performance: records = Model.find(ids).group_by(&:id); sorted_records = ids.map { |id| records[id].first }
– Ryan LeCompte
Dec 17 '11 at 0:04
add a comment |
Based on my previous comment to Jeroen van Dijk you can do this more efficiently and in two lines using each_with_object
result_hash = Model.find(ids).each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
For reference here is the benchmark i used
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
result_hash = results.each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
end
end.real
#=> 4.45757484436035 seconds
Now the other one
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
ids.collect {|id| results.detect {|result| result.id == id}}
end
end.real
# => 6.10875988006592
Update
You can do this in most using order and case statements, here is a class method you could use.
def self.order_by_ids(ids)
order_by = ["case"]
ids.each_with_index.map do |id, index|
order_by << "WHEN id='#{id}' THEN #{index}"
end
order_by << "end"
order(order_by.join(" "))
end
# User.where(:id => [3,2,1]).order_by_ids([3,2,1]).map(&:id)
# #=> [3,2,1]
add a comment |
Based on my previous comment to Jeroen van Dijk you can do this more efficiently and in two lines using each_with_object
result_hash = Model.find(ids).each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
For reference here is the benchmark i used
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
result_hash = results.each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
end
end.real
#=> 4.45757484436035 seconds
Now the other one
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
ids.collect {|id| results.detect {|result| result.id == id}}
end
end.real
# => 6.10875988006592
Update
You can do this in most using order and case statements, here is a class method you could use.
def self.order_by_ids(ids)
order_by = ["case"]
ids.each_with_index.map do |id, index|
order_by << "WHEN id='#{id}' THEN #{index}"
end
order_by << "end"
order(order_by.join(" "))
end
# User.where(:id => [3,2,1]).order_by_ids([3,2,1]).map(&:id)
# #=> [3,2,1]
add a comment |
Based on my previous comment to Jeroen van Dijk you can do this more efficiently and in two lines using each_with_object
result_hash = Model.find(ids).each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
For reference here is the benchmark i used
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
result_hash = results.each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
end
end.real
#=> 4.45757484436035 seconds
Now the other one
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
ids.collect {|id| results.detect {|result| result.id == id}}
end
end.real
# => 6.10875988006592
Update
You can do this in most using order and case statements, here is a class method you could use.
def self.order_by_ids(ids)
order_by = ["case"]
ids.each_with_index.map do |id, index|
order_by << "WHEN id='#{id}' THEN #{index}"
end
order_by << "end"
order(order_by.join(" "))
end
# User.where(:id => [3,2,1]).order_by_ids([3,2,1]).map(&:id)
# #=> [3,2,1]
Based on my previous comment to Jeroen van Dijk you can do this more efficiently and in two lines using each_with_object
result_hash = Model.find(ids).each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
For reference here is the benchmark i used
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
result_hash = results.each_with_object({}) {|result,result_hash| result_hash[result.id] = result }
ids.map {|id| result_hash[id]}
end
end.real
#=> 4.45757484436035 seconds
Now the other one
ids = [5,3,1,4,11,13,10]
results = Model.find(ids)
Benchmark.measure do
100000.times do
ids.collect {|id| results.detect {|result| result.id == id}}
end
end.real
# => 6.10875988006592
Update
You can do this in most using order and case statements, here is a class method you could use.
def self.order_by_ids(ids)
order_by = ["case"]
ids.each_with_index.map do |id, index|
order_by << "WHEN id='#{id}' THEN #{index}"
end
order_by << "end"
order(order_by.join(" "))
end
# User.where(:id => [3,2,1]).order_by_ids([3,2,1]).map(&:id)
# #=> [3,2,1]
edited Sep 24 '13 at 11:10
Substantial
6,30422238
6,30422238
answered Sep 13 '11 at 19:19
SchneemsSchneems
6,61754467
6,61754467
add a comment |
add a comment |
Apparently mySQL and other DB management system sort things on their own. I think that you can bypass that doing :
ids = [5,2,3]
@things = Object.find( ids, :order => "field(id,#{ids.join(',')})" )
That was the answer suggested in the link "Find by arrays of id in the same order?", but it doesn't seem to work for SQLite.
– Andrew Grimm
Apr 29 '09 at 23:30
3
And it doesn't seem to work for Postgres 9
– Jeroen van Dijk
Aug 24 '11 at 13:31
2
This no longer works. For more recent Rails:Object.where(id: ids).order("field(id, #{ids.join ','})")
– mahemoff
Apr 18 '15 at 8:37
Improved:Object.where(id: ids).order(ids.present? && "field(id, #{ids.join ','})")
. This prevents a SQL error if the IDs array is empty or nil.
– mahemoff
Apr 18 '15 at 9:09
add a comment |
Apparently mySQL and other DB management system sort things on their own. I think that you can bypass that doing :
ids = [5,2,3]
@things = Object.find( ids, :order => "field(id,#{ids.join(',')})" )
That was the answer suggested in the link "Find by arrays of id in the same order?", but it doesn't seem to work for SQLite.
– Andrew Grimm
Apr 29 '09 at 23:30
3
And it doesn't seem to work for Postgres 9
– Jeroen van Dijk
Aug 24 '11 at 13:31
2
This no longer works. For more recent Rails:Object.where(id: ids).order("field(id, #{ids.join ','})")
– mahemoff
Apr 18 '15 at 8:37
Improved:Object.where(id: ids).order(ids.present? && "field(id, #{ids.join ','})")
. This prevents a SQL error if the IDs array is empty or nil.
– mahemoff
Apr 18 '15 at 9:09
add a comment |
Apparently mySQL and other DB management system sort things on their own. I think that you can bypass that doing :
ids = [5,2,3]
@things = Object.find( ids, :order => "field(id,#{ids.join(',')})" )
Apparently mySQL and other DB management system sort things on their own. I think that you can bypass that doing :
ids = [5,2,3]
@things = Object.find( ids, :order => "field(id,#{ids.join(',')})" )
edited Sep 24 '13 at 11:10
Substantial
6,30422238
6,30422238
answered Apr 29 '09 at 16:13
marcggmarcgg
42.5k46163213
42.5k46163213
That was the answer suggested in the link "Find by arrays of id in the same order?", but it doesn't seem to work for SQLite.
– Andrew Grimm
Apr 29 '09 at 23:30
3
And it doesn't seem to work for Postgres 9
– Jeroen van Dijk
Aug 24 '11 at 13:31
2
This no longer works. For more recent Rails:Object.where(id: ids).order("field(id, #{ids.join ','})")
– mahemoff
Apr 18 '15 at 8:37
Improved:Object.where(id: ids).order(ids.present? && "field(id, #{ids.join ','})")
. This prevents a SQL error if the IDs array is empty or nil.
– mahemoff
Apr 18 '15 at 9:09
add a comment |
That was the answer suggested in the link "Find by arrays of id in the same order?", but it doesn't seem to work for SQLite.
– Andrew Grimm
Apr 29 '09 at 23:30
3
And it doesn't seem to work for Postgres 9
– Jeroen van Dijk
Aug 24 '11 at 13:31
2
This no longer works. For more recent Rails:Object.where(id: ids).order("field(id, #{ids.join ','})")
– mahemoff
Apr 18 '15 at 8:37
Improved:Object.where(id: ids).order(ids.present? && "field(id, #{ids.join ','})")
. This prevents a SQL error if the IDs array is empty or nil.
– mahemoff
Apr 18 '15 at 9:09
That was the answer suggested in the link "Find by arrays of id in the same order?", but it doesn't seem to work for SQLite.
– Andrew Grimm
Apr 29 '09 at 23:30
That was the answer suggested in the link "Find by arrays of id in the same order?", but it doesn't seem to work for SQLite.
– Andrew Grimm
Apr 29 '09 at 23:30
3
3
And it doesn't seem to work for Postgres 9
– Jeroen van Dijk
Aug 24 '11 at 13:31
And it doesn't seem to work for Postgres 9
– Jeroen van Dijk
Aug 24 '11 at 13:31
2
2
This no longer works. For more recent Rails:
Object.where(id: ids).order("field(id, #{ids.join ','})")
– mahemoff
Apr 18 '15 at 8:37
This no longer works. For more recent Rails:
Object.where(id: ids).order("field(id, #{ids.join ','})")
– mahemoff
Apr 18 '15 at 8:37
Improved:
Object.where(id: ids).order(ids.present? && "field(id, #{ids.join ','})")
. This prevents a SQL error if the IDs array is empty or nil.– mahemoff
Apr 18 '15 at 9:09
Improved:
Object.where(id: ids).order(ids.present? && "field(id, #{ids.join ','})")
. This prevents a SQL error if the IDs array is empty or nil.– mahemoff
Apr 18 '15 at 9:09
add a comment |
A portable solution would be to use an SQL CASE statement in your ORDER BY. You can use pretty much any expression in an ORDER BY and a CASE can be used as an inlined lookup table. For example, the SQL you're after would look like this:
select ...
order by
case id
when 5 then 0
when 2 then 1
when 3 then 2
end
That's pretty easy to generate with a bit of Ruby:
ids = [5, 2, 3]
order = 'case id ' + (0 .. ids.length).map { |i| "when #{ids[i]} then #{i}" }.join(' ') + ' end'
The above assumes that you're working with numbers or some other safe values in ids
; if that's not the case then you'd want to use connection.quote
or one of the ActiveRecord SQL sanitizer methods to properly quote your ids
.
Then use the order
string as your ordering condition:
Object.find(ids, :order => order)
or in the modern world:
Object.where(:id => ids).order(order)
This is a bit verbose but it should work the same with any SQL database and it isn't that difficult to hide the ugliness.
5
I did it withObject.order('case id ' + ids.each_with_index.map { |id, i| "when #{id} then #{i}" }.join(' ') + ' end')
– Peter Hamilton
Feb 13 '14 at 14:00
mu's wouldn't work for me, though Peter's did. +1
– dj.
Sep 20 '14 at 2:53
1
or to sql injection escape, put this in the map block:sanitize_sql_array(["when ? then ? ", id, i])
(works within an AR model)
– nruth
May 1 '16 at 22:19
1
@nruth: There was an unstated assumption thatids
contained numbers, assumptions are a bad idea (doubly so when they're not explicit) so I've hopefully cleared that up.
– mu is too short
May 2 '16 at 6:00
1
@muistooshort it's still a good answer, and generally a reasonable assumption, especially in 2013. But with javascript front-ends and uuids in fashion it seemed worth a mention, as Google points here for solving the ordering problem.
– nruth
May 3 '16 at 19:50
add a comment |
A portable solution would be to use an SQL CASE statement in your ORDER BY. You can use pretty much any expression in an ORDER BY and a CASE can be used as an inlined lookup table. For example, the SQL you're after would look like this:
select ...
order by
case id
when 5 then 0
when 2 then 1
when 3 then 2
end
That's pretty easy to generate with a bit of Ruby:
ids = [5, 2, 3]
order = 'case id ' + (0 .. ids.length).map { |i| "when #{ids[i]} then #{i}" }.join(' ') + ' end'
The above assumes that you're working with numbers or some other safe values in ids
; if that's not the case then you'd want to use connection.quote
or one of the ActiveRecord SQL sanitizer methods to properly quote your ids
.
Then use the order
string as your ordering condition:
Object.find(ids, :order => order)
or in the modern world:
Object.where(:id => ids).order(order)
This is a bit verbose but it should work the same with any SQL database and it isn't that difficult to hide the ugliness.
5
I did it withObject.order('case id ' + ids.each_with_index.map { |id, i| "when #{id} then #{i}" }.join(' ') + ' end')
– Peter Hamilton
Feb 13 '14 at 14:00
mu's wouldn't work for me, though Peter's did. +1
– dj.
Sep 20 '14 at 2:53
1
or to sql injection escape, put this in the map block:sanitize_sql_array(["when ? then ? ", id, i])
(works within an AR model)
– nruth
May 1 '16 at 22:19
1
@nruth: There was an unstated assumption thatids
contained numbers, assumptions are a bad idea (doubly so when they're not explicit) so I've hopefully cleared that up.
– mu is too short
May 2 '16 at 6:00
1
@muistooshort it's still a good answer, and generally a reasonable assumption, especially in 2013. But with javascript front-ends and uuids in fashion it seemed worth a mention, as Google points here for solving the ordering problem.
– nruth
May 3 '16 at 19:50
add a comment |
A portable solution would be to use an SQL CASE statement in your ORDER BY. You can use pretty much any expression in an ORDER BY and a CASE can be used as an inlined lookup table. For example, the SQL you're after would look like this:
select ...
order by
case id
when 5 then 0
when 2 then 1
when 3 then 2
end
That's pretty easy to generate with a bit of Ruby:
ids = [5, 2, 3]
order = 'case id ' + (0 .. ids.length).map { |i| "when #{ids[i]} then #{i}" }.join(' ') + ' end'
The above assumes that you're working with numbers or some other safe values in ids
; if that's not the case then you'd want to use connection.quote
or one of the ActiveRecord SQL sanitizer methods to properly quote your ids
.
Then use the order
string as your ordering condition:
Object.find(ids, :order => order)
or in the modern world:
Object.where(:id => ids).order(order)
This is a bit verbose but it should work the same with any SQL database and it isn't that difficult to hide the ugliness.
A portable solution would be to use an SQL CASE statement in your ORDER BY. You can use pretty much any expression in an ORDER BY and a CASE can be used as an inlined lookup table. For example, the SQL you're after would look like this:
select ...
order by
case id
when 5 then 0
when 2 then 1
when 3 then 2
end
That's pretty easy to generate with a bit of Ruby:
ids = [5, 2, 3]
order = 'case id ' + (0 .. ids.length).map { |i| "when #{ids[i]} then #{i}" }.join(' ') + ' end'
The above assumes that you're working with numbers or some other safe values in ids
; if that's not the case then you'd want to use connection.quote
or one of the ActiveRecord SQL sanitizer methods to properly quote your ids
.
Then use the order
string as your ordering condition:
Object.find(ids, :order => order)
or in the modern world:
Object.where(:id => ids).order(order)
This is a bit verbose but it should work the same with any SQL database and it isn't that difficult to hide the ugliness.
edited May 2 '16 at 5:59
answered Mar 22 '13 at 7:08
mu is too shortmu is too short
353k58696671
353k58696671
5
I did it withObject.order('case id ' + ids.each_with_index.map { |id, i| "when #{id} then #{i}" }.join(' ') + ' end')
– Peter Hamilton
Feb 13 '14 at 14:00
mu's wouldn't work for me, though Peter's did. +1
– dj.
Sep 20 '14 at 2:53
1
or to sql injection escape, put this in the map block:sanitize_sql_array(["when ? then ? ", id, i])
(works within an AR model)
– nruth
May 1 '16 at 22:19
1
@nruth: There was an unstated assumption thatids
contained numbers, assumptions are a bad idea (doubly so when they're not explicit) so I've hopefully cleared that up.
– mu is too short
May 2 '16 at 6:00
1
@muistooshort it's still a good answer, and generally a reasonable assumption, especially in 2013. But with javascript front-ends and uuids in fashion it seemed worth a mention, as Google points here for solving the ordering problem.
– nruth
May 3 '16 at 19:50
add a comment |
5
I did it withObject.order('case id ' + ids.each_with_index.map { |id, i| "when #{id} then #{i}" }.join(' ') + ' end')
– Peter Hamilton
Feb 13 '14 at 14:00
mu's wouldn't work for me, though Peter's did. +1
– dj.
Sep 20 '14 at 2:53
1
or to sql injection escape, put this in the map block:sanitize_sql_array(["when ? then ? ", id, i])
(works within an AR model)
– nruth
May 1 '16 at 22:19
1
@nruth: There was an unstated assumption thatids
contained numbers, assumptions are a bad idea (doubly so when they're not explicit) so I've hopefully cleared that up.
– mu is too short
May 2 '16 at 6:00
1
@muistooshort it's still a good answer, and generally a reasonable assumption, especially in 2013. But with javascript front-ends and uuids in fashion it seemed worth a mention, as Google points here for solving the ordering problem.
– nruth
May 3 '16 at 19:50
5
5
I did it with
Object.order('case id ' + ids.each_with_index.map { |id, i| "when #{id} then #{i}" }.join(' ') + ' end')
– Peter Hamilton
Feb 13 '14 at 14:00
I did it with
Object.order('case id ' + ids.each_with_index.map { |id, i| "when #{id} then #{i}" }.join(' ') + ' end')
– Peter Hamilton
Feb 13 '14 at 14:00
mu's wouldn't work for me, though Peter's did. +1
– dj.
Sep 20 '14 at 2:53
mu's wouldn't work for me, though Peter's did. +1
– dj.
Sep 20 '14 at 2:53
1
1
or to sql injection escape, put this in the map block:
sanitize_sql_array(["when ? then ? ", id, i])
(works within an AR model)– nruth
May 1 '16 at 22:19
or to sql injection escape, put this in the map block:
sanitize_sql_array(["when ? then ? ", id, i])
(works within an AR model)– nruth
May 1 '16 at 22:19
1
1
@nruth: There was an unstated assumption that
ids
contained numbers, assumptions are a bad idea (doubly so when they're not explicit) so I've hopefully cleared that up.– mu is too short
May 2 '16 at 6:00
@nruth: There was an unstated assumption that
ids
contained numbers, assumptions are a bad idea (doubly so when they're not explicit) so I've hopefully cleared that up.– mu is too short
May 2 '16 at 6:00
1
1
@muistooshort it's still a good answer, and generally a reasonable assumption, especially in 2013. But with javascript front-ends and uuids in fashion it seemed worth a mention, as Google points here for solving the ordering problem.
– nruth
May 3 '16 at 19:50
@muistooshort it's still a good answer, and generally a reasonable assumption, especially in 2013. But with javascript front-ends and uuids in fashion it seemed worth a mention, as Google points here for solving the ordering problem.
– nruth
May 3 '16 at 19:50
add a comment |
As I answered here, I just released a gem (order_as_specified) that allows you to do native SQL ordering like this:
Object.where(id: [5, 2, 3]).order_as_specified(id: [5, 2, 3])
Just tested and it works in SQLite.
add a comment |
As I answered here, I just released a gem (order_as_specified) that allows you to do native SQL ordering like this:
Object.where(id: [5, 2, 3]).order_as_specified(id: [5, 2, 3])
Just tested and it works in SQLite.
add a comment |
As I answered here, I just released a gem (order_as_specified) that allows you to do native SQL ordering like this:
Object.where(id: [5, 2, 3]).order_as_specified(id: [5, 2, 3])
Just tested and it works in SQLite.
As I answered here, I just released a gem (order_as_specified) that allows you to do native SQL ordering like this:
Object.where(id: [5, 2, 3]).order_as_specified(id: [5, 2, 3])
Just tested and it works in SQLite.
edited May 23 '17 at 10:30
Community♦
11
11
answered Mar 13 '15 at 18:30
JacobEvelynJacobEvelyn
2,47212840
2,47212840
add a comment |
add a comment |
Justin Weiss wrote a blog article about this problem just two days ago.
It seems to be a good approach to tell the database about the preferred order and load all records sorted in that order directly from the database. Example from his blog article:
# in config/initializers/find_by_ordered_ids.rb
module FindByOrderedIdsActiveRecordExtension
extend ActiveSupport::Concern
module ClassMethods
def find_ordered(ids)
order_clause = "CASE id "
ids.each_with_index do |id, index|
order_clause << "WHEN #{id} THEN #{index} "
end
order_clause << "ELSE #{ids.length} END"
where(id: ids).order(order_clause)
end
end
end
ActiveRecord::Base.include(FindByOrderedIdsActiveRecordExtension)
That allows you to write:
Object.find_ordered([2, 1, 3]) # => [2, 1, 3]
Nice drop-in solution but only applies to theid
. Would be nice to be able to dynamically determine the attribute to order by, likeuid
or something.
– Joshua Pinter
Sep 9 '15 at 19:03
@JoshPinter: Just add a parameterfield_name
to the method and use thatfield_name
instead of"CASE id"
like this:"CASE #{field_name} "
– spickermann
Sep 9 '15 at 21:12
Yep, thanks for the follow-up. Was thinking more of allowing forfind_ordered_by_uid(uids)
but it's just building off what you said. Cheers.
– Joshua Pinter
Sep 9 '15 at 22:01
1
This code breaks on empty arrays...upvoted anyways...
– alexandrecosta
Oct 6 '15 at 18:12
Recommend changing the name of the method tofind_by_ordered_ids
to be more explicit and match the name of the module file.
– Joshua Pinter
Nov 12 '15 at 16:38
|
show 5 more comments
Justin Weiss wrote a blog article about this problem just two days ago.
It seems to be a good approach to tell the database about the preferred order and load all records sorted in that order directly from the database. Example from his blog article:
# in config/initializers/find_by_ordered_ids.rb
module FindByOrderedIdsActiveRecordExtension
extend ActiveSupport::Concern
module ClassMethods
def find_ordered(ids)
order_clause = "CASE id "
ids.each_with_index do |id, index|
order_clause << "WHEN #{id} THEN #{index} "
end
order_clause << "ELSE #{ids.length} END"
where(id: ids).order(order_clause)
end
end
end
ActiveRecord::Base.include(FindByOrderedIdsActiveRecordExtension)
That allows you to write:
Object.find_ordered([2, 1, 3]) # => [2, 1, 3]
Nice drop-in solution but only applies to theid
. Would be nice to be able to dynamically determine the attribute to order by, likeuid
or something.
– Joshua Pinter
Sep 9 '15 at 19:03
@JoshPinter: Just add a parameterfield_name
to the method and use thatfield_name
instead of"CASE id"
like this:"CASE #{field_name} "
– spickermann
Sep 9 '15 at 21:12
Yep, thanks for the follow-up. Was thinking more of allowing forfind_ordered_by_uid(uids)
but it's just building off what you said. Cheers.
– Joshua Pinter
Sep 9 '15 at 22:01
1
This code breaks on empty arrays...upvoted anyways...
– alexandrecosta
Oct 6 '15 at 18:12
Recommend changing the name of the method tofind_by_ordered_ids
to be more explicit and match the name of the module file.
– Joshua Pinter
Nov 12 '15 at 16:38
|
show 5 more comments
Justin Weiss wrote a blog article about this problem just two days ago.
It seems to be a good approach to tell the database about the preferred order and load all records sorted in that order directly from the database. Example from his blog article:
# in config/initializers/find_by_ordered_ids.rb
module FindByOrderedIdsActiveRecordExtension
extend ActiveSupport::Concern
module ClassMethods
def find_ordered(ids)
order_clause = "CASE id "
ids.each_with_index do |id, index|
order_clause << "WHEN #{id} THEN #{index} "
end
order_clause << "ELSE #{ids.length} END"
where(id: ids).order(order_clause)
end
end
end
ActiveRecord::Base.include(FindByOrderedIdsActiveRecordExtension)
That allows you to write:
Object.find_ordered([2, 1, 3]) # => [2, 1, 3]
Justin Weiss wrote a blog article about this problem just two days ago.
It seems to be a good approach to tell the database about the preferred order and load all records sorted in that order directly from the database. Example from his blog article:
# in config/initializers/find_by_ordered_ids.rb
module FindByOrderedIdsActiveRecordExtension
extend ActiveSupport::Concern
module ClassMethods
def find_ordered(ids)
order_clause = "CASE id "
ids.each_with_index do |id, index|
order_clause << "WHEN #{id} THEN #{index} "
end
order_clause << "ELSE #{ids.length} END"
where(id: ids).order(order_clause)
end
end
end
ActiveRecord::Base.include(FindByOrderedIdsActiveRecordExtension)
That allows you to write:
Object.find_ordered([2, 1, 3]) # => [2, 1, 3]
edited Apr 22 '15 at 2:17
answered Apr 22 '15 at 0:57
spickermannspickermann
60.5k75779
60.5k75779
Nice drop-in solution but only applies to theid
. Would be nice to be able to dynamically determine the attribute to order by, likeuid
or something.
– Joshua Pinter
Sep 9 '15 at 19:03
@JoshPinter: Just add a parameterfield_name
to the method and use thatfield_name
instead of"CASE id"
like this:"CASE #{field_name} "
– spickermann
Sep 9 '15 at 21:12
Yep, thanks for the follow-up. Was thinking more of allowing forfind_ordered_by_uid(uids)
but it's just building off what you said. Cheers.
– Joshua Pinter
Sep 9 '15 at 22:01
1
This code breaks on empty arrays...upvoted anyways...
– alexandrecosta
Oct 6 '15 at 18:12
Recommend changing the name of the method tofind_by_ordered_ids
to be more explicit and match the name of the module file.
– Joshua Pinter
Nov 12 '15 at 16:38
|
show 5 more comments
Nice drop-in solution but only applies to theid
. Would be nice to be able to dynamically determine the attribute to order by, likeuid
or something.
– Joshua Pinter
Sep 9 '15 at 19:03
@JoshPinter: Just add a parameterfield_name
to the method and use thatfield_name
instead of"CASE id"
like this:"CASE #{field_name} "
– spickermann
Sep 9 '15 at 21:12
Yep, thanks for the follow-up. Was thinking more of allowing forfind_ordered_by_uid(uids)
but it's just building off what you said. Cheers.
– Joshua Pinter
Sep 9 '15 at 22:01
1
This code breaks on empty arrays...upvoted anyways...
– alexandrecosta
Oct 6 '15 at 18:12
Recommend changing the name of the method tofind_by_ordered_ids
to be more explicit and match the name of the module file.
– Joshua Pinter
Nov 12 '15 at 16:38
Nice drop-in solution but only applies to the
id
. Would be nice to be able to dynamically determine the attribute to order by, like uid
or something.– Joshua Pinter
Sep 9 '15 at 19:03
Nice drop-in solution but only applies to the
id
. Would be nice to be able to dynamically determine the attribute to order by, like uid
or something.– Joshua Pinter
Sep 9 '15 at 19:03
@JoshPinter: Just add a parameter
field_name
to the method and use that field_name
instead of "CASE id"
like this: "CASE #{field_name} "
– spickermann
Sep 9 '15 at 21:12
@JoshPinter: Just add a parameter
field_name
to the method and use that field_name
instead of "CASE id"
like this: "CASE #{field_name} "
– spickermann
Sep 9 '15 at 21:12
Yep, thanks for the follow-up. Was thinking more of allowing for
find_ordered_by_uid(uids)
but it's just building off what you said. Cheers.– Joshua Pinter
Sep 9 '15 at 22:01
Yep, thanks for the follow-up. Was thinking more of allowing for
find_ordered_by_uid(uids)
but it's just building off what you said. Cheers.– Joshua Pinter
Sep 9 '15 at 22:01
1
1
This code breaks on empty arrays...upvoted anyways...
– alexandrecosta
Oct 6 '15 at 18:12
This code breaks on empty arrays...upvoted anyways...
– alexandrecosta
Oct 6 '15 at 18:12
Recommend changing the name of the method to
find_by_ordered_ids
to be more explicit and match the name of the module file.– Joshua Pinter
Nov 12 '15 at 16:38
Recommend changing the name of the method to
find_by_ordered_ids
to be more explicit and match the name of the module file.– Joshua Pinter
Nov 12 '15 at 16:38
|
show 5 more comments
Here's a performant (hash-lookup, not O(n) array search as in detect!) one-liner, as a method:
def find_ordered(model, ids)
model.find(ids).map{|o| [o.id, o]}.to_h.values_at(*ids)
end
# We get:
ids = [3, 3, 2, 1, 3]
Model.find(ids).map(:id) == [1, 2, 3]
find_ordered(Model, ids).map(:id) == ids
add a comment |
Here's a performant (hash-lookup, not O(n) array search as in detect!) one-liner, as a method:
def find_ordered(model, ids)
model.find(ids).map{|o| [o.id, o]}.to_h.values_at(*ids)
end
# We get:
ids = [3, 3, 2, 1, 3]
Model.find(ids).map(:id) == [1, 2, 3]
find_ordered(Model, ids).map(:id) == ids
add a comment |
Here's a performant (hash-lookup, not O(n) array search as in detect!) one-liner, as a method:
def find_ordered(model, ids)
model.find(ids).map{|o| [o.id, o]}.to_h.values_at(*ids)
end
# We get:
ids = [3, 3, 2, 1, 3]
Model.find(ids).map(:id) == [1, 2, 3]
find_ordered(Model, ids).map(:id) == ids
Here's a performant (hash-lookup, not O(n) array search as in detect!) one-liner, as a method:
def find_ordered(model, ids)
model.find(ids).map{|o| [o.id, o]}.to_h.values_at(*ids)
end
# We get:
ids = [3, 3, 2, 1, 3]
Model.find(ids).map(:id) == [1, 2, 3]
find_ordered(Model, ids).map(:id) == ids
answered Oct 29 '15 at 20:40
Kanat BolazarKanat Bolazar
31839
31839
add a comment |
add a comment |
Another (probably more efficient) way to do it in Ruby:
ids = [5, 2, 3]
records_by_id = Model.find(ids).inject({}) do |result, record|
result[record.id] = record
result
end
sorted_records = ids.map {|id| records_by_id[id] }
you can use each_with_object to make this code two lines FWIW. also did some benchmarking and it looks like this code is much faster if you're iterating over a large set.
– Schneems
Sep 13 '11 at 19:07
Thanks for your comment. I didn't know of the existence of #each_with_object (api.rubyonrails.org/classes/…). Is there a significant difference between the above #inject and #each_with_object approach?
– Jeroen van Dijk
Sep 14 '11 at 14:19
Nope, each_with_object is essentially the same as inject except you don't have to return the object your modifying (in your caseresult
) at the end of your block. It simplifies building hashes IMHO.
– Schneems
Sep 16 '11 at 20:14
add a comment |
Another (probably more efficient) way to do it in Ruby:
ids = [5, 2, 3]
records_by_id = Model.find(ids).inject({}) do |result, record|
result[record.id] = record
result
end
sorted_records = ids.map {|id| records_by_id[id] }
you can use each_with_object to make this code two lines FWIW. also did some benchmarking and it looks like this code is much faster if you're iterating over a large set.
– Schneems
Sep 13 '11 at 19:07
Thanks for your comment. I didn't know of the existence of #each_with_object (api.rubyonrails.org/classes/…). Is there a significant difference between the above #inject and #each_with_object approach?
– Jeroen van Dijk
Sep 14 '11 at 14:19
Nope, each_with_object is essentially the same as inject except you don't have to return the object your modifying (in your caseresult
) at the end of your block. It simplifies building hashes IMHO.
– Schneems
Sep 16 '11 at 20:14
add a comment |
Another (probably more efficient) way to do it in Ruby:
ids = [5, 2, 3]
records_by_id = Model.find(ids).inject({}) do |result, record|
result[record.id] = record
result
end
sorted_records = ids.map {|id| records_by_id[id] }
Another (probably more efficient) way to do it in Ruby:
ids = [5, 2, 3]
records_by_id = Model.find(ids).inject({}) do |result, record|
result[record.id] = record
result
end
sorted_records = ids.map {|id| records_by_id[id] }
answered Aug 24 '11 at 11:00
Jeroen van DijkJeroen van Dijk
885913
885913
you can use each_with_object to make this code two lines FWIW. also did some benchmarking and it looks like this code is much faster if you're iterating over a large set.
– Schneems
Sep 13 '11 at 19:07
Thanks for your comment. I didn't know of the existence of #each_with_object (api.rubyonrails.org/classes/…). Is there a significant difference between the above #inject and #each_with_object approach?
– Jeroen van Dijk
Sep 14 '11 at 14:19
Nope, each_with_object is essentially the same as inject except you don't have to return the object your modifying (in your caseresult
) at the end of your block. It simplifies building hashes IMHO.
– Schneems
Sep 16 '11 at 20:14
add a comment |
you can use each_with_object to make this code two lines FWIW. also did some benchmarking and it looks like this code is much faster if you're iterating over a large set.
– Schneems
Sep 13 '11 at 19:07
Thanks for your comment. I didn't know of the existence of #each_with_object (api.rubyonrails.org/classes/…). Is there a significant difference between the above #inject and #each_with_object approach?
– Jeroen van Dijk
Sep 14 '11 at 14:19
Nope, each_with_object is essentially the same as inject except you don't have to return the object your modifying (in your caseresult
) at the end of your block. It simplifies building hashes IMHO.
– Schneems
Sep 16 '11 at 20:14
you can use each_with_object to make this code two lines FWIW. also did some benchmarking and it looks like this code is much faster if you're iterating over a large set.
– Schneems
Sep 13 '11 at 19:07
you can use each_with_object to make this code two lines FWIW. also did some benchmarking and it looks like this code is much faster if you're iterating over a large set.
– Schneems
Sep 13 '11 at 19:07
Thanks for your comment. I didn't know of the existence of #each_with_object (api.rubyonrails.org/classes/…). Is there a significant difference between the above #inject and #each_with_object approach?
– Jeroen van Dijk
Sep 14 '11 at 14:19
Thanks for your comment. I didn't know of the existence of #each_with_object (api.rubyonrails.org/classes/…). Is there a significant difference between the above #inject and #each_with_object approach?
– Jeroen van Dijk
Sep 14 '11 at 14:19
Nope, each_with_object is essentially the same as inject except you don't have to return the object your modifying (in your case
result
) at the end of your block. It simplifies building hashes IMHO.– Schneems
Sep 16 '11 at 20:14
Nope, each_with_object is essentially the same as inject except you don't have to return the object your modifying (in your case
result
) at the end of your block. It simplifies building hashes IMHO.– Schneems
Sep 16 '11 at 20:14
add a comment |
Here's the simplest thing I could come up with:
ids = [200, 107, 247, 189]
results = ModelObject.find(ids).group_by(&:id)
sorted_results = ids.map {|id| results[id].first }
add a comment |
Here's the simplest thing I could come up with:
ids = [200, 107, 247, 189]
results = ModelObject.find(ids).group_by(&:id)
sorted_results = ids.map {|id| results[id].first }
add a comment |
Here's the simplest thing I could come up with:
ids = [200, 107, 247, 189]
results = ModelObject.find(ids).group_by(&:id)
sorted_results = ids.map {|id| results[id].first }
Here's the simplest thing I could come up with:
ids = [200, 107, 247, 189]
results = ModelObject.find(ids).group_by(&:id)
sorted_results = ids.map {|id| results[id].first }
answered Jun 4 '12 at 21:08
jasongarberjasongarber
9481112
9481112
add a comment |
add a comment |
@things = [5,2,3].map{|id| Object.find(id)}
This is probably the easiest way, assuming you don't have too many objects to find, since it requires a trip to the database for each id.
1
you shouldn't do this (N+1 queries)
– Schneems
Sep 13 '11 at 19:20
It's N queries, not N+1. There are much better ways to do this, but as I said, this is the easiest. For performance, look them all up, then dump them in a hash with the ids as keys.
– jcnnghm
Sep 14 '11 at 2:43
add a comment |
@things = [5,2,3].map{|id| Object.find(id)}
This is probably the easiest way, assuming you don't have too many objects to find, since it requires a trip to the database for each id.
1
you shouldn't do this (N+1 queries)
– Schneems
Sep 13 '11 at 19:20
It's N queries, not N+1. There are much better ways to do this, but as I said, this is the easiest. For performance, look them all up, then dump them in a hash with the ids as keys.
– jcnnghm
Sep 14 '11 at 2:43
add a comment |
@things = [5,2,3].map{|id| Object.find(id)}
This is probably the easiest way, assuming you don't have too many objects to find, since it requires a trip to the database for each id.
@things = [5,2,3].map{|id| Object.find(id)}
This is probably the easiest way, assuming you don't have too many objects to find, since it requires a trip to the database for each id.
answered Apr 29 '09 at 16:44
jcnnghmjcnnghm
4,04662638
4,04662638
1
you shouldn't do this (N+1 queries)
– Schneems
Sep 13 '11 at 19:20
It's N queries, not N+1. There are much better ways to do this, but as I said, this is the easiest. For performance, look them all up, then dump them in a hash with the ids as keys.
– jcnnghm
Sep 14 '11 at 2:43
add a comment |
1
you shouldn't do this (N+1 queries)
– Schneems
Sep 13 '11 at 19:20
It's N queries, not N+1. There are much better ways to do this, but as I said, this is the easiest. For performance, look them all up, then dump them in a hash with the ids as keys.
– jcnnghm
Sep 14 '11 at 2:43
1
1
you shouldn't do this (N+1 queries)
– Schneems
Sep 13 '11 at 19:20
you shouldn't do this (N+1 queries)
– Schneems
Sep 13 '11 at 19:20
It's N queries, not N+1. There are much better ways to do this, but as I said, this is the easiest. For performance, look them all up, then dump them in a hash with the ids as keys.
– jcnnghm
Sep 14 '11 at 2:43
It's N queries, not N+1. There are much better ways to do this, but as I said, this is the easiest. For performance, look them all up, then dump them in a hash with the ids as keys.
– jcnnghm
Sep 14 '11 at 2:43
add a comment |
Thanks for contributing an answer to Stack Overflow!
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
To learn more, see our tips on writing great answers.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f801824%2fclean-way-to-find-activerecord-objects-by-id-in-the-order-specified%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
1
Where are those id's coming from? If it is the UI (via a user selecting them) then scaling shouldn't be an issue, that is the user is unlikely to spend time selecting 1000's of ids). If it's the database (e.g. from a join table), could you store the order in the join table and issue the find based on that?
– pauliephonic
Apr 29 '09 at 11:13
It looks like this is no longer true in Rails 5.
– XML Slayer
Jun 6 '18 at 19:49