Chinaunix首页 | 论坛 | 博客
  • 博客访问: 4196311
  • 博文数量: 240
  • 博客积分: 11504
  • 博客等级: 上将
  • 技术积分: 4277
  • 用 户 组: 普通用户
  • 注册时间: 2006-12-28 14:24
文章分类

全部博文(240)

分类: Mysql/postgreSQL

2014-01-10 17:01:36

PostgreSQL 和Oracle 都提供了 row_number() over() 这样的语句来进行对应的字段排名, 很是方便。  MySQL却没有提供这样的语法。 
最近由于从Oracle 迁移到MySQL的需求越来越多,所以这样的转化在所难免。 下面我在MySQL里面来实现这样的做法。


这次我提供的表结构如下,

点击(此处)折叠或打开

  1. Table "ytt.t1"
  2.  Column | Type | Modifiers
  3. --------+-----------------------+-----------
  4.  i_name | character varying(10) | not null
  5.  rank | integer | not null




我模拟了20条数据来做演示。

点击(此处)折叠或打开

  1. t_girl=# select * from t1 order by i_name;
  2.  i_name | rank
  3. ---------+------
  4.  Charlie | 12
  5.  Charlie | 12
  6.  Charlie | 13
  7.  Charlie | 10
  8.  Charlie | 11
  9.  Lily     | 6
  10.  Lily     | 7
  11.  Lily     | 7
  12.  Lily     | 6
  13.  Lily     | 5
  14.  Lily | 7
  15.  Lily | 4
  16.  Lucy | 1
  17.  Lucy | 2
  18.  Lucy | 2
  19.  Ytt | 14
  20.  Ytt | 15
  21.  Ytt | 14
  22.  Ytt | 14
  23.  Ytt | 15
  24. (20 rows)



在PostgreSQL下,我们来对这样的排名函数进行三种不同的执行方式1:
第一种,完整的带有排名字段以及排序。

点击(此处)折叠或打开

  1. t_girl=# select i_name,rank, row_number() over(partition by i_name order by rank desc) as rank_number from t1;
  2.  i_name | rank | rank_number
  3. ---------+------+-------------
  4.  Charlie     | 13 | 1
  5.  Charlie    | 12 | 2
  6.  Charlie    | 12 | 3
  7.  Charlie    | 11 | 4
  8.  Charlie    | 10 | 5
  9.  Lily     | 7 | 1
  10.  Lily     | 7 | 2
  11.  Lily     | 7 | 3
  12.  Lily     | 6 | 4
  13.  Lily     | 6 | 5
  14.  Lily     | 5 | 6
  15.  Lily     | 4 | 7
  16.  Lucy    | 2 | 1
  17.  Lucy    | 2 | 2
  18.  Lucy    | 1 | 3
  19.  Ytt     | 15 | 1
  20.  Ytt     | 15 | 2
  21.  Ytt     | 14 | 3
  22.  Ytt     | 14 | 4
  23.  Ytt     | 14 | 5
  24. (20 rows)



第二种,带有完整的排名字段但是没有排序。

点击(此处)折叠或打开

  1. t_girl=# select i_name,rank, row_number() over(partition by i_name ) as rank_number from t1;
  2.  i_name | rank | rank_number
  3. ---------+------+-------------
  4.  Charlie     | 12 | 1
  5.  Charlie    | 12 | 2
  6.  Charlie    | 13 | 3
  7.  Charlie    | 10 | 4
  8.  Charlie    | 11 | 5
  9.  Lily     | 6 | 1
  10.  Lily     | 7 | 2
  11.  Lily     | 7 | 3
  12.  Lily     | 6 | 4
  13.  Lily     | 5 | 5
  14.  Lily     | 7 | 6
  15.  Lily     | 4 | 7
  16.  Lucy    | 1 | 1
  17.  Lucy    | 2 | 2
  18.  Lucy    | 2 | 3
  19.  Ytt     | 14 | 1
  20.  Ytt     | 15 | 2
  21.  Ytt     | 14 | 3
  22.  Ytt     | 14 | 4
  23.  Ytt     | 15 | 5
  24. (20 rows)




第三种, 没有任何排名字段,也没有任何排序字段。

点击(此处)折叠或打开

  1. t_girl=# select i_name,rank, row_number() over() as rank_number from t1;
  2.  i_name | rank | rank_number
  3. ---------+------+-------------
  4.  Lily     | 7 | 1
  5.  Lucy    | 2 | 2
  6.  Ytt     | 14 | 3
  7.  Ytt     | 14 | 4
  8.  Charlie    | 12 | 5
  9.  Charlie    | 13 | 6
  10.  Lily     | 7 | 7
  11.  Lily     | 4 | 8
  12.  Ytt     | 14 | 9
  13.  Lily     | 6 | 10
  14.  Lucy    | 1 | 11
  15.  Lily     | 7 | 12
  16.  Ytt     | 15 | 13
  17.  Lily     | 6 | 14
  18.  Charlie    | 11 | 15
  19.  Charlie    | 12 | 16
  20.  Lucy    | 2 | 17
  21.  Charlie    | 10 | 18
  22.  Lily     | 5 | 19
  23.  Ytt     | 15 | 20
  24. (20 rows)





