在做数据分析时,我们会经常听到同比、环比同比的概念。各个企业和组织在发布统计数据时,通常喜欢用同比、环比来和之前的历史数据进行比较,用来说明数据的变化情况。例如,统计局公布2022年1月份CPI同比增长0.9%,环比增长0.6%。

实际中,在基于数据库的数据分析场景中,环比和同比是典型的复杂计算场景之一,特别是在Oracle等商业数据库的分析函数出现之前。以MySQL为例,在8.0版本中才引入了Lag和Lead函数,这两个函数结合开窗函数有效的提高了同比、环比等复杂运算的实现效率。在5.x系列版本中,MySQL需要依赖多次嵌套子查询和自关联才能实现此类计算。

我们以一个简单的例子,来分别看下,MySql 5.x和8.0是具体实现同比、环比计算的。

示例数据见表:

  1. CREATE TABLE sales (
  2. `产品ID` varchar(20),
  3. `销售数量` int(20) ,
  4. `销售时间` timestamp(6) NULL DEFAULT NULL
  5. )
  6. INSERT INTO sales VALUES ('C1001', 15, '2020-06-01 10:10:12');
  7. INSERT INTO sales VALUES ('C1002',26, '2020-05-02 0:10:12');
  8. INSERT INTO sales VALUES ('C1003', 21, '2020-04-03 0:10:12');
  9. INSERT INTO sales VALUES ('C1003', 23, '2020-04-04 0:10:12');
  10. INSERT INTO sales VALUES ('C1003', 0, '2020-03-05 0:10:12');
  11. INSERT INTO sales VALUES ('C1001', 16, '2020-02-06 3:0:12');
  12. INSERT INTO sales VALUES ('C1002', 32, '2020-01-07 0:10:12');
  13. INSERT INTO sales VALUES ('C1001', 16, '2019-12-08 0:12:24');
  14. INSERT INTO sales VALUES ('C1001', 32, '2019-06-09 0:12:24');
  15. INSERT INTO sales VALUES ('C1002', 17, '2019-05-09 0:12:24');

1、MySQL 5.x :通过子查询和关联实现同比和占比计算

以按年月统计不同年份的销售总值,并计算环比(销售总额同比上期)、同比(销售总额同比去年同期)为例。

示例表结构和数据

通过SQL计算环比和同比:

  1. select year(c.销售时间) yy,month(c.销售时间) mm,
  2. concat(ifnull(abs(round((sum(c.销售数量)-ss1)/ss1*100,2)),0),'%') 同比,
  3. concat(ifnull(abs(round((sum(c.销售数量)-ss2)/ss2*100,2)),0),'%') 环比
  4. from sales c
  5. left join (select month(a.销售时间) mm1,
  6. year(a.销售时间) yy1,
  7. sum(a.销售数量) ss1
  8. from sales a
  9. GROUP BY mm1,yy1) a
  10. on month(c.销售时间) = a.mm1
  11. and a.yy1 = year(c.销售时间)-1
  12. left join (select month(a.销售时间) mm2,
  13. year(a.销售时间) yy2,
  14. sum(a.销售数量) ss2
  15. from sales a
  16. GROUP BY mm2,yy2) b
  17. on (b.yy2 = year(c.销售时间) and b.mm2+1 = month(c.销售时间) OR (yy2=year(c.销售时间)-1
  18. AND b.mm2 = 12 AND month(c.销售时间) = 1))
  19. group by yy, mm
  20. order by yy,mm asc

计算结果:

2、 MySQL 8.0 :通过分析函数实现同比和占比计算**

MySql8.0支持了Lead和Lag分析函数,虽然可以大幅提高同、环比计算的效率,但仍然需要编写SQL语句处理。

