MySQL高级 | 用存储过程、定时器、触发器来解决数据分析问题

很多做开发、数据库相关工作的小伙伴可能经常会用到 mysql 的存储过程、定时器、触发器这些高级功能,但是做数据分析或者数据处理,我们也需要掌握这些技能,来解决特定的业务问题。比如:做自动化报表,如果数据需要每天实时更新(增量爬虫)、定时计算某个业务指标 、想要实时监控数据库表中的数据增、删、改情况等。

文章大纲

一、存储过程

1、啥是存储过程,有什么用?
  • 过程 :将若干条 sql 语句封装起来,起个名字
  • 存储过程 : 我们将此过程存储在数据库中
    有点类似于编程中用到的函数,区别是函数有返回值,而过程没有返回值,相同点是将代码封装可复用,可传参,调用即可执行。
  • 好处:① 代码封装可复用 ② 可以接收、返回参数 ③ 减少网络交互、提升效率
2、存储过程如何使用
创建
create procedure 名称()
begin
  sql语句;
end
查看
show procedure status;
调用
call 名称();
删除
drop procedure if exists 名称;
3、存储过程中的变量
变量的种类和定义

在sql中变量分为两种:① 系统变量:@@ ② 自定义变量:@


存储过程是可以编程的,意味着可以使用变量、表达式、控制结构,在存储过程中,声明变量用 declare

格式:declare 变量名 变量类型 【default 默认值】
变量运算与控制结构

变量的赋值,有两种方式:① set 变量名 = 值 ② set 变量名 := 值


if|else 控制结构语法格式

if  条件1 then
  sql 语句;
else if 条件2 then
  sql 语句;
else
  sql语句;
end if
存储过程中的参数传递

为了让存储过程更加灵活,可以传递参数,参数分为三种:① in :参数作为输入,调用时传入
② out : 参数作为输出,可以作为返回值 ③ inout : 参数即可传入又可输入

格式: in|out|inout 参数名 参数类型
存储过程中使用循环

while 循环格式 :

while 条件 do
  sql 语句;
end while

打印 1 - 100 之和

带输入参数n,求1-n之和

要求带输入参数n,和输出参数 total ,求1-n之和

要求 age 既是输入又是输出变量,传入一个年龄,就增加20

二、定时器

1、啥是定时器,怎么用?

所谓定时器,就是定时地去运行指定的函数和代码,mysql 的定时器就是 mysql 的事件
在开发过程中经常会遇到这样一个问题:每天或每月须定时去执行一条sql语句或更新或删除除数据。在我不了解mysql定时器时,是用 pyhton 程序代码去操作数据表,再将py程序,放到服务器,跑定时任务。现在用定时器,完全可以在数据层面操作了,非常方便。

语法结构
create event [if not exists] 事件名
[definer = user] 可选参数。执⾏事件的⽤⼾,不指定默认就是当前⽤⼾
on schedule 定时时间设置。定义事件执⾏的频率,可以指定具体时间也可以周期性执行
[on completion [not] preserve ] 可选参数。默认是not,表⽰时间过期后会⽴即删除(注意不是不激活);on completion preserve 表⽰时间过期后会继续保留
[enable | disable | disable on slave] 可选参数。默认enable。事件激活、不激活、从服务上不激活(事件在主服务商创建并赋值到从服务器上,仅在主服务上执行)
[comment "注释"] 可选参数。
do 事件内容 定义事件的sql语句,如果语句有多⾏需要⽤ begin end 括起来
指定时间定时执行

at子句:这里要求是timestamp时间格式,⼀般格式是“时间点 + interval 时间单位”。表示在什么时间节点执行,例如:current_timestamp + interval 2 minute

要求:两分钟后往 event_test 表插⼊⼀条语句" 事件启动了"



注意: 因为参数默认是on completion not preserve,事件运行完成后会自动删除

周期时间定时执行:

every子句:格式是“数字+时间单位”,表示时间周期,例如:1 hour / 2 minute / 3 second
starts子句:可选,跟上 timestamp 值,表示事件开始的时间点,如果没有指定就是当前时间
ends子句:可选,跟上 timesatamp 值,表示停止执行的时间,如果没有ends表示无限执行

