Mysql 中实现文章置顶分页

先说一下背景吧:现在有一张表learn_article,里面有一个字段hot,当hot值为1时表示文章置顶,现在要分页查询文章列表,要求置顶文章放在最前面然后按id倒序排列,其他文章按id倒序排列。
  提到置顶还获取列表,我当然想到使用UNION啦,于是就写出了下面这个SQL:

SELECT id,hot FROM 
((SELECT * FROM learn_article WHERE hot = 1 ORDER BY id DESC)
UNION 
(SELECT * FROM learn_article WHERE hot!= 1 ORDER BY id DESC))A
LIMIT 1,10
错误的置顶分页 1

然而,从结果也可以看出它不是我想要的结果,为毛它没有执行ORDER BY id DESC啊!查阅了UNION相关说明才知道:
联合查询不仅仅是将数据集合合并,他并不是将每个子查询一个一个查询出来后联接在一起,数据库是将整段查询语句解释之后统一查询得到的是整个的数据集合。另外order by在一个数据集合查询里也只能被执行一次并且要放在最后一个查询子句后,所以上面查询子句中的ORDER BY id DESC会被Mysql忽略!
  因此,在联合查询里,order by 要写在最后一个子查询之后,并且,该排序是对整个联合查询出来的结果集排序的,并不是只对最后一个子查询排序,像下面这样:

SELECT id,hot FROM learn_article WHERE hot = 1
UNION 
SELECT id,hot FROM learn_article WHERE hot!= 1 
ORDER BY id DESC  LIMIT 1,10
错误的置顶分页 2

  但是,这更不是我要的结果啊!!到底还要我怎样呢!后来查阅资料发现:在使用UNION连接多个查询结果的情况下,想要在每个select子句使用ORDER BY 就需要同时使用limit!这样的话问题好像就能解决啦:

SELECT id,hot FROM (
(SELECT * FROM learn_article WHERE hot = 1 ORDER BY id DESC LIMIT 0,3)
UNION 
(SELECT * FROM learn_article ORDER BY id DESC LIMIT 0,10))A
正确但麻烦的置顶分页查询

  但是问题来啦,我怎么知道置顶的文章查几条,非置顶的文章查几条呢?那只有再查询一下置顶文章的条数啦:

SELECT COUNT(*) FROM learn_article WHERE hot =1

然后再根据查询起点offset和查询条数limit,计算置顶文章查询的数量(tlimit)和非置顶文章的查询起点(loffset)和查询数量(flimit):

int offset = Integer.valueOf(request.getParameter("offset"));
int limit = Integer.valueOf(request.getParameter("limit"));
//查询置顶文章数量,置顶文章不超过一页
int topNum = this.articleService.countTop(params);
int tlimit = topNum ;
int foffset = offset;
int flimit = limit;

if(topNum < offset){
    //第二页以后
    offset-=topNum;
    tlimit =0;
}else{
    //第一页
    flimit = limit - topNum > 0 ? limit - topNum : 0;
    tlimit = topNum;
}

然后就将tlimit,foffset,flimit传到后台,进行查询:

SELECT id,hot FROM (
(SELECT * FROM learn_article WHERE hot = 1 ORDER BY id DESC LIMIT 0,#{tlimit})
UNION 
(SELECT * FROM learn_article ORDER BY id DESC LIMIT #{foffset},#{flimit}))A

这样就是实现了文章置顶的分页查询,是在太麻烦啦,到是有一种操作简单的方法就是使用FIND_IN_SET:
SELECT * FROM learn_article ORDER BY FIND_IN_SET(hot,'1') DESC ,id DESC LIMIT 0,10;

使用FIND_IN_SET实现置顶分页

但是呢,大家都知道的FIND_IN_SET执行效率太低啦,而且费内存!大家要是有更好的办法一定要告诉我哦!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容