我正在生成一个查询,该查询返回一组满足下面发布的所有条件的配置文件
我在Ruby中收到类似的数据,然后基于内容进行动态MySQL查询 -
[{ attribute_id: 58, parent_profile_name: 'Douglas-Connelly' },
{ attribute_id: 26, parent_profile_name: 'Brekke LLC' },
{ attribute_id: 35, val: 'Asa' },
{ attribute_id: 38, val: 'Stanton' }]
这些是数据库的当前内容
profile_attribute_values
profile_id attribute_id parent_profile_id val
6 58 2
6 26 5
6 35 nil 'Asa'
6 38 nil 'Stanton'
profile
id name
2 Douglas-Connelly
5 Brekke LLC
6 nil
我需要返回符合所有条件的所有配置文件 - 与 profile_attribute_values 有关系的配置文件,其中attribute_id为x,val为y,AND其中attribute_id是x,parent_profile名称= y
我目前有什么
SELECT * FROM
(SELECT P2. *
FROM profile_attribute_values PAV
INNER JOIN profiles P1 ON P1.id = PAV.parent_profile_id
INNER JOIN profiles P2 ON P2.id = PAV.profile_id
WHERE (PAV.ne_attribute_id = '58' AND P1.`name` = 'Douglas-Connelly')
) A,
(SELECT P1. *
FROM profiles P1
INNER JOIN profile_attribute_values PAV ON P1.id = PAV.profile_id
INNER JOIN profile_attribute_values PAV2 ON P1.id = PAV2.profile_id
WHERE (PAV.ne_attribute_id = '35' AND PAV.val = 'ASA')
AND (PAV2.ne_attribute_id = '38' AND PAV2.val = 'Stanton')
) B
WHERE A.id = B.id
这将返回
profile
id name
6 nil
这正是我想要的,虽然棘手的部分是第二个parent_profile条件,我需要attribute_id 26和parent_profile_name:'Brekke LLC'
我知道这不起作用,但我需要这样做这样的事情
SELECT * FROM
(SELECT P2. *
FROM profile_attribute_values PAV
INNER JOIN profiles P1 ON P1.id = PAV.parent_profile_id
INNER JOIN profiles P2 ON P2.id = PAV.profile_id
WHERE (PAV.ne_attribute_id = '58' AND P1.`name` = 'Douglas-Connelly')
AND (PAV.ne_attribute_id = '26' AND P1.`name` = 'Brekke LLC')
) A,
.....
我正在动态生成SQL语句,所以我真的需要它尽可能干净。我通常使用ruby活动记录来处理所有内容,因此在涉及SQL语句时我有点宝贝。谢谢!
更新
好的,我找到了一个很好的解决方案来生成我需要的动态查询,只需调用一次数据库。这是完成的课程
class ProfileSearch
def initialize(params, args = {})
@attr_vals = params
@options = args
filter_attrs
end
attr_accessor :parent_attrs, :string_attrs
def search
Profile.joins(generate_query)
end
def generate_query
q = ''
q << parents_query
q << attr_vals_query
end
def parents_query
str = ''
parent_attrs.each_with_index do |pa, i|
str << "INNER JOIN profile_attribute_values PAVP#{i} ON profiles.id = PAVP#{i}.profile_id "\
"AND PAVP#{i}.ne_attribute_id = #{pa} "\
"INNER JOIN profiles PP#{i} ON PP#{i}.id = PAVP#{i}.parent_profile_id "\
"AND PP#{i}.`name` = '#{@attr_vals[pa.to_s]}' "
end
str
end
def attr_vals_query
str = ''
string_attrs.each_with_index do |a, i|
str << "INNER JOIN profile_attribute_values PAVS#{i} ON profiles.id = PAVS#{i}.profile_id "\
"AND PAVS#{i}.ne_attribute_id = #{a} AND PAVS#{i}.val = '#{@attr_vals[a.to_s]}' "
end
str
end
def filter_attrs
ne = NeAttribute.find(@attr_vals.keys)
self.parent_attrs = ne.select{ |x| x.parent_profile)_attr? }.map(&:id)
self.string_attrs = ne.select{ |x| x.string_attr? }.map(&:id)
end
end
答案 0 :(得分:1)
您需要了解的第一件事是joins
。
像这样使用它:
relation_1 = ProfileAttributeValue.joins("
INNER JOIN profiles P1 ON P1.id = PAV.parent_profile_id
INNER JOIN profiles P2 ON P2.id = PAV.profile_id
").where(..).pluck(:id)
relation_2 = ProfileAttributeValue.joins...where...pluck(:id)
relation_1和relation_2尚未运行。它们是ActiveRecord关系。
我喜欢以下技巧:
ProfileAttributeValue.where(["profile_attribute_values.id in (?) or profile_attribute_values.id in (?)]", relation_1, relation_2)
同样,这只是一个积极的记录关系。如果好奇,请致电to_sql
。
添加更多.where
或其他内容,并且在运行时,它将是数据库上的单个查询。
答案 1 :(得分:1)
发布我所做的答案,万一有人错过了上面的内容。这实际上创建了一个单一的连接查询,允许我提取所需的确切数据。生成的查询看起来像这样 -
SELECT `profiles`.*
FROM `profiles`
INNER JOIN profile_attribute_values PAVP0 ON profiles.id = PAVP0.profile_id
AND PAVP0.ne_attribute_id = 26
INNER JOIN profiles PP0 ON PP0.id = PAVP0.parent_profile_id
AND PP0.`name` = 'Brekke LLC'
INNER JOIN profile_attribute_values PAVP1 ON profiles.id = PAVP1.profile_id
AND PAVP1.ne_attribute_id = 58
INNER JOIN profiles PP1 ON PP1.id = PAVP1.parent_profile_id
AND PP1.`name` = 'Douglas-Connelly'
INNER JOIN profile_attribute_values PAVS0 ON profiles.id = PAVS0.profile_id
AND PAVS0.ne_attribute_id = 35 AND PAVS0.val = 'Asa'
INNER JOIN profile_attribute_values PAVS1 ON profiles.id = PAVS1.profile_id
AND PAVS1.ne_attribute_id = 38 AND PAVS1.val = 'Stanton'
这是生成
的类class ProfileSearch
def initialize(params, args = {})
@attr_vals = params
@options = args
filter_attrs
end
attr_accessor :parent_attrs, :string_attrs
def search
Profile.joins(generate_query)
end
def generate_query
q = ''
q << parents_query
q << attr_vals_query
end
def parents_query
str = ''
parent_attrs.each_with_index do |pa, i|
str << "INNER JOIN profile_attribute_values PAVP#{i} ON profiles.id = PAVP#{i}.profile_id "\
"AND PAVP#{i}.ne_attribute_id = #{pa} "\
"INNER JOIN profiles PP#{i} ON PP#{i}.id = PAVP#{i}.parent_profile_id "\
"AND PP#{i}.`name` = '#{@attr_vals[pa.to_s]}' "
end
str
end
def attr_vals_query
str = ''
string_attrs.each_with_index do |a, i|
str << "INNER JOIN profile_attribute_values PAVS#{i} ON profiles.id = PAVS#{i}.profile_id "\
"AND PAVS#{i}.ne_attribute_id = #{a} AND PAVS#{i}.val = '#{@attr_vals[a.to_s]}' "
end
str
end
def filter_attrs
ne = NeAttribute.find(@attr_vals.keys)
self.parent_attrs = ne.select{ |x| x.parent_profile)_attr? }.map(&:id)
self.string_attrs = ne.select{ |x| x.string_attr? }.map(&:id)
end
end