要求:新建数据表 event_test,每分钟往里面插⼊⼀条数据,到5分钟结束

注意事项

1、需要激活事件,事件才会被执行,show events才可以查看。⼀个是全局参数开启,⼀个是事件的开启。

SET GLOBAL event_scheduler = 1;
设置事件的状态为enable:
ALTER EVENT event_name ON COMPLETION PRESERVE ENABLE; 开启
ALTER EVENT event_name ON COMPLETION PRESERVE DISABLE; 关闭

  1. 关掉了navicat,事件不会关闭,关闭了mysql服务器才会被关闭。
  2. 多语句执行的时候,可能需要修改结束分隔符,比如: delimiter $
  3. 如果事件的开始时间已经过去了,虽然创建语句不会报错,但是事件不会被创建以及执行。
  4. 事件⾥⾯不能嵌套事件,但是存储过程里面可以使用事件
  5. 在事件中使用select、show等返回结果语句没有意义,但是可以用select into、insert into等
    存储结果的语句
  6. 注意不要短周期内重复事件调度,不然数据会有问题。例如每分钟执行100w行数据,那这个会有问题,如果实在是需要那这个时候可以使用行锁、表锁来进行。
  7. 事件⽆法传递参数,但是可以用事件里面的参数使用存储过程
定时器可以结合存储过程

现在用定时器,就可以在数据层面操作,定时去执行sql 语句或一组 sql 语句(存储过程),设置好定时任务,可通过navicat —— 其它——事件,查看到当前事件的定义,计划,当然也可以手动完成上述操作。


三、触发器

1、啥是触发器,应用场景是 ?

触发器是一类特殊的事务,可以监视数据操作(数据表的变更日志),包括 insert | update | delete,并触发相关操作 insert | update | delete,运用触发器,不仅能简化程序,又可以增加程序的灵活性。

应用场景① :当向一张表中添加或删除数据时,需要在相关表中进行同步操作,比如:当一个订单产生时,订单所购的产品的库存量相应减少。
应用场景② :当表中某列数据的值与其他表中的数据有联系时,比如:某客户进行欠款消费,可以在生成订单时,设计触发器判断该用户的累计欠款是否超过最大限度。
应用场景③ :跟踪某张表时,比如当有新订单产生时,需通知相关人员进行处理,这时可以在订单表中添加触发器加以实现。

2、触发器如何使用
创建

触发器只支持行级触发(每一行受影响,触发器都执行,叫作行级触发器),不支持语句级触发。

Create trigger 触发器名称
before/after  
insert/update/delete
on 表名 for each row #行级触发器
Begin 
    trigger_state;
end
查看
Show triggers;
删除
Drop trigger 数据库.触发器名称;
要求:现有商品表 goods,订单表 orders,当下一个订单时,商品要相应减少(买几个商品,就少几个库存), 分析如下:

监视谁:orders
监视动作:insert
触发时间:after
触发事件:update

CREATE TABLE goods(gid INT,name VARCHAR(10),num SMALLINT);
CREATE TABLE ord(oid INT ,gid INT, buy_num SMALLINT)
INSERT INTO goods VALUES (1,'cat',20),(2,'dog',90),(3,'pig',26);
查看商品表
创建触发器,并查看

发现这样写触发器,并不灵活
3、触发器引用行变量

使用别名old、new来引用触发器中发生变化的记录内容。注意:


引用行变量
要求:删除一个订单时,商品要退回,库存量要还原(删)
要求:订单表中的数量3 要求改到2,并且让商品表的库存量也变化(改)
要求:假如现在剩余 26 只pig,但是客户下订单买27只,能否预防,能否将buy_num > num 时,将buy_num 自动改为 num(深入理解before 和after的区别)


上面跟大家介绍了,如何数据分析工作中,应用 MySQL 的存储过程、定时器、触发器来实现自动化更新数据。当然,用 python 或其他编程语言也能实现,个人认为在数据层面操作,更加简单、高效、稳定。具体还要看你当下的业务场景。希望通过本文能为你提供一个解决问题的思路 。 欢迎点赞支持 ~

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

推荐阅读更多精彩内容