QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2132|回复: 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
    拼多多面试问了数据库基础知识,今天分享出来
      ^& v4 S, W$ a7 H- q  u& |+ c+ }. r+ G+ D
    前言
    6 Y+ p3 x2 P9 n8 k
    - S8 }* W( `  r' J我是个标题党,所有文章的名字只是我的噱头,我们应该有一颗谦逊的心,所以希望大家怀着空杯心态好好学,一起进步。) P* m. a- t/ G; v

    , k4 p1 o: p! }数据库我想大家应该一点都不陌生吧,我想不管你写啥的,数据库就算没用过也听过了,是我们项目体系里面不可或缺的一环。
    + c. D! K% o+ v  R) w; l' V  h4 d2 J2 E
    问了一圈,身边朋友公司要么用自研,要么就清一色的MySQL,所以想看Oracle的朋友可能要失望了,但是不影响你了解数据库的通用知识。
    & G/ v5 }( m8 M3 l# V9 R) Z
    : O. Z$ [  B3 v( q* g) l3 S. @正文& H# U7 P5 A) g2 q4 V

    7 b! o- c% C  h- M+ x, @& A你知道MySQL的基本架构么?你能在纸上给我大致画出这个示意图么?" L( Z7 E6 J& S  {1 J+ i# `
    1.jpg
    0 r; j& Q% o! {+ k# z# F好的那我们按照顺序了解下,连接器是啥?( q  k9 S6 ]3 M0 w5 E& E  {
    我们要进行查询,第一步就是先去链接数据库,那这个时候就是连接器跟我们对接。/ O4 {' O% i- {. O) h: _6 I

    ( j" E! h, o) n: f" }2 _1 P5 @他负责跟客户端建立链接、获取权限、维持和管理连接。( y/ C  b9 x. D3 t! ~: ~% K( T

    0 D' h; @1 f0 v6 ]链接的时候会经过TCP握手,然后身份验证,然后我们输入用户名密码就好了。3 g  f% }3 L0 H! ^" t
    : C4 s3 A7 H* W" v& F% k  p9 P
    验证ok后,我们就连上了这个MySQL服务了,但是这个时候我们处于空闲状态。3 M9 P  S! w9 J+ H$ E

    & [3 ~) b) }. `# [: f怎么查看空闲连接列表?
    7 M: v5 C6 z0 `4 q8 L" c. Z/ y+ Ishow processlist,下图就是我在自己的数据库表执行命令的结果,其中的Command列显示为Sleep的这一行,就表示现在系统里面有一个空闲连接。6 u( t* Q  L; S- `4 f7 C4 C, c
    2.jpg
    6 q# ^% I1 ?6 T- ?; [) a这里需要注意的是,我们数据库的客户端太久没响应,连接器就会自动断开了,这个时间参数是wait_timeout控制住的,默认时长为8小时。1 V. W9 X: X* k: A5 z" w/ v
    ( X2 D; l- y) j# m5 j
    断开后重连的时候会报错,如果你想再继续操作,你就需要重连了。
    4 m( L  k/ O9 s6 S$ o$ A- b4 t$ G5 }$ c- `8 k- d$ R% r: O
    这个有个我看过的书本的案例:
    - I5 b: F+ p1 T, s' o$ ~8 v7 v& a9 c8 o
    一个在政府里的朋友说,他们的系统很奇怪,每天早上都得重启一下应用程序,否则就提示连接数据库失败,他们都不知道该怎么办。
    2 c* q3 K. E0 K1 [$ U3 k2 [我分析说,按照这个错误提示,应该就是连接时间过长了,断开了连接。数据库默认的超时时间是8小时,而你们平时六点下班,下班之后系统就没有人用了,等到第二天早上九点甚至十点才上班,这中间的时间已经超过10个小时了,数据库的连接肯定就会断开了。
    ; S" t# {# u" |( n1 i是的,就是超出了超时时间,然后写代码的人也没注意到这个细节,所以才会出现这个问题。: e7 x  _6 m0 u/ i! s! K

    6 z( m9 O5 [9 h/ t9 @0 L) I把超时时间改得长一点,问题就解决了。, ~  b8 M# z. n& l2 Z4 D
    ( P+ i' Z1 x7 Z- |+ N- f' G
    这种参数其实我们平时不一定能接触到,但是真的遇到问题的时候,知道每个参数的大概用法,不至于让你变成无头苍蝇。
    6 U1 i) ^; p% M' g
    2 ?5 e: @, i$ j' f/ v那除了重新链接,还有别的方式么?因为建立链接还是比较麻烦的。! Q* p" ~2 o$ U
    使用长连接。
    * s6 \6 P  Y1 x+ x7 F8 s1 D
    2 }. t% D3 S5 D' B; ?* Z但是这里有个缺点,使用长连接之后,内存会飙得很快,我们知道MySQL在执行过程中临时使用的内存是管理在连接对象里面的。
    7 i' a) U5 V, }  N1 Q% n# f
    & U2 D& ^! h, b只有在链接断开的时候才能得到释放,那如果一直使用长连接,那就会导致OOM(Out Of Memory),会导致MySQL重启,在JVM里面就会导致频繁的Full GC。
    3 K4 U9 S0 U, U- q
    - N* `6 o$ @2 r- W' w6 C那你会怎么解决?$ l. d2 z$ v! d( Q! a2 K& Y
    我一般会定期断开长连接,使用一段时间后,或者程序里面判断执行过一个占用内存比较大的查询后就断开连接,需要的时候重连就好了。
    2 j$ k0 E! p0 _2 r; v! m
    + V9 h1 D+ |9 _9 @& |还有别的方法么?你这种感觉不优雅呀小老弟。
    0 M* |8 R- Z& |/ K; c2 c: d3 B执行比较大的一个查询后,执行mysql_reset_connection可以重新初始化连接资源。这个过程相比上面一种会好点,不需要重连,但是会初始化连接的状态。
    ; M2 u# O5 n& S9 b3 n, B! D5 ]) o+ f! a$ x8 R( T- D* y
    你了解MySQL的查询缓存么?
    + o/ d, w( _6 |0 j+ g) gMySQL拿到一个查询请求后,会先到查询缓存看看,之前是不是执行过这条语句。
    ; d1 z  D4 r2 ~  g, D7 I  |7 s! X- E, R; K5 W) j0 G, ]
    大家是不是好奇同一条语句在MySQL执行两次,第一次和后面的时间是不一样的,后者明显快一些,这就是因为缓存的存在。
    ' y$ f! [6 n2 J) k  N5 b/ C+ k  o. i% i$ r2 o  i, a
    他跟Redis一样,只要是你之前执行过的语句,都会在内存里面用key-value形式存储着。
    " V) e- Z4 e' z7 X
      D6 E9 ]! {  X5 Q2 S' x查询的时候就会拿着语句先去缓存中查询,如果能够命中就返回缓存的value,如果不命中就执行后面的阶段。; `% J" S2 m0 B' h6 W- s

    9 O) m) o: m$ l但是我还是不喜欢用缓存,因为缓存弊大于利。( e4 L) `/ b/ z2 }
    ' n- l) N7 h* D/ ?% d
    哦?此话怎讲?
    1 G' l$ P7 c; E% q缓存的失效很容易,只要对表有任何的更新,这个表的所有查询缓存就会全部被清空,就会出现缓存还没使用,就直接被清空了,或者积累了很多缓存准备用来着,但是一个更新打回原形。5 u) f0 p  M+ L9 W8 N8 H; f( u
    5 _5 f  B0 ^- A2 [
    这就导致查询的命中率低的可怕,只有那种只查询不更新的表适用缓存,但是这样的表往往很少存在,一般都是什么配置表之类的。1 N+ {+ A* i! D3 y( e* ^

    ' c* o- `3 w4 _& \5 e9 m那我们查询的时候不想用缓存一般都是怎么操作的,或者是用缓存又怎么操作?
    7 |& u( X; c1 L6 r! e) N0 \可以显示调用,把query_cache_type设置成为DEMAND,这样SQL默认不适用缓存,想用缓存就用SQL_CACHE。
    4 L. c3 i: a6 a2 d: z
    9 J! t7 c% e; l+ P有个小技巧就是,我们之前开发的时候,都会去库里看看sql执行时间,但是可能是有缓存的,一般我们就在sql前面使用SQL_NO_CACHE就可以知道真正的查询时间了。
    4 f$ ]. H/ z0 X! N$ ?7 B' h* U% ]3 C+ p, |! V- W6 r, Y
    select SQL_NO_CACHE * from B
    / ?& C. k) I  `3 ?; P缓存在MySQL8.0之后就取消了,所以大家现在应该不需要太关注这个问题,主要是我之前用的版本都不高,所以缓存一直有,在《高性能MySQL》书中也看到了一些关于缓存的介绍,就想起来给大家也提一下了。4 a/ u2 A, [( q
    # A# R$ Z4 t- Z2 \5 c/ W4 A" z
    缓存查询完了应该做啥呢?2 q' [9 g$ g7 r1 u- O9 }8 n0 d
    在缓存没有命中的情况下,就开始执行语句了,你写的语句有没有语法错误,这是接下来MySQL比较关心的点。
    $ H, x, t1 Q# F) E' V$ [
      e# B$ i" |7 N" [那他会怎么做呢?会先做词法分析,你的语句有这么多单词、空格,MySQL就需要识别每个字符串所代表的是什么,是关键字,还是表名,还是列名等等。
    % A3 h+ b/ y$ b* C; N' U
    , Y3 u, X+ B( Z+ u然后就开始语法分析,根据词法分析的结果,语法分析会判断你sql的对错,错了会提醒你的,并且会提示你哪里错了。. s# z) V; ~/ `# o
    3.jpg
    2 T% ^" a2 I" o9 g; T分析没错之后就进入下一步,优化器。
    , T9 y1 D0 A* b6 `' i# F
    6 W4 t3 N; I$ W9 X+ Y: G5 i- X主要是优化什么呢?
    7 Q; |6 ~6 ?" M2 ~1 X) B% Q优化就比较简单了,因为我们建立表可能会建立很多索引,优化有一步就是要确认使用哪个索引,比如使用你的主键索引,联合索引还是什么索引更好。
    ; w" e2 T+ Y% j/ j) d! F9 u) e3 J2 y  [6 `5 W6 p$ j8 p
    还有就是对执行顺序进行优化,条件那么多,先查哪个表,还是先关联,会出现很多方案,最后由优化器决定选用哪种方案。
    : E% Y' l" h! T$ Q3 R0 m& u2 L) x8 {) b! c! R
    最后就是执行了,执行就交给执行器去做。
    : V. s7 [" c" K4 @: e" z, {/ h
    2 F: r. Y+ z( N1 B) W! N. D- _) M第一步可能就是权限的判断,其实这里我不确定的一个点就是,我接触的公司很多都是自研的线上查询系统,我们是不能用Navicat直连线上库,只能去网页操作,那表的权限是在MySQL层做的,还是系统做的,我猜应该是系统层做的,MySQL可能默认就全开放了,只是我们 不知道ip。" u. E( A; e" x0 b

    8 v, m! E+ A( `  _  D' \' ^有知道的小伙伴也可以在公众号【三太子敖丙】去加我微信跟我说。$ f' ~, m, Z: ?& w) x8 i6 p

    0 t  f2 n" Z1 ?- E3 g执行的时候,就一行一行的去判断是否满足条件,有索引的执行起来可能就好点,一行行的判断就像是接口都提前在引擎定义好了,所以他比较快。
    . }6 R5 q& D* t7 |8 Z% F, u6 U' k$ `8 C: Z
    数据库的慢日志有个rows_examined字段,扫描多少行可以看到,还有explain也可以看到执行计划,我们扫描了多少行。; m# ?" a' z' ?# f4 B! Q5 T! c
    . d" e) G& q5 ?+ Z$ C4 V
    可以小伙子,基础大致框架还是了解得很清楚的,我们下次深入了解下,索引和部分机制。
    + g6 r4 t5 ]) X( z- A7 A好的,我们下次见。
    6 ]8 S/ I/ z/ B3 h, P5 J$ C- T8 D7 w! x6 [; h7 \
    总结
    9 H" N) e6 o5 M( P0 `' J! M  W1 o& ?# `0 D0 j7 q
    基本上我把MySQL的逻辑架构的东西都简单聊了一遍,当然你去自信了解的话,你会发现其实里面还有很多细节的,我只是说了一些常见的问题,这还是阿里丁奇学长的《MySQL实战》的思路。' s: F9 c6 @. ]! c4 c! n2 x; E( i

    # b% K% Y" N7 C: O) m! S0 |6 g我自己在MySQL方面更多的可能就是理论知识了,还做不到深入了解的地步,大家如果有机会一定要深入的去学习一下。
    + {; g4 }; S" Q# {* B5 S* y————————————————
      U! b' l( _# g) i3 D, H( p版权声明:本文为CSDN博主「敖 丙」的原创文章,遵循 CC 4.0 BY 版权协议,转载请附上原文出处链接及本声明。
    8 M0 f* W* t, n0 V. g& y原文链接:https://blog.csdn.net/qq_35190492/article/details/104203466
    9 w2 {' X  O- b6 i. c# b
    2 B1 h7 O. h8 R! x$ l- U/ _4 }# _! D7 D1 I9 u1 @" h& V
    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, 2025-8-18 19:27 , Processed in 0.435919 second(s), 54 queries .

    回顶部