数据库优化的四大方法

目录

一、前言

二、架构优化

分布式缓存

读写分离

水平切分

架构优化小结

三、硬件优化

四、DB优化

五、SQL优化

执行计划

SQL优化小结


一、前言

首先,我们看一下,数据库优化可以从那些方面入手:

图片

正如上图所示,数据库优化可以从架构优化,硬件优化,DB优化,SQL优化四个维度入手。

此上而下,位置越靠前优化越明显,对数据库的性能提升越高。我们常说的SQL优化反而是对性能提高最小的优化。

接下来我们再看看每种优化该如何实施。

二、架构优化

        一般来说在高并发的场景下对架构层进行优化其效果最为明显,常见的优化手段有:分布式缓存,读写分离,分库分表等,每种优化手段又适用于不同的应用场景。

分布式缓存

        有句老话说的好,性能不够,缓存来凑。当需要在架构层进行优化时我们第一时间就会想到缓存这个神器,在应用与数据库之间增加一个缓存服务,如Redis或Memcache。

图片

         当接收到查询请求后,我们先查询缓存,判断缓存中是否有数据,有数据就直接返回给应用,如若没有再查询数据库,并加载到缓存中,这样就大大减少了对数据库的访问次数,自然而然也提高了数据库性能。

        不过需要注意的是,引入分布式缓存后系统需要考虑如何应对缓存穿透、缓存击穿和缓存雪崩的问题。

读写分离

        一主多从,读写分离,主动同步,是一种常见的数据库架构优化手段。

        一般来说当你的应用是读多写少,数据库扛不住读压力的时候,采用读写分离,通过增加从库数量可以线性提升系统读性能。

图片

 

        主库,提供数据库写服务;从库,提供数据库读能力;主从之间,通过binlog同步数据。

        当准备实施读写分离时,为了保证高可用,需要实现故障的自动转移,主从架构会有潜在主从不一致性问题。

水平切分

        水平切分,也是一种常见的数据库架构优化手段。

        当你的应用业务数据量很大,单库容量成为性能瓶颈后,采用水平切分,可以降低数据库单库容量,提升数据库写性能。

图片

        当准备实施水平切分时,需要结合实际业务选取合理的分片键(sharding-key)。

架构优化小结

  1. 读写分离主要是用于解决 “数据库读性能问题”

  2. 水平切分主要是用于解决“数据库数据量大的问题”

  3. 分布式缓存架构可能比读写分离更适用于高并发、大数据量大场景。

三、硬件优化

        我们使用数据库,不管是读操作还是写操作,最终都是要访问磁盘,所以说磁盘的性能决定了数据库的性能。一块PCIE固态硬盘的性能是普通机械硬盘的几十倍不止。这里我们可以从吞吐率、IOPS两个维度看一下机械硬盘、普通固态硬盘、PCIE固态硬盘之间的性能指标。

吞吐率:单位时间内读写的数据量

  • 机械硬盘:约100MB/s ~ 200MB/s

  • 普通固态硬盘:200MB/s ~ 500MB/s

  • PCIE固态硬盘:900MB/s ~ 3GB/s

IOPS:每秒IO操作的次数

  • 机械硬盘:100 ~200

  • 普通固态硬盘:30000 ~ 50000

  • PCIE固态硬盘:数十万

        通过上面的数据可以很直观的看到不同规格的硬盘之间的性能差距非常大,当然性能更好的硬盘价格会更贵,在资金充足并且迫切需要提升数据库性能时,尝试更换一下数据库的硬盘不失为一个非常好的举措,你之前遇到SQL执行缓慢问题在你更换硬盘后很可能将不再是问题。

四、DB优化

        SQL执行慢有时候不一定完全是SQL问题,手动安装一台数据库而不做任何参数调整,再怎么优化SQL都无法让其性能最大化。要让一台数据库实例完全发挥其性能,首先我们就得先优化数据库的实例参数。

        数据库实例参数优化遵循三句口诀:日志不能小、缓存足够大、连接要够用。

数据库事务提交后需要将事务对数据页的修改刷( fsync)到磁盘上,才能保证数据的持久性。这个刷盘,是一个随机写,性能较低,如果每次事务提交都要刷盘,会极大影响数据库的性能。数据库在架构设计中都会采用如下两个优化手法:

  • 先将事务写到日志文件RedoLog(WAL),将随机写优化成顺序写

  • 加一层缓存结构Buffer,将单次写优化成顺序写

        所以日志跟缓存对数据库实例尤其重要。而连接如果不够用,数据库会直接抛出异常,系统无法访问。

