子查询中的SQL AVG - 导致“子查询返回的值超过1 ...”

时间:2014-02-12 22:19:58

标签: sql sql-server group-by sql-server-2012 subquery

我发现的搜索结果与我正在尝试的结果大不相同。我很确定我再次思考问题了。任何帮助或指示将不胜感激。我正在使用mssql和sql server 2012

我有一个包含3列的表 - document_id,member_id和rating。我想对表中所有匹配的document_id的评级进行平均。

enter image description here

这是整个过程 - 但是我会专注于块之后的脏部分:

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

2 个答案:

答案 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(评级)分组

这也应该改善性能。