在 MySQL 创造类似 PipelineDB 的流视图(continuous view)
公司的系统采用的是 Google Cloud SQL 提供的 MySQL 数据库,由于历史原因,数据库成本极高,需要对它进行优化缩减成本。
相比 PostgresSQL,MySQL 主要缺少以下特性,导致优化难度极高:
1. 缺少部分索引。部分索引可以将一亿行数据中活跃的那部分数据(往往只有几百万行)隔离出来。
2. 缺少计算索引。MySQL 的索引不能是表达式。
3. 不支持并行。MySQL不能让一个查询分布到多个核,从而缩短查询执行时间。由于只能用一个核,运行时间较长的的SQL很容易造成锁超时,各个服务日志里充满了锁超时错误。
弃暗投明换PG或者上Lambda架构等做法都过于复杂,不考虑。
在优化过程中我发现,如 MySQL 能支持类似流视图,很多主要语句的执行问题可以迎刃而解。
什么是流视图?
基于 Postgresql 的 PipelineDB 支持 Continuous View,使数据库具备了类似流式计算的效果。
流计算风云再起 - PostgreSQL携PipelineDB力挺IoT(物联网), 大幅提升性能和开发效率-云栖社区-阿里云
采用速度层对新增数据执行计算后放入快速存储,而不是在查询时开始计算,这是 Lambda 之类大数据架构的精髓思想。流视图将这套架构包装为一个直观易懂的数据库新事物,和关系理论又不相违,是很巧妙的设计。显然,流视图在很多场景都可以取代 Spark 之类大数据架构,为用户提供更实时的统计查询服务。
羡慕是没用的,没有就自己搞,如何实现流视图?
流视图的原理可以想见:流视图本身是一个表,程序侦听视图相关的各个表的变化,将变化更新到相应的流视图的数据行。
侦听数据变化是实现流视图的必备基础。
在SQL中侦听变化的方式主要是触发器。给每个表建一个触发器很不明智,干出来也不漂亮。我想到一个很好的方案:BINLOG。找了一下,目前已有若干支持 binlog 转 MQ 的方案。我选择了 canal,备选 maxwell。在实施过程中发现,canal 往 MQ 推送的消息没有都是字符串且不支持枚举,而 maxwell 虽然消息很棒,还支持 js 做 filter,但是经常出错崩溃,CPU高,处理不了太大的 binlog,最终还是采用了 canal。相比来说 canal 一直在阿里实用,运行非常稳定,CPU 耗费低,从来不出错。
有了 BINLOG,后面就是搞发明创造了,做架构写代码,最终成果如下:
class HV_user_gender(HotView):
VIEW_SQL = '''
SELECT a.id, a.name, b.gender,b.age FROM a, b ON a.id = b.id WHERE b.age < 22
%s
''' def __init__(self, view_db: Engine, logger: logging.Logger, cache_db:Engine = None):
meta = MetaData()
mapping = [
TableMapping('a', meta,
ColumnMapping('id', cond_alias="a.id", view_alias="id", is_fixed=True, primary_key=True),
ColumnMapping('name', view_alias="name", is_fixed=False),
trace_insert=True), TableMapping('b', meta,
ColumnMapping('id', cond_alias="b.id", is_output=False, primary_key=True),
ColumnMapping('gender', view_alias="gender", is_fixed=True),
ColumnMapping('age', view_alias="age", is_fixed=False),
predicate=lambda row: int(row['age']) < 22
)
]
super().__init__(view_db, 'hot_user_gender', topics=['a', 'b'],
logger=logger,
primary_key="id",
view_sql=HV_user_gender.VIEW_SQL,
mapping=mapping,cache_db= cache_db) def init_view(self):
sql = '''
insert into hot_user_gender(id, name, gender, age)
''' + (self.view_sql % '')
r = self.view_db.execute(sql)
self.logger.info('init view insert %s items' % r.rowcount)
这里实现了一个基于 a b 两表的联合查询,b.age < 22 过滤条件的流视图。当 b.age 有变化时,如 b.age >= 22,视图行自动删除,如 b.age < 22,相应视图行更新。a 有增删时,视图行同步变化行。
该视图的代码不到100行,声明式代码风格。
目前对一个常用查询做了流视图改造,CPU 立即降低 15%!随处可见的 Read Lock timeout 之类异常也不见了。
流视图的改造成本很低,对于用 mybaits/ibatis 之类框架的程序来说,只要调整相关 SQL 代码,将原来的复杂查询挪到 python 中,替换成单表流视图的名称即可,原有的技术堆栈还能继续使用。什么难出的报表、加载耗时漫长的 dashboard,homepage,等等,都可以通过流视图改为实时秒开,不但用户体验上升,数据库压力也骤然下降,实乃 SQLer 的灵丹妙药,MySQL 这种弱爆了的数据库也有枯木逢春之感。
在 MySQL 创造类似 PipelineDB 的流视图(continuous view)的更多相关文章
- MySQL 系列(三)你不知道的 视图、触发器、存储过程、函数、事务、索引、语句
第一篇:MySQL 系列(一) 生产标准线上环境安装配置案例及棘手问题解决 第二篇:MySQL 系列(二) 你不知道的数据库操作 第三篇:MySQL 系列(三)你不知道的 视图.触发器.存储过程.函数 ...
- 我的MYSQL学习心得(十一) 视图
我的MYSQL学习心得(十一) 视图 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心得(四) 数据 ...
- MySQL基础(二):视图、触发器、函数、事务、存储过程
一.视图 视图是一个虚拟表(非真实存在),其本质是[根据SQL语句获取动态的数据集,并为其命名],用户使用时只需使用[名称]即可获取结果集,并可以将其当作表来使用. 视图和上一篇学到的临时表搜索类似. ...
- python进阶10 MySQL补充 编码、别名、视图、数据库修改
python进阶10 MySQL补充 编码.别名.视图.数据库修改 一.编码问题 #MySQL级别编码 #修改位置: /etc/mysql/mysql.conf.d/mysqld.cnf def ...
- Mybatis拦截器 mysql load data local 内存流处理
Mybatis 拦截器不做解释了,用过的基本都知道,这里用load data local主要是应对大批量数据的处理,提高性能,也支持事务回滚,且不影响其他的DML操作,当然这个操作不要涉及到当前所lo ...
- MySQL实现类似Oracle的序列
MySQL实现类似Oracle的序列 2013-10-22 10:33:35 我来说两句 作者:走过的足迹 收藏 我要投稿 MySQL实现类似Oracle的序列 Oracl ...
- 【转】MYSQL入门学习之十:视图的基本操作
转载地址:http://www.2cto.com/database/201212/176775.html 一.视图的基本介绍 www.2cto.com 视图是虚拟的表.与包含数据 ...
- mysql统计类似SQL语句查询次数
mysql统计类似SQL语句查询次数 vc-mysql-sniffer 工具抓取的sql分析. 1.先用shell脚本把所有enter符号替换为null,再根据语句前后的字符分隔语句 grep -Ev ...
- pipelinedb学习笔记 - 1. Continuous Views (连续视图)
Continuous Views 一.Continuous Views 英文直译过来叫连续视图, 在pipelindb中是被定义为专门用来展示 Stream中数据用的.例如:Stream中有一些用户信 ...
- mysql基础教程(四)-----事务、视图、存储过程和函数、流程控制
事务 概念 事务由单独单元的一个或多个SQL语句组成,在这 个单元中,每个MySQL语句是相互依赖的.而整个单独单 元作为一个不可分割的整体,如果单元中某条SQL语句一 旦执行失败或产生错误,整个单元 ...
随机推荐
- USB LPM状态
USB的u0.u1.u2和u3代表不同的电源状态或低功耗状态,主要用于USB 3.0及其更高版本,目的是在不活动时减少功耗,同时保证设备能够迅速恢复到工作状态.这些状态的具体作用如下: U0(Acti ...
- python中模块的概念以及常用模块
模块的概念:为了减少常用代码重复输入,同时方便多人协作,python中定义了模块一词. 模块是一些函数的集合.在python中可以使用import导入某一模块的单个或者所有功能. python中的模块 ...
- NeuVector 会是下一个爆款云原生安全神器吗?
近日一则<SUSE 发布 NeuVector:业内首个开源容器安全平台>的文章被转载于各大 IT 新闻网站.作为 SUSE 家族的新进成员,在 3 个月后便履行了开源承诺,着实让人赞叹.那 ...
- 在centos7系统上安装docker环境
基于 CentOS 7 安装 官网安装参考手册:https://docs.docker.com/install/linux/docker-ce/centos/ 确定你是CentOS7及以上版本 [ro ...
- 使用最小二乘法进行线性回归(Python)
已知测得某块地,当温度处于15至40度之间时,数得某块草地上小花朵的数量和温度值的数据如下表所示.现在要来找出这些数据中蕴含的规律,用来预测其它未测温度时的小花朵的数量. 测得数据如下图所示: imp ...
- Dash 2.18.2版本更新:模式匹配回调性能大提升
本文示例代码已上传至我的Github仓库:https://github.com/CNFeffery/dash-master Gitee同步仓库地址:https://gitee.com/cnfeffer ...
- 3.17 Linux移动或重命名文件和目录(mv命令)
mv 命令(move 的缩写),既可以在不同的目录之间移动文件或目录,也可以对文件和目录进行重命名.该命令的基本格式如下: [root@localhost ~]# mv [选项] 源文件 目标文件 选 ...
- .NET Core 泛型底层原理浅谈
简介 泛型参考资料烂大街,基本资料不再赘述,比如泛型接口/委托/方法的使用,逆变与协变. 泛型好处有如下几点 代码重用 算法重用,只需要预先定义好算法,排序,搜索,交换,比较等.任何类型都可以用同一套 ...
- C++进阶知识汇总
知识来源:https://www.imooc.com/learn/1305 二进制在计算机中的意义: 计算机如何存负整数: 原码:符号位变为1 反码:除符号位其余取反 补码:=反码+1 是-7的表示方 ...
- 基于Java+SpringBoot心理测评心理测试系统功能实现五
一.前言介绍: 1.1 项目摘要 心理测评和心理测试系统在当代社会中扮演着越来越重要的角色.随着心理健康问题日益受到重视,心理测评和心理测试系统作为评估个体心理状态.诊断心理问题.制定心理治疗方案的工 ...