2、1 计算同比

  1. select t2.年份,t2.月份,concat(round((t2.数量-t1.数量)/t1.数量,2)*100,'%') as 同比 from (
  2. SELECT year(销售时间) as 年份,month(销售时间) as 月份,sum(销售数量) as 数量 from sales
  3. group by year(销售时间),month(销售时间) order by year(销售时间) desc, month(销售时间) desc
  4. ) t1
  5. ,(
  6. SELECT year(销售时间) as 年份,month(销售时间) as 月份,sum(销售数量) as 数量 from sales
  7. group by year(销售时间),month(销售时间) order by year(销售时间) desc, month(销售时间) desc
  8. ) t2 where t1.年份=t2.年份-1 and t1.月份=t2.月份

2.2计算环比

  1. SELECT
  2. mm,
  3. CONCAT(
  4. ROUND(
  5. IFNULL(
  6. (xl - first_xl) / first_xl * 100,
  7. 2
  8. ),
  9. 0
  10. ),
  11. '%'
  12. ) AS 环比
  13. FROM
  14. (
  15. SELECT
  16. mm,
  17. xl,
  18. lead (xl, 1) over (ORDER BY mm DESC) AS first_xl
  19. FROM
  20. (
  21. SELECT
  22. DATE_FORMAT(销售时间, '%Y-%m') AS mm,
  23. sum(销售数量) AS xl
  24. FROM
  25. sales
  26. GROUP BY
  27. DATE_FORMAT(销售时间, '%Y-%m')
  28. ) t
  29. ) a

在SqlServer2008R2和Oracle10g之后,都提供了Lag和Lead分析函数。具体的计算逻辑和用法与上述MySQL8.0类似。

3、使用 BI工具的计算引擎

针对此类复杂的计算场景,商业智能BI数据分析工具提供了更加高效的解决方案。以Wyn Enterprise嵌入式商业智能软件为例,其内置的wax分析表达式和快速计算引擎,提供直接实现同比、环比等复杂计算的能力,而不再需要写复杂冗长的SQL。

3、1 使用内置的同比、环比快速计算功能**

同比、环比等计算一般是BI工具的标准功能,我们可以直接通过设置实现。

3、2 使用数据分析表达式

如果内置的快速计算无法满足要求,还可以通过分析表达式实现更复杂的计算。分析表达式是一种更加灵活、强大的数据计算方式,通过丰富的函数,用户可以像Excel公式一样自由组合,实现更加强大的分析能力。分析表达式基于数据模型进行业务计算,以一些定义好的函数运用正确的语法来完成某个复杂的业务逻辑计算。这样可以使用户更灵活的地使用数据,最大限度的利用数据。

各位老板们,通过对比SQL和BI数据分析工具在处理同比、环比等复杂计算中的差异,我们可以发现,还是专业的工具在数据计算和处理能力上要更加便捷。以后在工作中,如果有类似的分析计算需求,选择BI分析工具来处理就是再合适不过的了。

