热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

MySQL语句性能优化

MySQL概述1.数据库设计3范式2.数据库分表分库—会员系统()水平分割(分页如何查询)MyChar、垂直3.怎么定位慢查询———————数据库索引的优化、索引原理SQL语句



MySQL概述
1.数据库设计 3范式
2.数据库分表分库—会员系统() 水平分割(分页如何查询)MyChar 、垂直
3.怎么定位慢查询
———————
数据库索引的优化、索引原理
SQL语句调优
数据库读写分离–MyChar
———————
分组 having
存储过程、触发器、函数
存储过程:写了一块sql语句,类似Java中方法,只需调用传参数,弊端:sql语句是写死的,不好灵活改变。
mysql(免费、开源)oracle(收费)
mysql 分页 limit 。oracle:rownum 伪列

MySQL优化方案
1.数据库设计要合理(3F)
2.添加索引(普通索引、主键索引、唯一索引、全文索引)底层:B-Tree和B+Tree 和二叉树算法一样,减少全表扫描
3.分表分库技术(取模分表、水平分割、垂直分割)
4.读写分离
5.存储过程
6.配置MySQL最大连接数 my.ini
7.mysql服务器升级
8.随时清理碎片化
9.sql语句调优 核心

数据库三大范式
数据库设计
什么是数据设计(减少冗余量、3F)
什么事数据库3范式
1F 原子约束 表示每列不可再分
   id  name     sex   address
  1   zhangsan  0    北京

  是否保证原子(看业务)

2F 保证唯一,主键
   id orderNum(唯一) name     sex   address
  1  123              zhangsan  0    北京
  订单表中,是否用id作为订单号?不允许。
  项目内部rpc远程调用 大多数是使用id进行通讯的,给外部看的是orderNum,保证系统安全性。
 
  分布式系统解决并发生成订单号
  怎么保证抢票中,订单号不会重复生成?怎么保证订单的幂等性(幂等就是不重复)?
  提前将订单号生成号,存在redis中,需要是直接去resis中去取;分布式锁

3F 不要有冗余数据  classId  className重复,这个表只存classId即可
   id  name     sex   address  classId  className
  1   zhangsan  0    北京     1        一班
  2   lisi    0    北京     2        二班
  2   wangwu   0    北京     1        一班

  从新建张表:
 classId  className
 1        一班
 2        二班
 
 注意:不一定完全要遵循第3F。

MySQL分库分表
什么时候分库:
    电商项目当中,将一个项目拆分,拆分成多个小项目,每个小的项目有自己单独的数据库,互不影响。–垂直分割。
  会员数据库、订单数据库、支付数据库

什么时候分表:
    水平分割,分表规则,根据业务需求。存放日志 根据年分表、手机号 根据前三位分表 136 135 135
  水平分割(取模算法)

  user表  分成三张表 
  id  name   address
  1   张三   北京
  2   李四   北京
  3   王五   北京
  4   赵六   北京
  5   小明   北京
  6   小红   北京

  怎样将6条数据存放在三张不同的表中,怎样非常均匀? 取模算法
  表: user0 user1 user2  
  第一条数据:1%3=1 ,放user1表。  
  第二条数据:2%3=2 ,放user2 表。   
  第三条数据:3%3=0 ,放user0表。依次类推

  实现取模分表算法:三张表id不能自动增长,需要专门有一张表存放userId,给赋值过去。

水平分割取摸算法案例
好处:非常均匀的分配
怎样查在哪里表?找id为6在哪个表 6%3=0 在user0表找

Demo:

create table uuid(
 id int unsinged primary key auto_increment,
)engine=myisam charset utf8;

@Service
public class UserService{
 
 @Autowired
 private JdbcTemplate jdbcTemplate;

  /**
  * 生成用户信息
  */
 public String regit(String name,String pwd){
  //1.生成userId
  String insertUuidSql=”insert into uuid values (null)”;
  // 这里一般用直接返回主键id,不用下面的查询
    jdbcTemplate.update(insertUuidSql);
  //select last_insert_id()表示查询最近的主键ID的意思
    Long userId = jdbcTemplate.queryForObject(“select last_insert_id()”,requiredType,Long.class);
  //2.存放具体哪张表中
  String tableName = “user”+userId%3;
  //3插入到具体表中
  String insertUserSql = “insert into “+tableName+” values(“+userId+”,”+userName+”,”+pwd+”)”;
  System.out.println(insertUserSql);
  jdbcTemplate.update(insertUserSql);
  return “success”;
 }

