[LRUG] association finder query problem

Matthew Rudy Jacobs matthewrudyjacobs at gmail.com
Sat May 24 16:01:26 PDT 2008


That's fine
(although I have a feeling it'll break in 2.1, with the changes to
eager-loading, if Freddy is right)

But, I just feel like we're cheating using :include, when our goal is not
actually to :include.
And, it'll have us ending up with a "Monitor" with a random set of
"features" eager loaded
(all those that are "big", I think?)

Perhaps the best way, if we don't mind doing multiple queries is;

big_ids    = Feature.find(:all, :select => "DISTINCT monitor_id",
:conditions => {:name => "big"}).map(&:monitor_id)
white_ids = Feature.find(:all, :select => "DISTINCT monitor_id", :conditions
=> {:name => "white"}).map(&:monitor_id)

Monitor.find(big_ids & white_ids, :include => :features)

Although,
I don't really understand why the models are set up like this.

Feels like,

Monitor
  MonitorFeature
Feature

2008/5/24 Ben Griffiths <bengriffiths at gmail.com>:

> I don't know your application - particularly whether that function is
> frequently called, nor the size and cardinality of your database, -
> but it might be perfectly reasonable to just use the array
> intersection operator (&). Something along the lines of:
>
> monitors = Monitor.find(:all, :include=>:features :conditions => "name
> = 'big'") & Monitor.find(:all, :include=>:features :conditions =>
> "name = 'white'")
>
> The two queries and unnecessary marshaling, and so on may not be an
> issue (only you will know) - and it's much more readable and
> intentional than jumping through sql loops.
>
> Actually, having said all that, I don't know if this works with rails
> object arrays, should do though? Maybe you could try it and let me
> know!
>
> Ben
>
> On 5/24/08, Matthew Rudy Jacobs <matthewrudyjacobs at gmail.com> wrote:
> > 2008/5/24 Xin Zheng <xin at zenpow.com>:
> >> I want to find a Monitor that is BOTH white and big.
> >>
> >>My first shot at this:
> >>find(:all, :include => :features, :conditions => ["name IN ('white',
> >>'big')" )
> >>
> >>But this query will find Monitors that's EITHER white or big.
> >
> >
> > Yeah,
> > the short answer is;
> > "that's exactly what you've asked it to do"
> >
> > ===========
> > Your db
> > ===========
> > - Monitor#1
> > -- Feature#1 = big
> > -- Feature#2 = white
> > -- Feature#3 = small
> >
> > - Monitor#2
> > -- Feature#4 = big
> >
> > - Monitor#3
> > -- Feature#5 = white
> >
> > - Monitor#4
> > -- *no features*
> > ===========
> >
> > Doing an include gets you a left outer join on "features"
> > which creates a new line for each feature.
> > ===========
> > - Monitor#1 & Feature#1(big)
> > - Monitor#1 & Feature#2(white)
> > - Monitor#1 & Feature#3(small)
> > - Monitor#2 & Feature#4(big)
> > - Monitor#3 & Feature#5(white)
> > - Monitor#4 & *no feature*
> > ===========
> >
> > Then you say
> > "name IN (white, big)"
> > ===========
> > this grabs all those lines which satisfy your requirement
> > ===========
> > - Monitor#1 & Feature#1(big)
> > - Monitor#1 & Feature#2(white)
> > - Monitor#2 & Feature#4(big)
> > - Monitor#3 & Feature#5(white)
> > ===========
> >
> > Hence,
> > you don't get what you want.
> >
> > And,
> > you don't actually intend to ":include" these features anyway, (if you
> do,
> > then my solution is wrong)
> > you're just using the :include, because it does the join automatically.
> >
> > The right solution is debatable;
> > can do it with subqueries, or a double "inner join"
> >
> > but I think subqueries are the right way,
> > they look dirty, but it's exactly what you're asking for.
> >
> > Hopefully MySQL will optimise it correctly.
> >
> > condition_bit = "EXISTS(SELECT * FROM features WHERE monitor_id =
> > monitors.id & features.name = ?)"
> > condition_string = ([condition_bit]*2).join(" AND ")
> > conditions = [condition_string, "big", "white"]
> > Monitor.find(:all, :conditions => conditions)
> >
> > my friend Freddy Cheung wrote a nice blog post about similar stuff:
> > http://www.spacevatican.org/2008/4/29/include-and-conditions
> >
> _______________________________________________
> Chat mailing list
> Chat at lists.lrug.org
> http://lists.lrug.org/listinfo.cgi/chat-lrug.org
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.lrug.org/pipermail/chat-lrug.org/attachments/20080525/b3a1000d/attachment.html>


More information about the Chat mailing list