环比怎么算得比较简单(环比计算的N种姿势?)
在做数据分析时,我们会经常听到同比、环比的概念。各个企业和组织在发布统计数据时,通常喜欢用同比、环比来和之前的历史数据进行比较,用来说明数据的变化情况。例如,统计局公布20**年1月份CPI同比增长0.9%,环比增长0.6%。
实际在基于数据库的数据分析场景中,环比和同比是典型的复杂计算场景之一,特别是在Oracle等商业数据库的分析函数出现之前。以MySQL为例,在8.0版本中才引入了Lag和Lead函数,这两个函数结合开窗函数有效提高了同比、环比等复杂运算的实现效率。在5.x系列版本中,MySQL需要依赖多次嵌套子查询和自关联才能实现此类计算。
我们以一个简单的例子,来分别看下,MySql 5.x和8.0是具体实现同比、环比计算的。
示例数据见表:
CREATE TABLE sales (
`产品ID` varchar(20),
`销售数量` int(20) ,
`销售时间` timestamp(6) NULL DEFAULT NULL
)
INSERT INTO sales VALUES ('C1001', 15, '2020-06-01 10:10:12');
INSERT INTO sales VALUES ('C1002',26, '2020-05-02 0:10:12');
INSERT INTO sales VALUES ('C1003', 21, '2020-04-03 0:10:12');
INSERT INTO sales VALUES ('C1003', 23, '2020-04-04 0:10:12');
INSERT INTO sales VALUES ('C1003', 0, '2020-03-05 0:10:12');
INSERT INTO sales VALUES ('C1001', 16, '2020-02-06 3:0:12');
INSERT INTO sales VALUES ('C1002', 32, '2020-01-07 0:10:12');
INSERT INTO sales VALUES ('C1001', 16, '2019-12-08 0:12:24');
INSERT INTO sales VALUES ('C1001', 32, '2019-06-09 0:12:24');
INSERT INTO sales VALUES ('C1002', 17, '2019-05-09 0:12:24');

1、MySQL 5.x:通过子查询和关联实现同比和占比计算
以按年月统计不同年份的销售总值,并计算环比(销售总额同比上期)、同比(销售总额同比去年同期)为例。

示例表结构和数据
通过SQL计算环比和同比:
select year(c.销售时间) yy,month(c.销售时间) mm,
concat(ifnull(abs(round((sum(c.销售数量)-ss1)/ss1*100,2)),0),'%') 同比,
concat(ifnull(abs(round((sum(c.销售数量)-ss2)/ss2*100,2)),0),'%') 环比
from sales c
left join (select month(a.销售时间) mm1,
year(a.销售时间) yy1,
sum(a.销售数量) ss1
from sales a
GROUP BY mm1,yy1) a
on month(c.销售时间) = a.mm1
and a.yy1 = year(c.销售时间)-1
left join (select month(a.销售时间) mm2,
year(a.销售时间) yy2,
sum(a.销售数量) ss2
from sales a
GROUP BY mm2,yy2) b
on (b.yy2 = year(c.销售时间) and b.mm2+1 = month(c.销售时间) OR (yy2=year(c.销售时间)-1
AND b.mm2 = 12 AND month(c.销售时间) = 1))
group by yy, mm
order by yy,mm asc
计算结果:

2、MySQL 8.0:通过分析函数实现同比和占比计算
MySql8.0支持了Lead和Lag分析函数,虽然可以大幅提高同、环比计算的效率,但仍然需要编写SQL语句处理。
2.1计算同比:
select t2.年份,t2.月份,concat(round((t2.数量-t1.数量)/t1.数量,2)*100,'%') as 同比 from (
SELECT year(销售时间) as 年份,month(销售时间) as 月份,sum(销售数量) as 数量 from sales
group by year(销售时间),month(销售时间) order by year(销售时间) desc, month(销售时间) desc
) t1
,(
SELECT year(销售时间) as 年份,month(销售时间) as 月份,sum(销售数量) as 数量 from sales
group by year(销售时间),month(销售时间) order by year(销售时间) desc, month(销售时间) desc
) t2 where t1.年份=t2.年份-1 and t1.月份=t2.月份

2.2计算环比:
SELECT
mm,
CONCAT(
ROUND(
IFNULL(
(xl - first_xl) / first_xl * 100,
2
),
0
),
'%'
) AS 环比
FROM
(
SELECT
mm,
xl,
lead (xl, 1) over (ORDER BY mm DESC) AS first_xl
FROM
(
SELECT
DATE_FORMAT(销售时间, '%Y-%m') AS mm,
sum(销售数量) AS xl
FROM
sales
GROUP BY
DATE_FORMAT(销售时间, '%Y-%m')
) t
) a

在SqlServer2008R2和Oracle10g之后,都提供了Lag和Lead分析函数。具体的计算逻辑和用法与上述MySQL8.0类似。
3、使用BI工具的计算引擎
针对此类复杂的计算场景,商业智能BI数据分析工具提供了更加高效的解决方案。以Wyn Enterprise嵌入式商业智能软件为例,其内置的wax分析表达式和快速计算引擎,提供直接实现同比、环比等复杂计算的能力,而不再需要写复杂冗长的SQL。
3.1 使用内置的同比、环比快速计算功能
同比、环比等计算一般是BI工具的标准功能,我们可以直接通过设置实现。


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



通过对比SQL和BI数据分析工具在处理同比、环比等复杂计算中的差异,我们可以发现,还是专业的工具在数据计算和处理能力上要更加便捷。
相关文章
-
王者荣耀:“一斩流”宫本火了,落地伤害6000,铭文出装是关键
在很久之前,宫本是一名天下无敌的刺客。宫本每次释放技能都会说:又无敌了,想挑战我的人请排好队,一个一个来。旧版宫本确实无敌,瞬发的大招落地秒杀脆皮,一技能的吹风能把敌人击飞,只要被宫本盯上,基本无人生还。无敌的宫本终于引起了策划的注意,然后进行一系列削弱,最后削不动了,直接重做改成战士。重做之后的宫
2025-04-19 15:22:53 -
亚瑟六元皮要不要入手?街头感形象帅气十足,技能带有虎啸声!
最近王者荣耀推出了亚瑟的动物派对秒杀皮肤,从建模来看,鲜明的虎纹元素和街头感相融合让亚瑟看上去更为帅气,皮肤的颜色比较简单,黄蓝为主,对比鲜明更显活力,而亚瑟的头发也变成了黄色挑染的,上面还有猫耳朵做装饰,非常时尚。这款皮肤的特效当中,一技能亚瑟举起盾牌防护只见盾牌上面会有蓝色的虎纹图案,剑刃上也有
2025-04-19 10:40:53 -
《冰雪传奇》装备获取与强化全攻略!
一、装备获取途径刷怪掉落:游戏中的怪物是装备的主要来源之一。玩家可以通过击败不同等级的怪物,有机会获得各种品质的装备。建议玩家选择适合自己等级的地图进行刷怪,以提高装备掉落率。任务奖励:完成游戏中的主线任务、支线任务和日常任务,都有可能获得装备奖励。这些任务不仅提供了丰富的经验,还能让玩家快速积累装
2025-04-19 09:41:51 -
王者荣耀:安琪拉是大神最爱,杀伤力惊人!
玩家们,在《王者荣耀》这个竞技场上,每一个英雄都有他们的强势期和弱势期。但是有一位特殊的法师英雄,她从头到尾都散发着强大的力量,让其他的英雄望尘莫及。这个超强的法师英雄就是我们的安琪拉。在王者荣耀的世界里,东皇太一和马超的强势无人不知,他们在战场上威风凛凛,一举一动都能影响整个战局。然而,当他们面对
2025-04-19 03:33:54 -
梦幻西游:土豪砸6亿鉴定灵犀玉被嘲讽,结果疯狂逆袭,打脸主播
大多数梦幻西游主播都是捧老板土豪的,有土豪玩家提供直播资源就各种拍马屁,恨不得老板梭哈所有金币去鉴定装备。当然,也有一些比较良心的主播,总是为老板着想,想方设法去为粉丝省钱赚钱。比如老王就是其中一位。近日,一位土豪怒砸6亿去鉴定灵犀玉,结果遭到老王的劝阻。老王认为这个玩法只会肉包子打狗,有去无回。然
2025-04-19 00:55:03 -
王者荣耀:深探《星之破晓》,新资格追加获取方式揭秘!
玩家今天,让我们一起来谈谈那款引爆了整个游戏圈子的英雄格斗手游——《星之破晓》。据说,这款游戏现在正在火热招募测试玩家,咱们今天就来详细了解下这个测试的相关事宜。首先,我们得清楚,《星之破晓》是王者荣耀的一个子游戏,虽然它以横版格斗为主要玩法,但是这次的新动作玩法“星魂战场”却是和我们熟悉的吃鸡模式
2025-04-18 17:09:31