У меня есть модель пользователя с атрибутами "первая" и "последняя", Так, например, User.first.first # = > "Charlie" User.first.last # = > "Браун"
Эта модель пользователя также имеет виртуальный атрибут 'full_name'
#user.rb
def full_name
[first,last].join(' ')
end
def full_name=(name) #don't know what to do with people w/ middle names
split = name.split(' ')
self.first = split[0]
self.last = split[1]
end
Итак, например:
User.first.full_name = "Charlie Brown" #=> "Charlie Brown"
User.first.full_name = "Homer Simpson" #=> "Home Simpson"
User.first.save
User.first.first #=> "Homer"
User.first.last #=> "Simpson"
Было бы так приятно, если бы я мог искать по этому виртуальному атрибуту например, для динамического поиска:
User.find_by_full_name('Home Simpson') # this doesn't work
Пример условий в поиске:
User.all(:conditions => ['full_name LIKE ?', query]) #this doesn't work
Я надеюсь найти по крайней мере некоторые способы на языке SQL, которые могут это сделать; если есть динамический виртуальный атрибут, нахожу тоже, что дополнительный ванильный источник на штруделе. (кто имеет эту зиму?)
Я также беспокоился о поиске имени, например, "Холмс" можно искать только в "первом" столбце, но не "последнем" для получения, например, User.first.full_name #=> "Sherlock Holmes"
.
Я попытался сделать более полный поиск:
user.rb
def self.find_by_full_name(name) #returns an array of User model
return all if name.blank?
split = name.split(' ', 2)
output = []
if split.length > 1
with_scope( :find => { :conditions => ['first LIKE ?', "%#{split[0]}%"] }) do
output << all(:conditions => ['last LIKE ?', "%#{split[1]}%"])
output.flatten!
end
elsif split.length == 1
output << all(:conditions => ['first LIKE ?', "%#{split[0]}%"])
output << all(:conditions => ['last LIKE ?', "%#{split[0]}%"])
output.flatten!
end
end
Например
User.find_by_full_name("John").map(&:full_name) #=> ["John Resig", "John Doe"]
User.find_by_full_name("Doe").map(&:full_name) #=> ["John Doe", "Philips Doeringer"]
User.find_by_full_name("John Doe").map(&:full_name) #=> ["John Doe"]
Но я просто подумал, что метод find_by_full_name здесь немного громоздкий.
Я имею в виду, если бы у меня был столбец full_name, который каждый раз настраивается фильтром after save с конкат первым и последним. Поэтому найти имя человека, особенно с нечеткой памятью этого человека, полезно. Поэтому, если я помню "Doe" в этом человеке как с первой, так и с фамилией, я всегда могу сделать простое User.find_by_full_name ( "Doe" ), чтобы как можно больше скопировать его.
И поскольку это столбец, я могу найти его в условии find (: conditions [...]), если мне нужно сделать что-то вроде Project.find(:all,:include => :users, :conditions=>['users.full_name LIKE ?', query])
где
#project.rb
has_many :assignments
has_many :users, :through=>:assignments
#user.rb
has_many :assignments
has_many :projects, :through => :assignments
#assignment.rb
belongs_to :user
belongs_to :project
Счастливые праздники N