 /**
  * 根据id查询
  */
 public String get(Long userId){
  //1.存放具体哪张表中
  String tableName = “user”+userId%3;
  String selectUserSql=”select name from “+tableName+”where id=”+userId;
  String name = jdbcTemplate.queryForObject(selectUserSql,String.class);
  return name;
 }
}

@RestController
public class UserController{

 @Autowired
 private UserService userService;

 @RequestMapping(“/regit”)
 public String regit(String name,String pwd){
   return userService.regit(name,pwd);
 }

 @RequestMapping(“/getUser”)
 public String get(Long userId){
   return userService.get(userId);
 }
}

@SpringBootApplication
public class App{
 public static void main(String[] args) {
     SpringApplication.run(App.class,args);
  }
}

分表之后有什么缺点?
1.怎么分页查询
2.查询非常受限制,例如查性别男要查三张表
3.取模算法 如果表发生改变了,表要重新分,打乱了 使用阿里云RDS数据库

先主表存放所有数据 ,根据业务需求进行分表。

如何定位慢查询

什么是慢查询?
MySQL默认慢查询是10秒,如果10秒没有响应回来就是慢查询,一般在生产环境设置为1秒

慢查询都会有日志存放
show status 命令

《MySQL语句性能优化》

如何修改慢查询
–查询慢查询次数
show status like ‘slow_queries’;
–查询慢查询时间
show variables like ‘long_query_time’;
–修改慢查询时间
set long_query_time=1; 表示修改慢查询时间为1秒

如何将慢查询定位到日志中
在默认情况下,MySQL不会记录慢查询,需要在启动MySQL的时候,指定记录慢查询才可以。

编辑配置文件/etc/my.cnf加入如下内容
[mysqld]
slow_query_log = ON
slow_query_log_file = /var/lib/mysql/test-10-226-slow.log
long_query_time = 1

修改配置后重启mysql
systemctl restart mysqld
mysql -uroot -p

MySQL索引概述

为什么要索引?提高查询效率
为什么索引能够提高查询效率–索引实现原理 折半查找

索引分类:
主键索引 — primary key
添加主键索引方式:
1. 创建表的时候添加:id int unsinged primary key auto_increment,
2. 如果创建表的时候没有添加:alert table 表名 add primary key (列名);

删除主键索引:alert table 表名 drop primary key;

C:\ProgramData\MySQL\MySQL Server 5.6\data\test 下:

*.frm 表结构文件
*.MYD 数据结构文件
*.MYI 索引文件

唯一索引
组合索引
全文索引
普通索引

索引底层实现原理

索引底层采用b-tree 折半查找又叫二分查找

《MySQL语句性能优化》

没有索引是全表扫描–索引 减少全表扫描

索引 b-tree 首先生成索引文件

索引有什么缺点?增加、删除 索引文件也需要更新

普通索引与唯一索引

唯一索引:关键字unique
create table 表名称(
 id int primary key auto_increment, // 主键索引
 name varchar(32) unique;      //唯一索引
)

注意:unique字段可以为null,并可以有多个null,但是如果有具体内容,则不能重复。
唯一索引用的不多,被主键索引代替了。

普通索引:
create table 表名(
 id int unsigned,
 name varchar(32)
)

创建普通索引:creat index 索引名 on 表 (列1,列2);

creat index index_# on # (name);

–执行计划 查看有没有使用索引
explain select * from # where name=”zhangsan”;
查询出type为ref表示使用索引 all是全表扫描

全文索引:

create table #(
 id int primary key,
 title varchar(200),
 body TEXT,
 FULLTEXT(title,body)  // 创建全文索引
)engine = innodb;

模糊查询:select * from # where body like ‘%张%’ ; 错误用法,索引不会生效

使用执行计划查看是否使用索引:
explain select * from # where body like ‘%张%’ ;

使用索引查询: select * from # where match(title,body) against(‘张’)

使用执行计划查看是否使用索引:
explain select * from # where match(title,body) against(‘张’)