实现同比、环比计算的N种姿势的更多相关文章

  1. Hive之同比环比的计算

    Hive系列文章 Hive表的基本操作 Hive中的集合数据类型 Hive动态分区详解 hive中orc格式表的数据导入 Java通过jdbc连接hive 通过HiveServer2访问Hive Sp ...

  2. 再谈Cognos利用FM模型来做同比环比

    很早之前已经讲过 <Cognos利用DMR模型开发同比环比>这篇文章里说的是不利用过滤器,而是采用 except (lastPeriods (-9000,[订单数据分析].[日期维度].[ ...

  3. 【九天教您南方cass 9.1】 10 DTM土方计算的四种方法

    同学们大家好,欢迎收看由老王测量上班记出品的cass9.1视频课程 我是本节课主讲老师九天. 我们讲课的教程附件也是共享的,请注意索取测量空间中. [点击索取cass教程]5元立得 (给客服说暗号:“ ...

  4. cognos report同比环比以及默认为当前月分析

    现在的需求是按月份分析不同时期的余额数据,.(报表工具:cognos report:建模工具:FM) ------------------------------------------------- ...

  5. 又拍云张聪:OpenResty 动态流控的几种姿势

    2019 年 1 月 12 日,由又拍云.OpenResty 中国社区主办的 OpenResty × Open Talk 全国巡回沙龙·深圳站圆满结束,又拍云首席架构师张聪在活动上做了< Ope ...

  6. python打开文件的N种姿势

    # python打开文件的N种姿势 print('[1]使用open()函数+简单for循环') f1 = open('python.txt') for line in f1: print(line. ...

  7. Python 高级特性介绍 - 迭代的99种姿势 与协程

    Python 高级特性介绍 - 迭代的99种姿势 与协程 引言 写这个笔记记录一下一点点收获 测试环境版本: Python 3.7.4 (default, Sep 28 2019, 16:39:19) ...

  8. 快速了解IOC的几种姿势

    一.首先我们了解IOC如何注入的几种姿势 构造函数注入(Constructor Injection) Ioc容器会智能的选择和调用合适的构造函数以创建依赖的对象.如果被选择的构造函数具有相应的参数,I ...

  9. 两个文件去重的N种姿势

    最近利用shell帮公司优化挖掘关键词的流程,用shell替代了多个环节的操作,极大提高了工作效率. shell在文本处理上确有极大优势,比如多文本合并.去重等,但是最近遇到了一个难搞的问题,即两个大 ...

随机推荐

  1. Python初学笔记之字符串

    一.字符串的定义 字符串是就一堆字符,可以使用""(双引号).''(单引号)来创建. 1 one_str = "定义字符串" 字符串内容中包含引号时,可以使用转 ...

  2. uni微信小程序优化,打包后的import vue路径是可删除的

    这次的优化我公司项目主包只减小了32kb,但是减小的不仅仅是主包,所有分包均在没有改动任何业务代码的情况下完成了压缩空间的优化. 主包分包压缩空间的优化都要视项目而定,32kb只是我公司的小程序项目. ...

  3. Log4j2日志技术总结

    前言 现在流行是SLF4j和Log4j2组合的日志技术,但为了日志技术归类,故前因后果都将做一下介绍. 市场上流行的日志框架 JUL java util logging Java开发团队开发,Jdk原 ...

  4. 汉字编码(【Unicode】 【UTF-8】 【Unicode与UTF-8之间的转换】 【汉字 Unicode 编码范围】【中文标点Unicode码】【GBK编码】【批量获取汉字UNICODE码】)

    Unicode与UTF-8互转(C语言实现):http://blog.csdn.net/tge7618291/article/details/7599902 汉字 Unicode 编码范围:http: ...

  5. python继承关系中,类属性的修改

    class Grandfather(object): mylist = [] def __init__(self): pass class Father(Grandfather): pass Gran ...

  6. Linux 内核引导参数简介

    概述 内核引导参数大体上可以分为两类:一类与设备无关.另一类与设备有关.与设备有关的引导参数多如牛毛,需要你自己阅读内核中的相应驱动程序源码以获取其能够接受的引导参数.比如,如果你想知道可以向 AHA ...

  7. linux实用技巧:在虚拟机vmware16软件上安装CentOs8.2虚拟机,重置可用源和安装输入法

    前言   开发服务器应用,需要使用到CentOs8.2,安装到虚拟机上方便快捷.   提前准备 Vmware 16虚拟机软件  下载VM16版本及以上的vmware虚拟机版本,否则没有CentOs8选 ...

  8. 【BZOJ2654】tree(生成树 二分)

    题目链接 大意 给你一个无向带权连通图,每条边是黑色或白色,求一棵最小权的恰好有\(Need\)条白色边的生成树. 题目保证有解,输出最小权值. 其中每条边权在\([1,100]\)范围内. 思路 首 ...

  9. Python中set集合常用操作

    功能 Python符号 Python方法 备注 交集 & intersection, intersection_update &:取两者交集>>> set3 = se ...

  10. 组合&反射&面向对象内置函数

    内容概要 组合 反射 面向对象的内置函数 异常 内容详细 一.组合 组合:在对象中定义一个属性,属性的值是另一个对象 除了继承父类的方法,这是获取另一个类中属性的另一种方式 如果想给学生对象添加课程属 ...