在Windows 10、Yii2.0、MySQL,实现组合唯一索引,解决高并发情形下的数据重复写入问题的流程

1、在生产环境上出现了技能数据表中的数据被重复写入的情况,如图1

图1

2、但是在写入之前是做了记录是否存在的判断的,game_code与team_code的组合应该是唯一的才是了,如图2

图2

3、建议先查看网址:http://www.shuijingwanwq.com/2017/04/18/1568/ ,由于post所请求的程序的机制,ab的post文件仅支持一次请求的参数,无法模拟出真实的场景,真实的场景应该是每一次请求的参数是不同的,因此,无法复现出生产环境上的bug

4、通过在注释掉判断记录是否存在的条件,实现每次请求,数据皆是插入了,模拟出一场比赛存在多条技能数据的场景,如图3

图3

5、基于postman,连续执行了两次参数不一样的请求,最终MySQL中存在4条记录,如图4、5

图4

图5

6、为了避免此种情况的出现,决定在MySQL中修改表结构,删除表中的4条记录,将game_code与team_code添加为组合唯一索引,在phpMyAdmin中选择game_code与team_code,点击唯一按钮,如图6

图6

7、SQL 语句已成功运行,添加组合唯一索引成功,如图7、8

ALTER TABLE `kq_game_team_skill` ADD UNIQUE( `game_code`, `team_code`);

图7

图8

8、基于Gii,重新生成新的模型文件,如图9

图9

9、在rules()中新插入了一行,如图10

图10

10、基于postman,连续执行了两次参数不一样的请求,执行第2次的时候报错,如图11

图11

11、报错信息已经经过处理,打印出原始的报错信息,如图12

图12

12、原始的报错信息:$gameTeamSkill->errors 如下,如图13

图13

13、此时查看MySQL,一场比赛所对应的技能数据只存在2条记录,一个球队只存在1条了的,符合预期,如图14

图14

14、即使通过ab压测没有模拟出生产环境上的bug,但是通过对于MySQL添加组合唯一索引的实现,理论上是防止了此bug的重现了的,只是code显示的并非为数字,有待优化处理了的

 

永夜