使用全文索引的时候:
不要like
企业实际中不会采用表的全文索引。全文索引有非常大的缺点,InnoDB(数据库存储引擎)中不支持全文索引

 
SQL语句优化方案总结

索引优缺点:
优点:提高程序效率
缺点:增加、删除慢,索引文件需要更新,增加内存

什么字段适合加索引?
查询次数比较多,值有非常多的不同。

建立索引场景:
建立索引的时候,where条件需要查询的,并且值非常多的不同的。唯一几个值(sex:0,1),不需要建立索引。

索引注意事项,sql调优部分:

–创建主键索引
alert talbe 表名 add primary key (列名);
–创建组合索引文件
alert table 表名 add index my_ind(列1,列2);

注意:
1.对于创建的多列索引,如果不使用第一部分,则不会创建索引。
explain select * from 表名 where 列1 = ‘#’;  — 使用索引
explain select * from 表名 where 列2 = ‘bbb’;  — 使用全表扫描

2.使用索引的时候,不要使用like’%%’ ,这样会全表扫描,使用like,开头不要%
explain select * from 表名 where 列1 like ‘%#%’;  — 使用全表扫描
explain select * from 表名 where 列1 like ‘#%’;   — 使用索引

3.使用or,条件都必须加索引,只要有一个条件不加索引,就会全表扫描
explain select * from 表名 where 列1 = ‘#’ or 列3 =’ccc’;  — 使用索引

4.判断是否为null ,使用is null 不要=null
5.使用group by 分组不会采用索引,会全表扫描。
explain select * from 表名 group by 列2;
6.分组需要效率高,禁止排序(分组默认排序)
explain select * from 表名 group by 列2 order by null ;
7. select * from # where userId>=100 和 select * from # where userId>100 哪个效率高
不要使用大于等于,会判断两次全表扫描

8. in和notin 加上索引后也不会使用索引,能用between就不要用in
9.查询量非常大时,采用缓存、分表、分页。

MySQL存储引擎区别

MySQL存储引擎:innodb/ myisam/ memory

主流:innodb 支持事务机制

innodb与myisam区别:
批量添加–myisam效率高
innodb–事务机制非常安全

锁机制:
myisam是表锁
innodb是行锁,不会影响整个表

数据结构:
myisam 支持全文检索,一般不用数据库自带的全文检索。

都支持b-tree数据结构

索引缓存 都支持。

Myisam注意事项

创建myisam引擎表结构:
create table ccc(id int,name varchar(32))engine=myisam;
添加数据:
insert into ccc values(1,’a’);
insert into ccc values(2,’b’);
insert into ccc values(3,’b’);
insert into ccc select id,name from ccc;

删除id为3的所有数据
delete from ccc where id = 3;

会发现:ccc.MYD文件大小没有改变,缺点:没有真正的删除,删除后能够非常快的恢复过来

真要删除使用:optimize table ccc; 表示对myisam进行整理,清了碎片化。
企业实际中是不会物理删除数据的。数据迁移。


推荐阅读
  • 从0到1搭建大数据平台
    从0到1搭建大数据平台 ... [详细]
  • 针对MySQL Undo空间满载及Oracle Undo表空间溢出的问题,本文详细探讨了其原因与解决策略。首先,通过启动SQL*Plus并以SYS用户身份登录数据库,查询当前数据库的UNDO表空间名称,确认当前状态。接着,分析导致Undo空间满载的常见原因,如长时间运行的事务、频繁的更新操作等,并提出相应的解决方案,包括调整Undo表空间大小、优化事务管理、定期清理历史数据等。最后,结合实际案例,提供具体的实施步骤和注意事项,帮助DBA有效应对这些问题。 ... [详细]
  • 本文介绍如何使用 Python 的 DOM 和 SAX 方法解析 XML 文件,并通过示例展示了如何动态创建数据库表和处理大量数据的实时插入。 ... [详细]
  • 在什么情况下MySQL的可重复读隔离级别会导致幻读现象? ... [详细]
  • MySQL Decimal 类型的最大值解析及其在数据处理中的应用艺术
    在关系型数据库中,表的设计与SQL语句的编写对性能的影响至关重要,甚至可占到90%以上。本文将重点探讨MySQL中Decimal类型的最大值及其在数据处理中的应用技巧,通过实例分析和优化建议,帮助读者深入理解并掌握这一重要知识点。 ... [详细]
  • 在CentOS 7环境中安装配置Redis及使用Redis Desktop Manager连接时的注意事项与技巧
    在 CentOS 7 环境中安装和配置 Redis 时,需要注意一些关键步骤和最佳实践。本文详细介绍了从安装 Redis 到配置其基本参数的全过程,并提供了使用 Redis Desktop Manager 连接 Redis 服务器的技巧和注意事项。此外,还探讨了如何优化性能和确保数据安全,帮助用户在生产环境中高效地管理和使用 Redis。 ... [详细]
  • 在《Cocos2d-x学习笔记:基础概念解析与内存管理机制深入探讨》中,详细介绍了Cocos2d-x的基础概念,并深入分析了其内存管理机制。特别是针对Boost库引入的智能指针管理方法进行了详细的讲解,例如在处理鱼的运动过程中,可以通过编写自定义函数来动态计算角度变化,利用CallFunc回调机制实现高效的游戏逻辑控制。此外,文章还探讨了如何通过智能指针优化资源管理和避免内存泄漏,为开发者提供了实用的编程技巧和最佳实践。 ... [详细]
  • PTArchiver工作原理详解与应用分析
    PTArchiver工作原理及其应用分析本文详细解析了PTArchiver的工作机制,探讨了其在数据归档和管理中的应用。PTArchiver通过高效的压缩算法和灵活的存储策略,实现了对大规模数据的高效管理和长期保存。文章还介绍了其在企业级数据备份、历史数据迁移等场景中的实际应用案例,为用户提供了实用的操作建议和技术支持。 ... [详细]
  • 在Linux系统中避免安装MySQL的简易指南
    在Linux系统中避免安装MySQL的简易指南 ... [详细]
  • 在使用 Cacti 进行监控时,发现已运行的转码机未产生流量,导致 Cacti 监控界面显示该转码机处于宕机状态。进一步检查 Cacti 日志,发现数据库中存在 SQL 查询失败的问题,错误代码为 145。此问题可能是由于数据库表损坏或索引失效所致,建议对相关表进行修复操作以恢复监控功能。 ... [详细]
  • 本文介绍了如何利用Shell脚本高效地部署MHA(MySQL High Availability)高可用集群。通过详细的脚本编写和配置示例,展示了自动化部署过程中的关键步骤和注意事项。该方法不仅简化了集群的部署流程,还提高了系统的稳定性和可用性。 ... [详细]
  • 优化后的标题:深入探讨网关安全:将微服务升级为OAuth2资源服务器的最佳实践
    本文深入探讨了如何将微服务升级为OAuth2资源服务器,以订单服务为例,详细介绍了在POM文件中添加 `spring-cloud-starter-oauth2` 依赖,并配置Spring Security以实现对微服务的保护。通过这一过程,不仅增强了系统的安全性,还提高了资源访问的可控性和灵活性。文章还讨论了最佳实践,包括如何配置OAuth2客户端和资源服务器,以及如何处理常见的安全问题和错误。 ... [详细]
  • 本文详细介绍了在 Oracle 数据库中使用 MyBatis 实现增删改查操作的方法。针对查询操作,文章解释了如何通过创建字段映射来处理数据库字段风格与 Java 对象之间的差异,确保查询结果能够正确映射到持久层对象。此外,还探讨了插入、更新和删除操作的具体实现及其最佳实践,帮助开发者高效地管理和操作 Oracle 数据库中的数据。 ... [详细]
  • 数字图书馆近期展出了一批精选的Linux经典著作,这些书籍虽然部分较为陈旧,但依然具有重要的参考价值。如需转载相关内容,请务必注明来源:小文论坛(http://www.xiaowenbbs.com)。 ... [详细]
  • 在深入掌握Spring框架的事务管理之前,了解其背后的数据库事务基础至关重要。Spring的事务管理功能虽然强大且灵活,但其核心依赖于数据库自身的事务处理机制。因此,熟悉数据库事务的基本概念和特性是必不可少的。这包括事务的ACID属性、隔离级别以及常见的事务管理策略等。通过这些基础知识的学习,可以更好地理解和应用Spring中的事务管理配置。 ... [详细]
author-avatar
潮爆啊--_317
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有