免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2466 | 回复: 5
打印 上一主题 下一主题

mysql 百万级更新性能 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2016-09-06 12:27 |只看该作者 |倒序浏览
本帖最后由 chenjintao_ii 于 2016-09-06 12:30 编辑

ne_elementparam 表有400多万条记录,我们的业务场景是这400万条数据每2小时更新一遍,我们有两种方法更新记录。

第一种方法是每次更新一条记录,重复400万次。更新一条的 explain 解释:

mysql> explain UPDATE ne_elementparam  SET epm_curvalue='', epm_updatetime = sysdate() WHERE epm_neid = 1 AND epm_objid = 2;           

+----+-------------+-----------------+-------+----------------------------+---------+---------+-------+------+-------------+

| id | select_type | table           | type  | possible_keys              | key     | key_len | ref   | rows | Extra       |

+----+-------------+-----------------+-------+----------------------------+---------+---------+-------+------+-------------+

|  1 | SIMPLE      | ne_elementparam | range | PRIMARY,PK_NE_ELEMENTPARAM | PRIMARY | 5       | const |    1 | Using where |

+----+-------------+-----------------+-------+----------------------------+---------+---------+-------+------+-------------+

1 row in set (0.00 sec)


第二种方法是先把记录插到临时表 tt_elementparam 中,插完400万条后,再一次性更新到 ne_elementparam 表中。下面是更新数据的 explain 解释:

mysql> explain UPDATE ne_elementparam ne, tt_elementparam tt SET ne.epm_curvalue=tt.epm_curvalue, ne.epm_updatetime = sysdate()  WHERE ne.epm_neid = tt.epm_neid AND ne.epm_objid = tt.epm_objid AND ne.epm_neid=1;

+----+-------------+-------+--------+----------------------------+---------+---------+---------------------------+------+-------------+

| id | select_type | table | type   | possible_keys              | key     | key_len | ref                       | rows | Extra       |

+----+-------------+-------+--------+----------------------------+---------+---------+---------------------------+------+-------------+

|  1 | SIMPLE      | ne    | ref    | PRIMARY,PK_NE_ELEMENTPARAM | PRIMARY | 5       | const                     |    1 | NULL        |

|  1 | SIMPLE      | tt    | eq_ref | PRIMARY                    | PRIMARY | 23      | const,das_uq.ne.EPM_OBJID |    1 | Using where |

+----+-------------+-------+--------+----------------------------+---------+---------+---------------------------+------+-------------+

2 rows in set (0.00 sec)


第一种方法中,type 字段为 range,而第二种方法中,type 字段为 ref 和 eq_ref,说明第二种方法效率更高。
我想问有没有人是用第二种方法来更新数据的?我实在想不通第二种方法怎么能提高效率啊~~



论坛徽章:
146
2015年亚洲杯之日本
日期:2015-04-28 13:32:012015年亚洲杯之朝鲜
日期:2015-05-06 10:16:442015年亚洲杯之日本
日期:2015-05-06 10:21:342015年亚洲杯纪念徽章
日期:2015-05-13 17:16:442015亚冠之北京国安
日期:2015-05-13 17:18:292015亚冠之鹿岛鹿角
日期:2015-05-13 17:19:062015亚冠之德黑兰石油
日期:2015-05-27 16:47:402015亚冠之塔什干棉农
日期:2015-05-28 15:24:122015亚冠之卡尔希纳萨夫
日期:2015-06-01 13:52:392015亚冠之柏斯波利斯
日期:2015-06-04 17:37:292015亚冠之阿尔纳斯尔
日期:2015-06-16 11:31:202015亚冠之塔什干火车头
日期:2015-06-23 10:12:33
2 [报告]
发表于 2016-09-06 15:50 |只看该作者
谢谢分享

论坛徽章:
93
2015年辞旧岁徽章
日期:2019-10-10 10:51:15CU大牛徽章
日期:2014-02-21 14:21:56CU十二周年纪念徽章
日期:2020-10-15 16:55:55CU大牛徽章
日期:2014-02-21 14:22:07羊年新春福章
日期:2019-10-10 10:51:39CU大牛徽章
日期:2019-10-10 10:55:38季节之章:春
日期:2020-10-15 16:57:40ChinaUnix元老
日期:2019-10-10 10:54:42季节之章:冬
日期:2019-10-10 10:57:17CU大牛徽章
日期:2014-02-21 14:22:52CU大牛徽章
日期:2014-03-13 10:40:30CU大牛徽章
日期:2014-02-21 14:23:15
3 [报告]
发表于 2016-09-06 17:14 |只看该作者
谢谢分享,比较少想到第二种方式,不知道是不是和表的索引结构有关系。

论坛徽章:
223
2022北京冬奥会纪念版徽章
日期:2015-08-10 16:30:32操作系统版块每日发帖之星
日期:2016-05-10 19:22:58操作系统版块每日发帖之星
日期:2016-02-18 06:20:00操作系统版块每日发帖之星
日期:2016-03-01 06:20:00操作系统版块每日发帖之星
日期:2016-03-02 06:20:0015-16赛季CBA联赛之上海
日期:2019-09-20 12:29:3219周年集字徽章-周
日期:2019-10-01 20:47:4815-16赛季CBA联赛之八一
日期:2020-10-23 18:30:5320周年集字徽章-20	
日期:2020-10-28 14:14:2615-16赛季CBA联赛之广夏
日期:2023-02-25 16:26:26CU十四周年纪念徽章
日期:2023-04-13 12:23:10操作系统版块每日发帖之星
日期:2016-05-10 19:22:58
4 [报告]
发表于 2016-09-06 17:33 来自手机 |只看该作者
临时表跑在内存,性能刚刚的

论坛徽章:
0
5 [报告]
发表于 2016-09-09 07:52 |只看该作者
回复 2# 王楠w_n

第二种方法中,不知道是不是 MySQL 在做批量更新的时候对性能进行优化了。因为按我的理解,临时表有100百万数据需要更新到 elementparam 表时,过程应该是这样的:临时表拿到一条数据 -> 从 elementparam 表的100百万数据中查到对应的数据 -> 更新数据。此流程走100百万遍。

而第一种方法,是“从 elementparam 表的100百万数据中查到对应的数据 -> 更新数据。此流程走100百万遍”,少一步去临时表拿数据的过程(也少了插入临时表的过程)。

理论上来说第一种方法比第二种更高效,但真实环境效果相反了。

论坛徽章:
0
6 [报告]
发表于 2016-09-09 07:55 |只看该作者
回复 4# action08

多谢指点,刚看了下临时表的属性,我现在的临时表只是业务上的临时,数据还是会写磁盘的。替换成 MySQL 的临时表,估计效率会有一定提升。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP