Mysql 平均额定值,额定值存储在两列中

Mysql 平均额定值,额定值存储在两列中,mysql,average,Mysql,Average,我已经搜索了这个网站,所有类似的答案都指向一个单独的评分表,而不是像这样的实现 我已经用PHP/MYSQL将wordpress网站转换成了一个自建网站,想象一下实际的帖子是一篇评论,而评论是与原始帖子相关的实际评论 我遇到的问题是,posts表有一个评级字段,输入的每个评论都有一个评级字段 我试着使用别名AvgRating作为一个,但是它只返回评论评级(c.rating),如果删除该行,它会从posts表中返回评论评级,有没有办法将别名合并为平均评级?显然,只有这部分是错误的,但不会抛出错误 S

我已经搜索了这个网站,所有类似的答案都指向一个单独的评分表,而不是像这样的实现

我已经用PHP/MYSQL将wordpress网站转换成了一个自建网站,想象一下实际的帖子是一篇评论,而评论是与原始帖子相关的实际评论

我遇到的问题是,posts表有一个评级字段,输入的每个评论都有一个评级字段

我试着使用别名AvgRating作为一个,但是它只返回评论评级(c.rating),如果删除该行,它会从posts表中返回评论评级,有没有办法将别名合并为平均评级?显然,只有这部分是错误的,但不会抛出错误

SELECT p.post_id
     , p.title
     , AVG(p.rating) AvgRating
     , AVG(c.rating) AvgRating -- returns average from comments field, if line is removed will return posts rating value instead..
  FROM posts p
  LEFT 
  JOIN comments c 
    ON p.post_id = c.post_id 
 WHERE p.post_id = $post_id
   AND c.post_id = $post_id
   AND active = 1  
 GROUP 
    BY p.post_id
     , p.title;
我正在尝试获取模式聚合计数标记的平均值。有没有一种方法可以在一个查询中实现这一点?或者,我唯一的选择是使用2个查询返回2个平均值,并使用PHP函数返回两者之间的平均值

下面是一个测试数据库:

CREATE TABLE `posts` (
`post_id` int(11) NOT NULL,
`cat_id` int(11) NOT NULL,
`rating` int(11) NOT NULL,
`title` varchar(255) NOT NULL,
`content` varchar(5000) NOT NULL,
`date_added` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
`author` varchar(255) NOT NULL,
`active` int(11) DEFAULT '0',
`slug` varchar(255) NOT NULL,
`user_submitted` int(11) NOT NULL DEFAULT '1'
) ENGINE=InnoDB AUTO_INCREMENT=2157 DEFAULT CHARSET=latin1;

CREATE TABLE `comments` (
`cmt_id` int(11) NOT NULL,
`post_id` int(11) NOT NULL,
`rating` int(11) NOT NULL,
`comment` varchar(5000) NOT NULL,
`cmt_author` varchar(255) NOT NULL,
`cmt_email` varchar(255) NOT NULL,
`ip_address` varchar(100) NOT NULL,
`cmt_date` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
`cmt_status` int(11) NOT NULL DEFAULT '0'
) ENGINE=InnoDB AUTO_INCREMENT=397 DEFAULT CHARSET=latin1;

INSERT INTO `posts` (`post_id`, `cat_id`, `rating`, `title`, `content`, `date_added`, `author`, `active`, `slug`, `user_submitted`) VALUES
(2, 9, 5, 'site1.com', 'Original post review body', '2010-05-07 05:00:00', '', 1, 'site1-com', 1);

INSERT INTO `comments` (`cmt_id`, `post_id`, `rating`, `comment`, `cmt_author`, `cmt_email`, `ip_address`, `cmt_date`, `cmt_status`) VALUES
(1, 2, 5, 'it is a good site test review', 'Anonymous', '', '', '2010-01-06 21:51:00', 1),
(2, 2, 3, 'it is an average site test review', 'Anonymous', '', '', '2010-01-06 20:51:00', 1),
(3, 2, 1, 'it is an bad site test review', 'Anonymous', '', '', '2010-01-06 23:51:00', 1),
(4, 2, 0, 'neutral test review', 'Anonymous', '', '', '2010-01-06 22:51:00', 1),
(5, 2, 2, 'below average test review', 'Anonymous', '', '', '2010-01-06 22:51:00', 1);
要运行的查询:

SELECT p.post_id
 , p.title
 , AVG(p.rating) AvgRating
 , AVG(c.rating) AvgRating -- returns average from comments field, if line is removed will return posts rating value instead..
FROM posts p
LEFT 
JOIN comments c 
ON p.post_id = c.post_id 
WHERE p.post_id = 2
AND active = 1  
GROUP 
BY p.post_id
 , p.title;
在测试数据库的PHPmyadmin中运行此选项将返回:AvgRating:Current selection不包含唯一列。栅格编辑、复选框、编辑、复制和删除功能不可用

(这是意料之中的,因为我错误地认为这会将两者的平均值放入alias列的一个实例中)

结果:

post_id  title      AvgRating  AvgRating
   2     site1.com  5.0000     2.2000 
从某种意义上说,这是正确的,但如何将原来的帖子评级5.000纳入评论评级2.2000


