【问题标题】:MySql Query: Select top 3 rows from table for each categoryMySql 查询:从表中为每个类别选择前 3 行
【发布时间】:2011-03-09 14:36:52
【问题描述】:

我有一个记录表,其中有一行名为category。我插入了太多文章,我只想从每个类别中选择两篇文章。

我试图做这样的事情:

我创建了一个视图:

CREATE VIEW limitrows AS 
   SELECT * FROM tbl_artikujt ORDER BY articleid DESC LIMIT 2 

然后我创建了这个查询:

SELECT * 
FROM tbl_artikujt 
WHERE 
   artikullid IN
   (
      SELECT artikullid
      FROM limitrows
      ORDER BY category DESC
   )
ORDER BY category DESC;

但这不起作用,只给我两条记录?

【问题讨论】:

标签: mysql view limit


【解决方案1】:

LIMIT 仅停止语句返回的结果数。您正在寻找的通常称为分析/窗口/排名函数 - MySQL 不支持但您可以使用变量进行模拟:

SELECT x.*
  FROM (SELECT t.*,
               CASE 
                 WHEN @category != t.category THEN @rownum := 1 
                 ELSE @rownum := @rownum + 1 
               END AS rank,
               @category := t.category AS var_category
          FROM TBL_ARTIKUJT t
          JOIN (SELECT @rownum := NULL, @category := '') r
      ORDER BY t.category) x
 WHERE x.rank <= 3

如果您不更改 SELECT x.*,结果集将包含 rankvar_category 值 - 如果不是这种情况,您必须指定您真正需要的列。

【讨论】:

  • 完美运行,这正是我所需要的
  • 我可以从这个选择中以某种方式创建视图吗,我试过但它说:1349 - 视图的选择在 FROM 子句中包含一个子查询
  • @AXheladini:对不起,MySQL 不允许这样做有几个原因——子查询、使用变量……恐怕 MySQL 视图受到极大限制——它们在 CREATE 中列出了限制查看文档:dev.mysql.com/doc/refman/5.1/en/create-view.html
  • 对我来说,这似乎从每个类别中选择前 3 名,就好像它被订购为 t.category ASC, t.articleid ASC 但你会如何选择它作为 t.category ASC, t.articleid DESC
【解决方案2】:
SELECT * FROM (   
    SELECT  VD.`cat_id` ,  
       @cat_count := IF( (@cat_id = VD.`cat_id`), @cat_count + 1, 1 ) AS 'DUMMY1', 
       @cat_id := VD.`cat_id` AS 'DUMMY2',
       @cat_count AS 'CAT_COUNT'   
     FROM videos VD   
     INNER JOIN categories CT ON CT.`cat_id` = VD.`cat_id`  
       ,(SELECT @cat_count :=1, @cat_id :=-1) AS CID  
     ORDER BY VD.`cat_id` ASC ) AS `CAT_DETAILS`
     WHERE `CAT_COUNT` < 4

------- STEP FOLLOW ----------  
1 . select * from ( 'FILTER_DATA_HERE' ) WHERE 'COLUMN_COUNT_CONDITION_HERE' 
2.  'FILTER_DATA_HERE'   
    1. pass 2 variable @cat_count=1 and  @cat_id = -1  
    2.  If (@cat_id "match" column_cat_id value)  
        Then  @cat_count = @cat_count + 1    
        ELSE @cat_count = 1      
    3. SET @cat_id = column_cat_id    

 3. 'COLUMN_COUNT_CONDITION_HERE'   
    1. count_column < count_number    

4. ' EXTRA THING '
   1. If you want to execute more than one statement inside " if stmt "
   2. IF(condition, stmt1 , stmt2 )
      1. stmt1 :- CONCAT(exp1, exp2, exp3) 
      2. stmt2 :- CONCAT(exp1, exp2, exp3) 
   3. Final "If" Stmt LIKE 
      1. IF ( condition , CONCAT(exp1, exp2, exp3) , CONCAT(exp1, exp2, exp3) )    
share

【讨论】:

    【解决方案3】:

    使用 group by 而不是 order by。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2017-07-13
      • 1970-01-01
      • 1970-01-01
      • 2019-06-19
      • 1970-01-01
      相关资源
      最近更新 更多