QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2484|回复: 0
打印 上一主题 下一主题

[参考文献] 拼多多面试问了数据库基础知识,今天分享出来

[复制链接]
字体大小: 正常 放大
杨利霞        

5273

主题

82

听众

17万

积分

  • TA的每日心情
    开心
    2021-8-11 17:59
  • 签到天数: 17 天

    [LV.4]偶尔看看III

    网络挑战赛参赛者

    网络挑战赛参赛者

    自我介绍
    本人女,毕业于内蒙古科技大学,担任文职专业,毕业专业英语。

    群组2018美赛大象算法课程

    群组2018美赛护航培训课程

    群组2019年 数学中国站长建

    群组2019年数据分析师课程

    群组2018年大象老师国赛优

    跳转到指定楼层
    1#
    发表于 2020-4-28 15:19 |只看该作者 |正序浏览
    |招呼Ta 关注Ta
    拼多多面试问了数据库基础知识,今天分享出来
    0 U. S/ u5 I$ a- D5 M% `7 W* B& v. W( L6 W- Z& j6 h% v! |
    前言7 s0 i, B: h2 {( f* T8 f
    ; M  o" [4 o* j" i, o
    我是个标题党,所有文章的名字只是我的噱头,我们应该有一颗谦逊的心,所以希望大家怀着空杯心态好好学,一起进步。
    ) @7 b6 y- Q* W2 j* b
    8 T8 ^; [3 y2 z8 \数据库我想大家应该一点都不陌生吧,我想不管你写啥的,数据库就算没用过也听过了,是我们项目体系里面不可或缺的一环。
    9 n6 p- e" M8 z- B4 e. n, L) f/ g0 y0 v' m
    问了一圈,身边朋友公司要么用自研,要么就清一色的MySQL,所以想看Oracle的朋友可能要失望了,但是不影响你了解数据库的通用知识。
    0 u! n& `  z/ z3 Z% X# A; a5 }+ O5 ~3 p: y8 T
    正文, y2 S1 r" D3 O; U8 }
    $ p( h: W) U& D* H- n; G
    你知道MySQL的基本架构么?你能在纸上给我大致画出这个示意图么?8 h0 L% m4 V. {9 Y2 M3 s  K" z
    1.jpg
    % U5 e! a1 {% P6 j9 p  c好的那我们按照顺序了解下,连接器是啥?
    ( [2 D( w2 v. L我们要进行查询,第一步就是先去链接数据库,那这个时候就是连接器跟我们对接。; ]$ y2 a4 O! j- I3 }  j; O2 Y
    . i  c; {9 E" r
    他负责跟客户端建立链接、获取权限、维持和管理连接。# k5 {" A7 g# X
    ) l6 i3 T* s3 G3 g: p/ ?9 ]
    链接的时候会经过TCP握手,然后身份验证,然后我们输入用户名密码就好了。
    * P& t1 K0 E2 u& ?7 y1 E
    + N9 `  p0 A; @6 v7 z0 m验证ok后,我们就连上了这个MySQL服务了,但是这个时候我们处于空闲状态。
    9 M$ ~. C$ |  f; b
    % \& G, R4 |1 ~0 K5 A/ I怎么查看空闲连接列表?
    + k/ G  x! `! O7 @9 w* ~- Tshow processlist,下图就是我在自己的数据库表执行命令的结果,其中的Command列显示为Sleep的这一行,就表示现在系统里面有一个空闲连接。
    - j& V0 V3 k; F4 t" D8 B7 u 2.jpg ! J$ {/ a! s" b" k! H$ w6 c7 M1 S
    这里需要注意的是,我们数据库的客户端太久没响应,连接器就会自动断开了,这个时间参数是wait_timeout控制住的,默认时长为8小时。& E( k) a$ B6 t) f

    4 v, e5 F# i; l9 Q5 V' G- f断开后重连的时候会报错,如果你想再继续操作,你就需要重连了。8 a  m: ~' o& Q. |+ e1 n# b; j, a
    , c: [" L0 C' D& P* `
    这个有个我看过的书本的案例:
    , w/ O4 M: I2 z4 Z
    $ u* w: o* t6 y7 `7 L+ e2 z一个在政府里的朋友说,他们的系统很奇怪,每天早上都得重启一下应用程序,否则就提示连接数据库失败,他们都不知道该怎么办。
    ' C( L/ |0 Z. h' x- _' m. H* \6 h$ o我分析说,按照这个错误提示,应该就是连接时间过长了,断开了连接。数据库默认的超时时间是8小时,而你们平时六点下班,下班之后系统就没有人用了,等到第二天早上九点甚至十点才上班,这中间的时间已经超过10个小时了,数据库的连接肯定就会断开了。, _' w( a( c+ c6 I  z' P
    是的,就是超出了超时时间,然后写代码的人也没注意到这个细节,所以才会出现这个问题。+ h9 |# b. Z, C8 C2 \: P

    ! {/ l/ ?1 s1 a把超时时间改得长一点,问题就解决了。
    # B& D' G( p+ s3 f
    ! f7 y1 i1 |; ]$ m5 b' T  x- q. n这种参数其实我们平时不一定能接触到,但是真的遇到问题的时候,知道每个参数的大概用法,不至于让你变成无头苍蝇。( R7 `6 z$ K! p- N2 U& z
    / G6 n/ E/ C1 K
    那除了重新链接,还有别的方式么?因为建立链接还是比较麻烦的。
    & [8 I- I1 R0 k9 |, h! {使用长连接。
    1 t/ a+ a( ?# ]9 h& F1 l
    4 j5 ~& Z$ m2 W3 M4 [但是这里有个缺点,使用长连接之后,内存会飙得很快,我们知道MySQL在执行过程中临时使用的内存是管理在连接对象里面的。8 g$ h, T2 ^$ G; t" {* I) T! ?

    / g, c( D4 |' B& d! F' f. G只有在链接断开的时候才能得到释放,那如果一直使用长连接,那就会导致OOM(Out Of Memory),会导致MySQL重启,在JVM里面就会导致频繁的Full GC。
    1 g$ E8 B' \8 l' `0 m5 |
    2 Z3 x$ J7 p- W0 ]那你会怎么解决?9 [5 M" T! G& h" w  P) j
    我一般会定期断开长连接,使用一段时间后,或者程序里面判断执行过一个占用内存比较大的查询后就断开连接,需要的时候重连就好了。
    5 V( A) S4 E* e- U: {! V3 l$ l* r4 l+ z$ [( ]9 u. f
    还有别的方法么?你这种感觉不优雅呀小老弟。
    & X0 p5 T8 }7 b; ], y执行比较大的一个查询后,执行mysql_reset_connection可以重新初始化连接资源。这个过程相比上面一种会好点,不需要重连,但是会初始化连接的状态。) M0 x4 W4 b, [9 i) o

    ! z' ^! i' W  ]9 g5 @$ z你了解MySQL的查询缓存么?
    ) M1 a: Z7 t! {1 c; D& p6 i8 G. RMySQL拿到一个查询请求后,会先到查询缓存看看,之前是不是执行过这条语句。
    0 X% k- H: o- P. U0 f0 G
    # v- i, Q/ d* f1 S大家是不是好奇同一条语句在MySQL执行两次,第一次和后面的时间是不一样的,后者明显快一些,这就是因为缓存的存在。
    6 i, ~: b1 Q2 k) Q# |) t, |3 j& o; V1 Q
    他跟Redis一样,只要是你之前执行过的语句,都会在内存里面用key-value形式存储着。% Z, U! c! m& t: F

    * m* r$ C/ `+ H4 b1 Q9 m2 r6 Y查询的时候就会拿着语句先去缓存中查询,如果能够命中就返回缓存的value,如果不命中就执行后面的阶段。/ n; b. O" s6 u3 `  t& G7 S
    / Y. Q( x* n, v3 Y5 H
    但是我还是不喜欢用缓存,因为缓存弊大于利。* M1 P6 m& n5 z- O" ?3 {# t) M
    # ]! C# o0 f, Y4 c! V$ R8 F
    哦?此话怎讲?
    : ^# `. s* k+ V+ t缓存的失效很容易,只要对表有任何的更新,这个表的所有查询缓存就会全部被清空,就会出现缓存还没使用,就直接被清空了,或者积累了很多缓存准备用来着,但是一个更新打回原形。2 a6 ~* L* o6 ?& m/ u& A

    - g3 D3 c% \$ o% U这就导致查询的命中率低的可怕,只有那种只查询不更新的表适用缓存,但是这样的表往往很少存在,一般都是什么配置表之类的。
    6 {) {3 b; E) }; j4 v. L/ H
    : d7 s* Q5 V( f, }那我们查询的时候不想用缓存一般都是怎么操作的,或者是用缓存又怎么操作?
    + J9 Q4 q7 }; a9 s9 u可以显示调用,把query_cache_type设置成为DEMAND,这样SQL默认不适用缓存,想用缓存就用SQL_CACHE。
    ! I; s; V7 M& k! T/ U& A
    9 ~& g5 U; A3 P0 z! l有个小技巧就是,我们之前开发的时候,都会去库里看看sql执行时间,但是可能是有缓存的,一般我们就在sql前面使用SQL_NO_CACHE就可以知道真正的查询时间了。
    ( T0 o: p) N( {7 Z- d* b$ C3 X4 F4 r1 t0 _
    select SQL_NO_CACHE * from B
    # Q) {1 T! {1 H$ n缓存在MySQL8.0之后就取消了,所以大家现在应该不需要太关注这个问题,主要是我之前用的版本都不高,所以缓存一直有,在《高性能MySQL》书中也看到了一些关于缓存的介绍,就想起来给大家也提一下了。
    / o; `& d, Q) `3 [; a$ H- d- C0 S7 e) E1 A) y- I
    缓存查询完了应该做啥呢?
    $ j# x/ {, X' ?9 C* J. R; n在缓存没有命中的情况下,就开始执行语句了,你写的语句有没有语法错误,这是接下来MySQL比较关心的点。
    # G4 D3 a! i$ p: H, e8 \( [3 Y; W: X/ G' h
    那他会怎么做呢?会先做词法分析,你的语句有这么多单词、空格,MySQL就需要识别每个字符串所代表的是什么,是关键字,还是表名,还是列名等等。
    + F+ I+ U4 J, ]
    + R  o' `2 F4 ?' e" C然后就开始语法分析,根据词法分析的结果,语法分析会判断你sql的对错,错了会提醒你的,并且会提示你哪里错了。
    ; z8 |. O# z- A- r: C5 ?3 s 3.jpg ) [; B: ~- f9 @1 o  w8 q' ~1 J* R
    分析没错之后就进入下一步,优化器。
    / }2 D- A1 a- o9 S$ P
    ! ^8 K% m" g1 _7 b  y& c主要是优化什么呢?3 `% u6 D8 C# q7 @$ G8 Q
    优化就比较简单了,因为我们建立表可能会建立很多索引,优化有一步就是要确认使用哪个索引,比如使用你的主键索引,联合索引还是什么索引更好。
    2 c6 W4 P5 K& L' E4 h# l
    9 K/ \3 o8 \/ A, T' r( V' d% V: S还有就是对执行顺序进行优化,条件那么多,先查哪个表,还是先关联,会出现很多方案,最后由优化器决定选用哪种方案。
    % ?6 v! z/ D) h+ S- p/ V/ r. s
    # b' E4 U3 m- f: }; |$ e$ L最后就是执行了,执行就交给执行器去做。' r2 n- M) f- b

    4 s4 H! h" B3 p( d+ w第一步可能就是权限的判断,其实这里我不确定的一个点就是,我接触的公司很多都是自研的线上查询系统,我们是不能用Navicat直连线上库,只能去网页操作,那表的权限是在MySQL层做的,还是系统做的,我猜应该是系统层做的,MySQL可能默认就全开放了,只是我们 不知道ip。
    : G# V8 I7 Q: E- q7 o/ c6 u
    ( J4 W! d; z8 }3 }' V, }4 H4 I% o有知道的小伙伴也可以在公众号【三太子敖丙】去加我微信跟我说。
    8 M/ J7 a3 P* `/ X
    / X: }, M: [8 U9 m4 G& ^9 h执行的时候,就一行一行的去判断是否满足条件,有索引的执行起来可能就好点,一行行的判断就像是接口都提前在引擎定义好了,所以他比较快。
    ' N/ V! b8 p7 D5 K+ W# m
    ' F; N6 B  @  I+ S' o6 N5 z. L数据库的慢日志有个rows_examined字段,扫描多少行可以看到,还有explain也可以看到执行计划,我们扫描了多少行。
    # S$ \: T/ T  }( D$ _5 S, ~" L1 j, c/ C$ f: W8 P' J9 `
    可以小伙子,基础大致框架还是了解得很清楚的,我们下次深入了解下,索引和部分机制。  A4 x$ X( J/ G  N
    好的,我们下次见。
    ( U$ B+ ~6 ^' m2 D2 A  k; L* w
    ; t' @: X$ Y; h$ P. b- z2 |总结0 Z1 x  b& R  z$ J* S+ j8 V

    6 m8 o- k9 ]* n7 ?+ f5 A基本上我把MySQL的逻辑架构的东西都简单聊了一遍,当然你去自信了解的话,你会发现其实里面还有很多细节的,我只是说了一些常见的问题,这还是阿里丁奇学长的《MySQL实战》的思路。; _/ a* {* v- s  s5 B+ V
    6 j. Z0 U4 |/ u' _6 J8 R, K  P0 Y
    我自己在MySQL方面更多的可能就是理论知识了,还做不到深入了解的地步,大家如果有机会一定要深入的去学习一下。
    % R4 y! F6 P1 I0 `3 z% j9 d————————————————
    . W: o6 I$ W0 ^2 e/ k版权声明:本文为CSDN博主「敖 丙」的原创文章,遵循 CC 4.0 BY 版权协议,转载请附上原文出处链接及本声明。
    $ P$ ]1 W3 E  D) O7 z1 F  \, R原文链接:https://blog.csdn.net/qq_35190492/article/details/104203466
    9 f8 w! j$ ^0 c8 {8 |2 ~# e, j* u! i! d9 ], G4 ?8 S/ c. l7 C

    " w! g$ i# a, O  m
    zan
    转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
    您需要登录后才可以回帖 登录 | 注册地址

    qq
    收缩
    • 电话咨询

    • 04714969085
    fastpost

    关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

    手机版|Archiver| |繁體中文 手机客户端  

    蒙公网安备 15010502000194号

    Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

    GMT+8, 2026-4-13 20:25 , Processed in 0.418644 second(s), 54 queries .

    回顶部