Rails虚拟属性搜索或sql组合列搜索

我有一个属性’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”只能在“first”列中搜索,而不能在“last”中搜索,例如, 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 < ['last LIKE ?', "%#{split[1]}%"]) output.flatten! end elsif split.length == 1 output < ['first LIKE ?', "%#{split[0]}%"]) output < ['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,每次都由一个后保存filter设置,其中包含first和last的concat。 因此,找到一个人的名字,特别是对这个人的模糊记忆,是有帮助的。 因此,如果我记得那个人的名字中的’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 

节日快乐

您可以在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')

新的东西,以响应需求的变化

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

我发现吉姆的回答也很有帮助。 谢谢。 我做了一点改变。 此当前代码会导致您丢失任何中间名称。 我下面的内容有点乱,但保留了中间名和复合姓(想想Jean-Claude van Damme)。 第一个名称后面的所有内容都放在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(' ')} } } 

当然,任何更清洁的方式都是受欢迎的。