Rails虚拟属性搜索或sql组合列搜索
-
05-07-2019 - |
题
我有一个属性'first'和'last'的用户模型 所以举个例子 User.first.first#=> "查理" 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语言中至少找到一些可以做到这一点的方法;如果还有一个动态虚拟属性查找,那就是在结构上的额外香草源。 (今年冬天有这个人吗?)
我还担心被搜查的名字,例如“Holmes”。只能在“第一”列中搜索,而不能在“最后”列中搜索,例如 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,每次都由一个后保存过滤器设置,其中包含first和last的concat。因此,找到一个人的名字,特别是对这个人的模糊记忆,是有帮助的。因此,如果我记得那个人的名字中的'Doe',无论是名字还是姓,我总是可以做一个简单的User.find_by_full_name('Doe')来尽可能多地返回以进一步确定它。
因为它是一个列,所以如果我必须做 Project.find(:all,:include =&gt;: users,:conditions =&gt; ['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(' ')}
}
}
当然,任何更干净的方式都是受欢迎的。