Witaj na blogu prowadzonym przez Sebana. Spisuję tutaj swoje uwagi na różny temat. Przeważają tematy związane z Internetem, popieranymi przeze mnie rozwiązaniami dotyczącymi wykorzystania komputerów, oraz kilka innych.

ActiveRecord find i join

12 stycznia 2008 | Klucze: programowanie, rails, ruby, Techblog
7 komentarzy. trackback

Dziś chciałem znowu ponarzekać na ActiveRecord. Słówko 'narzekać' jest tu zastosowane na wyrost, więc proszę nie robić ze mnie zaraz wiecznego malkontenta. Problem, który dziś chcę tu opisać zauważyłem już wcześniej, ale potem zapomniałem o nim i nie czułem chęci i potrzeby opisywania go.

Wyobraź sobie drogi czytelniku, że masz takie zadanie: ,,poszukaj wszystkie projekty użytkownika o loginie 'quentin'. Modele wyglądają tak: Można rzucić zapytaniem z JOIN Project.find(:all, :joins => "JOIN users ON users.id = projects.user_id AND users.login LIKE 'quentin'") i znaleźć ... Ale co właściwie znajdzie takie zapytanie? Obiekty klasy Project? Tak. Dla uproszczenia załóżmy, że powyższy Project.find zapakowałem do metody Projects.find_by_sql. Project.find_by_login('quentin').map{ |p| p.id }
=> [1, 1]
Jeśli przez głowę przeszła Ci myśli, że to ten sam projekt znaleziony jakimś cudem dwa razy to nic mylnego. To dwa różne projekty wyciągnięte z bazy danych! Podanie :joins => w parametrach metody find tworzy hybrydę kolumn z tabel projects i users. Zapytanie SQL w wynikach zwróci dwa razy kolumnę id, a ActiveRecord zmapuje sobie wartość, którą napotka jako ostatnią czyli id użytkownika. Myśląc logicznie to AR zachowuje się dobrze SQL zwraca połączone talibce, AR je mapuje na swój użytek, robi sieczkę na kolumnach, których nazwy się powtarzają i zwraca jako obiekt szukanej klasy. Ta ostatnia część podoba mi się najmniej. Warto też zwrócić uwagę, że obiekty powstałe w wyniku takiego wyszukiwania mają ustawioną opcje readonly!

Innym rozwiązaniem jest wyszukanie użytkownika o pożądanym loginie i iterowanie po wynikach zwracając należące do niego projekty.

Bardziej jest eleganckie użycie w zapytaniu LEFT JOIN Project.find(:all, :include => :user, :conditions => ["users.login LIKE ?", 'quntin']).map{ |p| p.id } [1, 3] => Konia z rzędem temu, kto po wywołaniu czegoś takiego zerknie w logi railsowe i zaraz będzie wiedział co to zapytanie robi. W takim wywołaniu metoda find rzuca sobie dość długi SQL i zwraca obiekt Project wraz z użytkownikiem. Ale zwrócony obiekt nie jest już zbitką atrybutów z obu klas ale faktycznie czystym obiektem Project.

Nie mam pretensji o takie zachowanie ActiveRecorda. Nie mam doświadczenia z innymi ORM takimi jak Sequel czy DataMaper jestem ciekaw czy one zachowują się tak samo w takich zapytaniach.


KOMENTARZE

12 stycznia 2008 | neaf |

Hmmm… w django to to się robi po prostu tak: User.projects.all() .

12 stycznia 2008 | neaf |

Źle…
User.objects.filter(name=‘quntin’).projects.all()

I dodaj sobie ‘edytuj’ komentarzy do szablonu. :)

24 stycznia 2008 | Tomash |

User.find_by_login(‘quentin’).projects

Polecam dokładne czytanie dokumentacji. ActiveRecord już nie takie potrzeby ma ładnie przekryte swoją funkcjonalnością :)

24 stycznia 2008 | Sebastian Nowak |

Tak Tomashu wiem o tym. Tylko, że takie coś nie podobało się koledze z zespołu i mówił, że trzeba łączyć, łączyć, łączyć, a najlepiej właśnie :joins.

24 stycznia 2008 | Tomash |

Kolega z zespołu zna się na railsach jak świnia na gwiazdach, wnioskuję ;)

Skoro „wie lepiej”, dlaczego od razu nie każe pisać find_by_sql, zamiast korzystać z ARowych udogodnień?

24 stycznia 2008 | Sebastian Nowak |

Co do pierwszego zdania. W dużej mierze tak.

24 stycznia 2008 | Tomash |

Błogosławieni ubodzy duchem :D