MySQL 没有提供这样的语句,所以我用了以下的存储过程来实现。



点击(此处)折叠或打开

  1. DELIMITER $$


  2. USE `t_girl`$$


  3. DROP PROCEDURE IF EXISTS `sp_rownumber`$$


  4. CREATE PROCEDURE `sp_rownumber`(
  5.     IN f_table_name VARCHAR(64),
  6.     IN f_column_partitionby VARCHAR(64),
  7.     IN f_column_orderby VARCHAR(64),
  8.     IN f_is_asc CHAR(4)
  9.     )
  10. BEGIN
  11.       -- Created by ytt at 2014/1/10
  12.       -- Do a row_number() over()
  13.       DECLARE i INT;
  14.       -- Create a temporary table to save result.
  15.       DROP TABLE IF EXISTS tmp_rownum;
  16.       SET @stmt = CONCAT('create temporary table tmp_rownum select *,''rownum'' from ',f_table_name,' where 1 = 0');
  17.       PREPARE s1 FROM @stmt;
  18.       EXECUTE s1;
  19.       
  20.   
  21.       SET i = 0;
  22.       SET @j = 0;
  23.       SET @v_column_paritionby = '';


  24.       -- Check whether parition column is null or not.
  25.       IF (f_column_partitionby = '' OR f_column_partitionby IS NULL) THEN
  26.          -- No additional parition column.
  27. SET @stmt = CONCAT('insert into tmp_rownum select *,@j:= @j+1 as rownum from ',
  28. f_table_name);    
  29. PREPARE s1 FROM @stmt;
  30. EXECUTE s1;
  31.       ELSE
  32.        -- Give partition column.
  33. SET @stmt = CONCAT('select count(*) from (select count(*) from ',f_table_name,' group by ',
  34. f_column_partitionby,') as a into @cnt');
  35. PREPARE s1 FROM @stmt;
  36. EXECUTE s1;
  37.       
  38.         WHILE i < @cnt
  39.         DO
  40.  -- Get the partition value one by one.
  41.  SET @stmt = CONCAT('select ',f_column_partitionby,' from ',f_table_name,' group by ',f_column_partitionby,' limit ',i,',1 into @v_column_partitionby');
  42.  PREPARE s1 FROM @stmt;
  43.  EXECUTE s1;
  44.  -- Check whether sort is needed.
  45.           IF f_column_orderby = '' OR f_column_orderby IS NULL THEN
  46.             SET @stmt = CONCAT('insert into tmp_rownum select *,@j:= @j+1 as rownum from ',
  47. f_table_name,' where ',f_column_partitionby,' = ''',@v_column_partitionby,'''');
  48.  ELSE
  49.    SET @stmt = CONCAT('insert into tmp_rownum select *,@j:= @j+1 as rownum from ',
  50. f_table_name,' where ',f_column_partitionby,' = ''',@v_column_partitionby,'''
  51. order by ',f_column_orderby,' ',f_is_asc);
  52.           END IF;
  53.           SET @j = 0;
  54.  PREPARE s1 FROM @stmt;
  55.  EXECUTE s1;

  56.           SET i = i + 1;
  57.         END WHILE;
  58.       END IF;
  59.       -- Reset all session variables.
  60.       SET @j = NULL;
  61.       SET @v_column_paritionby = NULL;
  62.       SET @cnt = NULL;
  63.       SELECT * FROM tmp_rownum;
  64.     END$$


  65. DELIMITER ;



我们同样来执行第一种,第二种以及第三种查询,结果如下:
第一种,

query result

i_name rank rownum
Charlie 13 1
Charlie 12 2
Charlie 12 3
Charlie 11 4
Charlie 10 5
Lily 7 1
Lily 7 2
Lily 7 3
Lily 6 4
Lily 6 5
Lily 5 6
Lily 4 7
Lucy 2 1
Lucy 2 2
Lucy 1 3
Ytt 15 1
Ytt 15 2
Ytt 14 3
Ytt 14 4
Ytt 14 5

第二种,

query result

i_name rank rownum
Charlie 12 1
Charlie 13 2
Charlie 11 3
Charlie 12 4
Charlie 10 5
Lily 7 1
Lily 7 2
Lily 4 3
Lily 6 4
Lily 7 5
Lily 6 6
Lily 5 7
Lucy 2 1
Lucy 1 2
Lucy 2 3
Ytt 14 1
Ytt 14 2
Ytt 14 3
Ytt 15 4
Ytt 15 5


第三种,

query result

i_name rank rownum
Lily 7 1
Lucy 2 2
Ytt 14 3
Ytt 14 4
Charlie 12 5
Charlie 13 6
Lily 7 7
Lily 4 8
Ytt 14 9
Lily 6 10
Lucy 1 11
Lily 7 12
Ytt 15 13
Lily 6 14
Charlie 11 15
Charlie 12 16
Lucy 2 17
Charlie 10 18
Lily 5 19
Ytt 15 20

阅读(6397) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~