Поиск виртуальных атрибутов Rails или комбинированный поиск по столбцам sql

StackOverflow https://stackoverflow.com/questions/1806394

Вопрос

У меня есть пользовательская модель с атрибутами 'first' и 'last' Так например User.first.first # = > & Quot; Charlie & Quot; User.first.last # = > & Quot; Коричневый & Quot;

Эта модель пользователя также имеет виртуальный атрибут 'полное_имя'

#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

Пример условий в find:

User.all(:conditions => ['full_name LIKE ?', query]) #this doesn't work

Я надеюсь найти хотя бы несколько путей в языке SQL, которые могут это сделать; если есть динамический виртуальный атрибут find, то это лишний ванильный источник на штруделе. (у кого есть эта зима?)

Я также был обеспокоен тем, что имя ищется, например, "Холмс" можно искать только в «первом» столбце, но не в «последнем» для извлечения, например, User.first.full_name # = > "Шерлок Холмс" .

Я попытался сделать более полный поиск:

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, который устанавливается каждый раз фильтром после сохранения с concat first и last. Так что поиск имени человека, особенно с нечеткой памятью об этом человеке, полезен. Поэтому, если я запомнил «Доу» по имени или фамилии этого человека, я всегда могу сделать простое User.find_by_full_name («Доу»), чтобы вернуть как можно больше, чтобы закрепить его.

И так как это столбец, я могу искать его в предложении find (: условие [...]), если мне нужно сделать что-то вроде Project.find (: all,: include = >: пользователи,: условие = > ['users.full_name LIKE?', запрос]) где

#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

Это было полезно?

Решение

Вы можете использовать named_scope в вашем user.rb:

named_scope :find_by_full_name, lambda {|full_name| 
  {:conditions => {:first => full_name.split(' ').first, 
     :last => full_name.split(' ').last}}
}

Тогда вы можете сделать User.find_by_full_name ('John Carver')

новый материал в ответ на изменения в требовании

named_scope :find_by_full_name, lambda {|full_name| 
  {:conditions => ["first LIKE '%?%' or last LIKE '%?%'", 
    full_name.split(' ').first, full_name.split(' ').last]}}

Другие советы

Я также нашел ответ Джима полезным. Благодарю. Я бы сделал небольшое изменение, хотя. Этот текущий код приводит к потере любых отчеств. То, что я имею ниже, является немного грязным, но сохраняет отчества и сложные фамилии (думаю, Жан-Клод ван Дамм). Все после имени идет в поле last_name.

named_scope :find_by_full_name, lambda { |full_name| 
 {:conditions => {:first_name => full_name.split(' ').first, 
   :last_name => full_name.split(' ')[1, full_name.split(' ').length-1].join(' ')}
   }
}

Конечно, любой более чистый способ сделать это приветствуется.

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top