期望的结果应该是2.6667而不是2.2000,因为(5+5+3+1+0+2)/6=2.6667。2.2000只是评论评分的平均值,并没有将原来的5.000分考虑在内。一个方法是联合

SELECT post_id
     , AVG(rating) avg_rating
  FROM 
     ( SELECT post_id
            , rating FROM posts
        UNION 
          ALL
       SELECT post_id
            , rating 
         FROM comments
     ) x 
 GROUP 
    BY post_id;

一种方法是联合

SELECT post_id
     , AVG(rating) avg_rating
  FROM 
     ( SELECT post_id
            , rating FROM posts
        UNION 
          ALL
       SELECT post_id
            , rating 
         FROM comments
     ) x 
 GROUP 
    BY post_id;

我混合使用PHP和SQL解决了这个问题:(我最初是在寻找一种在SQL中实现这一点的不那么麻烦的方法)


我是通过混合使用PHP和SQL来解决这个问题的:(我最初是在寻找一种在SQL中实现这一点的不那么麻烦的方法)


我错过什么了吗?这似乎太明显了。两个别名相同。因此,在PHP中,第一个别名被第二个别名覆盖。另外,
和c.post_id=$post_id
是多余的(事实上,事与愿违,因为它将左[OUTER]连接呈现为内部连接)。您好,是的,很抱歉,我意识到了这一点,我正在努力从两个表中获取平均值,平均值来自评论评级领域,这是很好的,但是需要从帖子评级字段中考虑原始评级,并不知何故将其添加到总体平均值中。如果你愿意,考虑下面简单的两步行动:1。如果您还没有这样做,请提供适当的CREATE和INSERT语句(和/或SQLFIDLE),以便我们可以更轻松地复制问题。2.如果您还没有这样做,请提供与步骤1中提供的信息相对应的所需结果集。您好,谢谢,我已经更新了问题,因为我无法让SQL fiddle工作,即使它声明它创建了架构,运行的查询声明posts表不存在。我是否遗漏了什么?这似乎太明显了。两个别名相同。因此,在PHP中,第一个别名被第二个别名覆盖。另外,
和c.post_id=$post_id
是多余的(事实上,事与愿违,因为它将左[OUTER]连接呈现为内部连接)。您好,是的,很抱歉,我意识到了这一点,我正在努力从两个表中获取平均值,平均值来自评论评级领域,这是很好的,但是需要从帖子评级字段中考虑原始评级,并不知何故将其添加到总体平均值中。如果你愿意,考虑下面简单的两步行动:1。如果您还没有这样做,请提供适当的CREATE和INSERT语句(和/或SQLFIDLE),以便我们可以更轻松地复制问题。2.如果您还没有这样做,请提供与步骤1中提供的信息相对应的所需结果集。您好,谢谢,我已经更新了问题,因为我无法让SQL FIDLE工作,即使它声明它创建了架构,运行一个查询声明posts表不存在。嘿,我尽了最大努力解释,非常感谢您的帮助,这很复杂,因为我以前从未尝试过。我明确表示,返回的平均值没有考虑到原始帖子表的评分,只返回了评论的平均评分。我是如何得到预期结果的,就是在评论评级字段下添加另一行,使其与帖子评级字段的值完全相同,然后运行查询。嘿,谢谢!这是一种更干净的方法!事实上,我创建的功能在我发布时未在现场进行测试,虽然它按预期在测试数据库上运行,但在如上所述进行现场测试时,它没有考虑原始评级是否为零等。但现在我将其考虑到所有这些条件,因此它工作得很好,但当我有更多时间时,我将修改它并使用您的方法,因为这是一种更干净的方法。非常感谢你的帮助!嘿,我试着尽我所能解释,真的很感谢你的帮助,这很复杂,因为我以前从未尝试过。我明确表示,返回的平均值没有考虑到原始帖子表的评分,只返回了评论的平均评分。我是如何得到预期结果的,就是在评论评级字段下添加另一行,使其与帖子评级字段的值完全相同,然后运行查询。嘿,谢谢!这是一种更干净的方法!事实上,我创建的功能在我发布它时没有在现场测试过,虽然它按预期在测试数据库上工作,但在如上所述的现场测试时,它没有考虑原始评级是否为零等。但是现在我已经考虑了所有这些条件,所以它工作得很好,但是当我有更多的时间时,我会
function aggRating($db,$post_id){
    foreach($db->query("SELECT 
    p.post_id,
    p.title,
    p.rating AS oreview,
    SUM(c.rating) AS creview
    FROM posts p
    LEFT JOIN comments c ON p.post_id = ".$post_id." where active=1 
    GROUP BY 
    p.post_id,
    p.title") as $ratr) {
//get review count
    $stmt = $db->query("SELECT * FROM posts INNER JOIN comments ON comments.post_id=posts.post_id where comments.post_id=".$post_id." AND active=1");
    $row_count = $stmt->rowCount();
    $numreviews = $row_count+1; //add original post review to total
echo round(($ratr['creview']+$ratr['oreview'])/$numreviews,2);
         }
  }

//return result (2.67)
echo aggRating($db,$post_id); 
?>