- 在线时间
- 1630 小时
- 最后登录
- 2024-1-29
- 注册时间
- 2017-5-16
- 听众数
- 82
- 收听数
- 1
- 能力
- 120 分
- 体力
- 561553 点
- 威望
- 12 点
- 阅读权限
- 255
- 积分
- 173839
- 相册
- 1
- 日志
- 0
- 记录
- 0
- 帖子
- 5313
- 主题
- 5273
- 精华
- 18
- 分享
- 0
- 好友
- 163
TA的每日心情 | 开心 2021-8-11 17:59 |
|---|
签到天数: 17 天 [LV.4]偶尔看看III 网络挑战赛参赛者 网络挑战赛参赛者 - 自我介绍
- 本人女,毕业于内蒙古科技大学,担任文职专业,毕业专业英语。
 群组: 2018美赛大象算法课程 群组: 2018美赛护航培训课程 群组: 2019年 数学中国站长建 群组: 2019年数据分析师课程 群组: 2018年大象老师国赛优 |
拼多多面试问了数据库基础知识,今天分享出来$ c* W% R2 @% x/ f
M5 h2 g. c4 E7 V
前言
( b# {, `( _$ r; W P1 d) J! ^$ `; J% ~1 j8 j* i
我是个标题党,所有文章的名字只是我的噱头,我们应该有一颗谦逊的心,所以希望大家怀着空杯心态好好学,一起进步。1 V |0 [1 y. I# B6 ~; x3 ~- }
g. o4 Y+ I( a2 G数据库我想大家应该一点都不陌生吧,我想不管你写啥的,数据库就算没用过也听过了,是我们项目体系里面不可或缺的一环。
! b- T6 c1 \. r/ E* T2 }$ i, J [5 o0 H2 t
问了一圈,身边朋友公司要么用自研,要么就清一色的MySQL,所以想看Oracle的朋友可能要失望了,但是不影响你了解数据库的通用知识。
. s5 ^# I- d" d2 _7 N
6 C9 J, i) Y# X, w- W( p正文" a; l8 M0 F* ]: h5 W6 I# D7 d
& }8 }- O0 o+ V5 Q) f
你知道MySQL的基本架构么?你能在纸上给我大致画出这个示意图么?! d" |4 G+ U$ A5 o; [
6 K9 b o, S9 _- p
好的那我们按照顺序了解下,连接器是啥?6 O0 q7 k/ t B1 K
我们要进行查询,第一步就是先去链接数据库,那这个时候就是连接器跟我们对接。
! X! }! b% C# y% R) T n% \4 j
5 k- [- g4 R! o+ | R! o3 ]他负责跟客户端建立链接、获取权限、维持和管理连接。3 A) ]& {: D7 Q7 m
( I: z0 ~" o# L R; A4 \( i链接的时候会经过TCP握手,然后身份验证,然后我们输入用户名密码就好了。
! b4 b. q* x$ @ l" _. r6 K
+ v: r" {; M7 v验证ok后,我们就连上了这个MySQL服务了,但是这个时候我们处于空闲状态。
- a+ \1 e3 {7 ?) R$ o% ~% G. A9 x: ^: n2 ^: k( g6 T
怎么查看空闲连接列表?; n4 ]# X) K/ d6 a/ ^8 l$ q1 f
show processlist,下图就是我在自己的数据库表执行命令的结果,其中的Command列显示为Sleep的这一行,就表示现在系统里面有一个空闲连接。
% q+ |2 m' ~' n! b9 M( b4 o' E
; R: f3 v! [2 k, f
这里需要注意的是,我们数据库的客户端太久没响应,连接器就会自动断开了,这个时间参数是wait_timeout控制住的,默认时长为8小时。
3 V" g6 y3 F+ l. \5 C/ [3 l, v8 V7 }1 l7 L& j
断开后重连的时候会报错,如果你想再继续操作,你就需要重连了。4 V' R$ M J$ k6 E
6 n3 L+ Y! E- U: P$ z2 i4 K0 T这个有个我看过的书本的案例:7 M% u( w. i5 g" `% w
: T4 @6 e$ m- ]! J3 ]& G' l
一个在政府里的朋友说,他们的系统很奇怪,每天早上都得重启一下应用程序,否则就提示连接数据库失败,他们都不知道该怎么办。4 ^+ \9 j" V2 Q6 \- ~; Y9 ~
我分析说,按照这个错误提示,应该就是连接时间过长了,断开了连接。数据库默认的超时时间是8小时,而你们平时六点下班,下班之后系统就没有人用了,等到第二天早上九点甚至十点才上班,这中间的时间已经超过10个小时了,数据库的连接肯定就会断开了。
0 n3 U0 b. f0 C0 |6 U$ I是的,就是超出了超时时间,然后写代码的人也没注意到这个细节,所以才会出现这个问题。
+ |- Z& J& Q5 |7 H* ^5 u' O6 A6 d Y( J# o/ {" r6 k1 p
把超时时间改得长一点,问题就解决了。/ a. [ L( H7 m/ R6 B
$ E6 S1 l, f# k* G( J! L这种参数其实我们平时不一定能接触到,但是真的遇到问题的时候,知道每个参数的大概用法,不至于让你变成无头苍蝇。6 v; V* y: }+ v- Q' F# i M
5 W2 d3 S% k& [, B3 \# `那除了重新链接,还有别的方式么?因为建立链接还是比较麻烦的。
. u; \) W/ h( P$ d使用长连接。
7 u6 V, o7 u6 A( d7 G5 g3 K0 W7 }. e1 [$ }. h0 l# Y: H3 m' h: g
但是这里有个缺点,使用长连接之后,内存会飙得很快,我们知道MySQL在执行过程中临时使用的内存是管理在连接对象里面的。/ j; F0 C3 m+ l
9 G% z7 `& T+ |% @8 `只有在链接断开的时候才能得到释放,那如果一直使用长连接,那就会导致OOM(Out Of Memory),会导致MySQL重启,在JVM里面就会导致频繁的Full GC。
# e) e+ }$ l. w+ t/ R1 r1 D3 X4 q0 P L( A
那你会怎么解决?* G! n8 r7 Y8 K, Z5 g6 k0 x2 X* G
我一般会定期断开长连接,使用一段时间后,或者程序里面判断执行过一个占用内存比较大的查询后就断开连接,需要的时候重连就好了。
+ j. u9 Z! ?4 |6 L: Z/ i& f
8 C& f2 n( _! u9 A4 V0 [还有别的方法么?你这种感觉不优雅呀小老弟。
0 P; O }5 q6 |/ ~执行比较大的一个查询后,执行mysql_reset_connection可以重新初始化连接资源。这个过程相比上面一种会好点,不需要重连,但是会初始化连接的状态。6 _4 M, z$ _" C$ M
1 J' p( X9 L2 l0 _8 \# \ q你了解MySQL的查询缓存么?
- z+ @6 T7 L& q. v0 [* ^MySQL拿到一个查询请求后,会先到查询缓存看看,之前是不是执行过这条语句。
& u$ c. |8 Y) b! Z4 q
; O2 M; K3 \# ?1 U) d5 W2 g1 J大家是不是好奇同一条语句在MySQL执行两次,第一次和后面的时间是不一样的,后者明显快一些,这就是因为缓存的存在。( k0 |( l0 o# m' Z& u! m; |
4 _2 Y' _9 o% p5 ]* X0 m
他跟Redis一样,只要是你之前执行过的语句,都会在内存里面用key-value形式存储着。
$ A/ y! Y! a& ~( h% u7 c/ G( b" g; g5 K* x8 f: ^+ q$ r7 W6 |
查询的时候就会拿着语句先去缓存中查询,如果能够命中就返回缓存的value,如果不命中就执行后面的阶段。5 \; Q! \1 m2 @
& f! p, e* w- l+ X" H2 e2 X& b
但是我还是不喜欢用缓存,因为缓存弊大于利。& [! |7 j) W% d7 s. ~: y0 g/ S
+ r; h4 p' O: n) k/ v
哦?此话怎讲?$ J6 p- d4 E; i
缓存的失效很容易,只要对表有任何的更新,这个表的所有查询缓存就会全部被清空,就会出现缓存还没使用,就直接被清空了,或者积累了很多缓存准备用来着,但是一个更新打回原形。
/ f0 U' X# R. L: R% `7 h# _2 n2 R. P) R9 P) L0 E4 X
这就导致查询的命中率低的可怕,只有那种只查询不更新的表适用缓存,但是这样的表往往很少存在,一般都是什么配置表之类的。
. z* v! |6 `) n1 l
% H% W3 Y5 _+ M6 z' Q6 h. E8 {那我们查询的时候不想用缓存一般都是怎么操作的,或者是用缓存又怎么操作?# k! K4 N$ C- z3 g
可以显示调用,把query_cache_type设置成为DEMAND,这样SQL默认不适用缓存,想用缓存就用SQL_CACHE。
0 c! r6 g" T$ I6 U O7 m: I4 i$ |# A" j L/ K& `/ {& l
有个小技巧就是,我们之前开发的时候,都会去库里看看sql执行时间,但是可能是有缓存的,一般我们就在sql前面使用SQL_NO_CACHE就可以知道真正的查询时间了。
- t+ M5 Y' h8 `! G1 t! U
9 d& w' Q2 [. d! Y2 Q select SQL_NO_CACHE * from B
' b: ?4 u6 | S) U* Q u: O O缓存在MySQL8.0之后就取消了,所以大家现在应该不需要太关注这个问题,主要是我之前用的版本都不高,所以缓存一直有,在《高性能MySQL》书中也看到了一些关于缓存的介绍,就想起来给大家也提一下了。
+ n; P6 I( y" P/ a( d+ W, H3 f( t- B$ b a' Z$ m) @: z) E
缓存查询完了应该做啥呢?
7 o) b" V9 [0 \在缓存没有命中的情况下,就开始执行语句了,你写的语句有没有语法错误,这是接下来MySQL比较关心的点。
3 i5 t& P' N; ^9 b& b' U2 S$ f" X6 R# b2 U w4 |5 G
那他会怎么做呢?会先做词法分析,你的语句有这么多单词、空格,MySQL就需要识别每个字符串所代表的是什么,是关键字,还是表名,还是列名等等。( j& z$ q- a; @5 r; y
. |9 H0 @8 W6 E% L9 K$ U0 R9 v然后就开始语法分析,根据词法分析的结果,语法分析会判断你sql的对错,错了会提醒你的,并且会提示你哪里错了。" g" `" J8 Y6 R/ u, T' r6 s
9 E& `+ e, J5 h( s8 H' ~. \+ B
分析没错之后就进入下一步,优化器。7 V( D1 w8 d1 S9 n( Z: t- m# y
( {' s3 J3 p/ I; o! l) j6 n b$ V
主要是优化什么呢?
% f& \0 j- }9 N0 p7 C5 X1 T* x: {, A: J优化就比较简单了,因为我们建立表可能会建立很多索引,优化有一步就是要确认使用哪个索引,比如使用你的主键索引,联合索引还是什么索引更好。1 W1 e, s, H* @/ T# X
: e5 }6 m' v& u3 C; J- |& s. I
还有就是对执行顺序进行优化,条件那么多,先查哪个表,还是先关联,会出现很多方案,最后由优化器决定选用哪种方案。0 z) a( ^( z; u/ K" @
" ^, { W0 i2 M. L8 ^3 F/ Y最后就是执行了,执行就交给执行器去做。
# q7 n+ }( {$ U" M+ A ~& X! H. c( m! L
第一步可能就是权限的判断,其实这里我不确定的一个点就是,我接触的公司很多都是自研的线上查询系统,我们是不能用Navicat直连线上库,只能去网页操作,那表的权限是在MySQL层做的,还是系统做的,我猜应该是系统层做的,MySQL可能默认就全开放了,只是我们 不知道ip。
( ~8 e! R' G2 e* ?* a: ?4 A
- B6 q$ D9 ^7 @4 P/ D! t2 ]1 C( z- ~% G有知道的小伙伴也可以在公众号【三太子敖丙】去加我微信跟我说。2 \+ X9 y* V" K
3 i, D* ]' Z3 h |# @4 T! w/ |执行的时候,就一行一行的去判断是否满足条件,有索引的执行起来可能就好点,一行行的判断就像是接口都提前在引擎定义好了,所以他比较快。
: x) R0 g" @! }6 w# s& [5 K2 ?9 {" t+ P
数据库的慢日志有个rows_examined字段,扫描多少行可以看到,还有explain也可以看到执行计划,我们扫描了多少行。
" T! n* R5 E5 m$ _8 E8 K. I: @7 E9 Y- C4 i
可以小伙子,基础大致框架还是了解得很清楚的,我们下次深入了解下,索引和部分机制。3 v; ^& R, U6 n. E. R) F
好的,我们下次见。
' F& `% n$ }! |4 _( c0 U
( t) W5 v4 S) W: s. I) F6 R总结
5 ^) Y6 U3 _: v7 X4 t0 b+ I% F! B3 q. u0 p
基本上我把MySQL的逻辑架构的东西都简单聊了一遍,当然你去自信了解的话,你会发现其实里面还有很多细节的,我只是说了一些常见的问题,这还是阿里丁奇学长的《MySQL实战》的思路。0 j0 [* i3 b' i0 V1 }) @
g8 c( ~' }3 ^' r# f- t: c7 m
我自己在MySQL方面更多的可能就是理论知识了,还做不到深入了解的地步,大家如果有机会一定要深入的去学习一下。
2 O# {0 |6 j1 D: q9 P8 G————————————————
% u, M+ X& Y- k$ M t& P; g8 r版权声明:本文为CSDN博主「敖 丙」的原创文章,遵循 CC 4.0 BY 版权协议,转载请附上原文出处链接及本声明。2 C X$ o0 Y% E. Y
原文链接:https://blog.csdn.net/qq_35190492/article/details/104203466" `' Q2 C7 j# l+ k- j0 Q3 ^ j
0 G, n+ @1 I' H+ r1 ]- I& H
) `& t' {' z7 o; Y |
zan
|