我发现的搜索结果与我正在尝试的结果大不相同。我很确定我再次思考问题了。任何帮助或指示将不胜感激。我正在使用mssql和sql server 2012
我有一个包含3列的表 - document_id,member_id和rating。我想对表中所有匹配的document_id的评级进行平均。
这是整个过程 - 但是我会专注于块之后的脏部分:
select @title = title,
@description = document.[description],
@authors = authors,
@create_date = case when create_date is null then null else convert(varchar(10),create_date, 101) end,
@revision_date = case when revision_date is null then null else convert(varchar(10),revision_date, 101) end,
@file_format_id = document.file_format_id,
@document_category_id = document.document_category_id,
@size = [size],
@document_key = document_key,
@visibility = document.visibility,
@upload_member_id = upload_member_id,
@calculated_rating = case when document_category.allow_member_ratings = 1 then (select avg(convert(decimal,rating)) from document_rating where [document].document_id = document_rating.document_id) else null end,
@member_rating = case when (document_category.allow_member_ratings = 1) then (select convert(decimal,rating) from document_rating where [document].document_id = document_rating.document_id) else null end
from club.dbo.document
inner join club.dbo.document_category
on (document.document_category_id = document_category.document_category_id)
inner join club.dbo.file_format
on (document.file_format_id = file_format.file_format_id)
where document_id = @document_id
and document_category.club_id = @club_id
给我错误的部分是@calculated_rating输出参数的子查询。
@calculated_rating = case when document_category.allow_member_ratings = 1
then (select avg(convert(decimal,rating))
from document_rating
where [document].document_id = document_rating.document_id)
else null end,
上面的结果是“子查询返回的值超过1。当子查询跟随=,!=,<”时,不允许这样做。如果document_id有超过1个评级条目,那么proc炸弹。显而易见的是,特定document_id的所有评级的总和未被添加/平均。
所以 - 我试过这样的事情:
@calculated_rating = case when document_category.allow_member_ratings = 1 then
(SELECT AVG(convert(decimal,document_rating.rating))
FROM( SELECT SUM(document_rating.rating) rating
FROM document_rating
WHERE [document].document_id = document_rating.document_id
GROUP BY document_rating.document_id) else null end,
那里有一个语法错误,因为我弄得一团糟 - 但我应该在这里使用别名吗?我正在努力做甚么可能吗?
解决
问题不在于平均值 - 而在于会员评级。需要在where语句中添加额外的限制。
@member_rating = case when (document_category.allow_member_ratings = 1) then (select convert(decimal,rating) from document_rating where @member_id = document_rating.member_id and [document].document_id = document_rating.document_id) else null end
答案 0 :(得分:0)
您的问题不是@avg_rating
。它是@member_rating
。这个子查询是:
@member_rating = case when (document_category.allow_member_ratings = 1)
then (select convert(decimal,rating)
from document_rating
where [document].document_id = document_rating.document_id
)
else null end
这可能会返回多个值。您可能也希望在此子查询中使用avg()
。
您的@avg_rating
查询没问题。它是一个返回一行的聚合查询,因为avg()
语句中没有select
且没有group by
子句。它将返回一行和一个值,这是子查询应该做的。
答案 1 :(得分:0)
我个人不会使用子查询,因为你有3次相同的条件,而是在document.document_id = document_rating.document_id上使用Left outer join document_rating 这意味着您获得前一个查询和评级的所有行,只有null,没有评级 然后按document_id(评级)分组
这也应该改善性能。