我有一个具有’first’和’last’属性的用户模型,因此,例如User.first.first#=>“ Charlie” User.first.last#=>“ Brown”
该用户模型还具有虚拟属性“ 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语言的方法可以做到这一点。如果还找到动态的虚拟属性,那是果馅卷上额外的香草来源。(这个冬天有人吗?)
我还担心要搜索的名称,例如,“ Holmes”只能在“第一”列中搜索,而不能在“最后”列中搜索,例如User.first.full_name #=> "Sherlock Holmes"。
User.first.full_name #=> "Sherlock Holmes"
我确实尝试进行更全面的搜索:
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列,则每次通过一次保存后过滤器(带有first和last的连接)进行设置。因此,找到一个人的名字,特别是对这个人的记忆模糊,会很有帮助。因此,如果我记得该人的名字或姓氏中的“ Doe”,那么我总是可以做一个简单的User.find_by_full_name(’Doe’)以尽可能多地返回以进一步固定它。
而且由于它是一列,因此如果需要执行类似Project.find(:all,:include => :users, :conditions=>['users.full_name LIKE ?', query]) where的操作,则可以在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
您可以在user.rb中使用named_scope:
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')
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]}}