五、SQL优化

        SQL优化很容易理解,就是通过给查询字段添加索引或者改写SQL提高其执行效率,一般而言,SQL编写有以下几个通用的技巧:

  • 合理使用索引

索引少了查询慢;索引多了占用空间大,执行增删改语句的时候需要动态维护索引,影响性能 选择率高(重复值少)且被where频繁引用需要建立B树索引;一般join列需要建立索引;复杂文档类型查询采用全文索引效率更好;索引的建立要在查询和DML性能之间取得平衡;复合索引创建时要注意基于非前导列查询的情况

  • 使用UNION ALL替代UNION

UNION ALL的执行效率比UNION高,因为UNION执行时需要排重;

  • 避免select * 写法

执行SQL时优化器需要将 * 转成具体的列;每次查询都要回表,不能走覆盖索引。

  • JOIN字段建议建立索引

一般JOIN字段都提前加上索引

  • 避免复杂SQL语句

提升可阅读性;避免慢查询的概率;可以转换成多个短查询,用业务端处理

  • 避免where 1=1写法

  • 避免order by rand()类似写法

RAND()导致数据列被多次扫描

执行计划

要想优化SQL必须要会看执行计划,执行计划会告诉你哪些地方效率低,哪里可以需要优化。通过explain sql 可以查看执行计划

SQL优化小结

这里给大家总结一下SQL优化的套路:

  1. 查看执行计划 explain sql

  2. 如果有告警信息,查看告警信息 show warnings;

  3. 查看SQL涉及的表结构和索引信息

  4. 根据执行计划,思考可能的优化点

  5. 按照可能的优化点执行表结构变更、增加索引、SQL改写等操作

  6. 查看优化后的执行时间和执行计划

  7. 如果优化效果不明显,重复第四步操作

  • 75
    点赞
  • 486
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
在 Android 应用开发中,为了提高应用的性能和内存使用效率,可以采取一些优化方法优化四大组件的内存使用。下面是各个组件的内存优化方法的详细解释: 1. Activity 内存优化: - 使用 `android:launchMode` 属性:根据业务需求合理设置 Activity 的启动模式,避免创建多个相同的 Activity 实例。 - 谨慎使用静态成员变量:避免在 Activity 中使用静态成员变量,因为静态变量会常驻内存,容易导致内存泄漏。 - 及时释放资源:在 `onDestroy()` 方法中释放不再需要的资源,如取消注册广播接收器、关闭数据库连接、停止耗时任务等。 2. Service 内存优化: - 使用启动方式合理的 Service:根据业务需求选择合适的启动方式,如使用 `startService()` 启动后及时调用 `stopSelf()` 或 `stopService()` 停止服务。 - 使用 IntentService:IntentService 是一种自动停止的服务,当任务执行完毕后会自动停止服务,避免忘记手动停止服务。 - 优化线程使用:合理使用线程池来管理线程,避免创建过多的线程,控制线程数量。 3. BroadcastReceiver 内存优化: - 尽早取消注册广播接收器:在不需要接收广播时,及时调用 `unregisterReceiver()` 方法取消注册广播接收器。 - 使用动态注册:根据需要动态注册广播接收器,避免静态注册导致无法及时取消注册。 4. ContentProvider 内存优化: - 懒加载数据:在需要时再去加载数据,避免一次性加载所有数据。 - 使用 CursorLoader:CursorLoader 是一个异步加载数据的工具类,可以在后台线程中加载数据,避免阻塞主线程。 此外,通用的内存优化方法包括: - 避免内存泄漏:合理使用对象的引用,避免对象持有不必要的引用导致内存泄漏。 - 使用资源合理:避免过度使用大量的 Bitmap、Drawable 等资源,及时释放不再使用的资源。 - 使用轻量级的数据结构:选择合适的数据结构来存储和处理数据,避免占用过多的内存。 这些方法可以帮助你优化四大组件的内存使用,提高应用的性能和内存效率。但是需要根据具体的业务需求和场景来选择适合的优化方法

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值