QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2478|回复: 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
    拼多多面试问了数据库基础知识,今天分享出来' n+ q/ v( ?* Q1 z$ F8 R
    : o9 b/ B2 H8 \! c) v3 X7 y' U
    前言
    2 Y8 k4 i2 d4 `
    1 L% ]. z/ o4 d% G5 K5 O. E我是个标题党,所有文章的名字只是我的噱头,我们应该有一颗谦逊的心,所以希望大家怀着空杯心态好好学,一起进步。
    + L1 i6 \* o6 F& Y
    + t8 h: x; l7 h8 a% s数据库我想大家应该一点都不陌生吧,我想不管你写啥的,数据库就算没用过也听过了,是我们项目体系里面不可或缺的一环。4 Q1 y9 k8 G$ {, X) X4 ]

    . Z  O, O3 V% J! L  T问了一圈,身边朋友公司要么用自研,要么就清一色的MySQL,所以想看Oracle的朋友可能要失望了,但是不影响你了解数据库的通用知识。
      d' G' s8 ]5 U  v: P. F. q3 }. v, v6 D: K2 ^% o$ ^: i
    正文+ T2 X( y  U  Y0 }
    ! w; A! B1 @. d& o3 X
    你知道MySQL的基本架构么?你能在纸上给我大致画出这个示意图么?' C+ Q, y6 ?& w5 F$ x, e  `0 M
    1.jpg
    5 I6 _0 r* l9 u5 P2 @7 ?好的那我们按照顺序了解下,连接器是啥?
    & D& M- b5 o: r, M% i7 r我们要进行查询,第一步就是先去链接数据库,那这个时候就是连接器跟我们对接。& B/ {5 P% [) z2 F0 h

    / B0 k$ G- Y8 ?. o/ v) d; o他负责跟客户端建立链接、获取权限、维持和管理连接。
    / |+ R3 Y6 d4 J9 [& J+ B
    ( I  c! A* _, o# B3 W' ?  J链接的时候会经过TCP握手,然后身份验证,然后我们输入用户名密码就好了。
    5 O# n' \/ h6 Y6 Z6 L6 a0 T1 z1 i) n, |2 f
    验证ok后,我们就连上了这个MySQL服务了,但是这个时候我们处于空闲状态。1 [* M5 K5 n" \  ?1 ]5 y6 j7 F
    ( `# u# \0 p7 E6 i
    怎么查看空闲连接列表?' ^: O$ ]2 X0 }2 ~* M  `
    show processlist,下图就是我在自己的数据库表执行命令的结果,其中的Command列显示为Sleep的这一行,就表示现在系统里面有一个空闲连接。
    . x4 F4 P. c1 Y! Y, G5 f3 I+ R 2.jpg
    8 d8 Q3 \7 j$ M/ s, M4 W6 t这里需要注意的是,我们数据库的客户端太久没响应,连接器就会自动断开了,这个时间参数是wait_timeout控制住的,默认时长为8小时。
    0 N! L2 v. N" n5 w. R7 l+ q8 n$ r. |. A1 i9 U, @
    断开后重连的时候会报错,如果你想再继续操作,你就需要重连了。
    " g' W$ z3 v$ H4 R. G' H8 w& D$ j# `% i2 x, \4 B! L
    这个有个我看过的书本的案例:9 N# @. ~2 }/ ], G! w

    ) d2 c2 c8 L- O  q" Y, T一个在政府里的朋友说,他们的系统很奇怪,每天早上都得重启一下应用程序,否则就提示连接数据库失败,他们都不知道该怎么办。: E8 v- L) D) X
    我分析说,按照这个错误提示,应该就是连接时间过长了,断开了连接。数据库默认的超时时间是8小时,而你们平时六点下班,下班之后系统就没有人用了,等到第二天早上九点甚至十点才上班,这中间的时间已经超过10个小时了,数据库的连接肯定就会断开了。6 o# \6 B' s, K4 k# n0 L0 k9 I
    是的,就是超出了超时时间,然后写代码的人也没注意到这个细节,所以才会出现这个问题。4 a& t7 S# T3 h! o6 B! v
    & r( _1 v) x  f0 k2 ^2 Y% E
    把超时时间改得长一点,问题就解决了。
    & f3 [- y* M  P6 p
    : k- n, W8 S, Q2 y这种参数其实我们平时不一定能接触到,但是真的遇到问题的时候,知道每个参数的大概用法,不至于让你变成无头苍蝇。4 O% o4 V! g& J7 Y

    1 X" n5 B; K1 }7 l/ q那除了重新链接,还有别的方式么?因为建立链接还是比较麻烦的。1 e& G" m$ S; x$ y
    使用长连接。
    * q2 Y4 |+ Q* b. W3 k" l4 \  z3 o- Z- U/ h  Y* }0 |
    但是这里有个缺点,使用长连接之后,内存会飙得很快,我们知道MySQL在执行过程中临时使用的内存是管理在连接对象里面的。
    4 R7 ?/ w7 e* f6 `* l% t; ^2 L  s3 N0 H9 R  E
    只有在链接断开的时候才能得到释放,那如果一直使用长连接,那就会导致OOM(Out Of Memory),会导致MySQL重启,在JVM里面就会导致频繁的Full GC。
    " V. a5 O2 x0 n! m( z3 w: I9 c& a5 E6 [  Y) N( g2 ?
    那你会怎么解决?: a: F+ b8 _3 L/ }, s# u( Y
    我一般会定期断开长连接,使用一段时间后,或者程序里面判断执行过一个占用内存比较大的查询后就断开连接,需要的时候重连就好了。+ u# i+ L, y0 W
    8 ~' s" q  a" `9 l+ }6 o
    还有别的方法么?你这种感觉不优雅呀小老弟。
    0 \- b7 k. J  l5 A7 p. c. M, N执行比较大的一个查询后,执行mysql_reset_connection可以重新初始化连接资源。这个过程相比上面一种会好点,不需要重连,但是会初始化连接的状态。
    1 {$ i9 n0 w! S
    5 l# X4 s8 }, Q" t0 u你了解MySQL的查询缓存么?* B; o/ h. ?+ {: d" b+ t3 f  h
    MySQL拿到一个查询请求后,会先到查询缓存看看,之前是不是执行过这条语句。$ F3 u0 X& o  V' o& i  n5 ]

    ) ~' C8 h' x/ Z7 |大家是不是好奇同一条语句在MySQL执行两次,第一次和后面的时间是不一样的,后者明显快一些,这就是因为缓存的存在。" `9 c- j7 h+ S. `6 j, I

    8 W& U7 Y6 e/ r9 r) M他跟Redis一样,只要是你之前执行过的语句,都会在内存里面用key-value形式存储着。
    " Q3 a. [* x: T, D! b- v' [/ Y: U
    查询的时候就会拿着语句先去缓存中查询,如果能够命中就返回缓存的value,如果不命中就执行后面的阶段。8 f5 T: m0 F' `* `! m* P

    * I( P; ?$ W$ \. ?7 K+ `+ H但是我还是不喜欢用缓存,因为缓存弊大于利。. D: L( k: ^/ ~# N2 X2 ~

      C( @6 `+ Z8 G* {4 U哦?此话怎讲?# N2 }4 R+ _+ O6 r! N- D2 _
    缓存的失效很容易,只要对表有任何的更新,这个表的所有查询缓存就会全部被清空,就会出现缓存还没使用,就直接被清空了,或者积累了很多缓存准备用来着,但是一个更新打回原形。
    8 N: i- K. s  }1 s
    0 b, R, V  b( u" p- M. ?' m这就导致查询的命中率低的可怕,只有那种只查询不更新的表适用缓存,但是这样的表往往很少存在,一般都是什么配置表之类的。
    ) G$ z; ?+ O  W& B9 J' g1 i) b' P( J2 }' j
    那我们查询的时候不想用缓存一般都是怎么操作的,或者是用缓存又怎么操作?
    # T7 ^, f9 B+ n6 C4 f可以显示调用,把query_cache_type设置成为DEMAND,这样SQL默认不适用缓存,想用缓存就用SQL_CACHE。/ @, u6 T( I3 \

    $ z3 X7 e# \; _9 }7 F0 O有个小技巧就是,我们之前开发的时候,都会去库里看看sql执行时间,但是可能是有缓存的,一般我们就在sql前面使用SQL_NO_CACHE就可以知道真正的查询时间了。! u5 q. f$ N: E

    & ~% o  q% `, U# f  B+ |% m  d select SQL_NO_CACHE * from B
    6 Y* ^+ I9 y1 V* R/ Q% }缓存在MySQL8.0之后就取消了,所以大家现在应该不需要太关注这个问题,主要是我之前用的版本都不高,所以缓存一直有,在《高性能MySQL》书中也看到了一些关于缓存的介绍,就想起来给大家也提一下了。
    2 U$ v* ?6 ]# c* w  F' J% N5 E, j# Z0 c- D4 {+ M, V
    缓存查询完了应该做啥呢?
    8 s9 A& u' D) M9 M: J9 q& M( Y7 h在缓存没有命中的情况下,就开始执行语句了,你写的语句有没有语法错误,这是接下来MySQL比较关心的点。
    " |% e4 k. O$ }. M/ H$ k, S1 ?! b- k% q
    那他会怎么做呢?会先做词法分析,你的语句有这么多单词、空格,MySQL就需要识别每个字符串所代表的是什么,是关键字,还是表名,还是列名等等。4 u2 S. X$ x  W* q

    5 \8 J- u1 W% H( S, r然后就开始语法分析,根据词法分析的结果,语法分析会判断你sql的对错,错了会提醒你的,并且会提示你哪里错了。9 A. S' \7 L5 B& O8 I, r# j1 K; U  l
    3.jpg + }* Y- N- F1 a# Y% [: K
    分析没错之后就进入下一步,优化器。% }; @. R; J7 P6 a6 _$ i
    ! d1 |' ^: m: d  R  v" C
    主要是优化什么呢?2 @" t9 s4 t: Y+ [4 {, Y% g5 t0 p
    优化就比较简单了,因为我们建立表可能会建立很多索引,优化有一步就是要确认使用哪个索引,比如使用你的主键索引,联合索引还是什么索引更好。
    # m/ b9 G% g4 b3 b
    & p2 n) x5 u" K( I, y还有就是对执行顺序进行优化,条件那么多,先查哪个表,还是先关联,会出现很多方案,最后由优化器决定选用哪种方案。& O, V( A' J2 x! P) u
    * v2 z: ~& {# L2 v) B1 O
    最后就是执行了,执行就交给执行器去做。: h$ C! ~6 x5 n7 \
    - y& g  `: t/ [) \
    第一步可能就是权限的判断,其实这里我不确定的一个点就是,我接触的公司很多都是自研的线上查询系统,我们是不能用Navicat直连线上库,只能去网页操作,那表的权限是在MySQL层做的,还是系统做的,我猜应该是系统层做的,MySQL可能默认就全开放了,只是我们 不知道ip。
    # x, W7 e/ V% D) I5 ?: U
    9 r& b& N. ]: x9 w: q2 T有知道的小伙伴也可以在公众号【三太子敖丙】去加我微信跟我说。$ o& f( h! r/ o
    + e) n6 N! ^# z. p) x
    执行的时候,就一行一行的去判断是否满足条件,有索引的执行起来可能就好点,一行行的判断就像是接口都提前在引擎定义好了,所以他比较快。
    7 [0 @" Y' y3 x9 E+ m0 N, E4 g# d5 h" j
    数据库的慢日志有个rows_examined字段,扫描多少行可以看到,还有explain也可以看到执行计划,我们扫描了多少行。
    ) r8 L  d4 P% Q2 E6 E
    ' I  m% |6 }' N* _( t可以小伙子,基础大致框架还是了解得很清楚的,我们下次深入了解下,索引和部分机制。/ @2 ?* a- R2 y& j
    好的,我们下次见。
    - C3 P1 p+ L* H3 E* n$ k# h) X; Y' I0 ?/ s+ l+ m
    总结
    4 A2 E: C' O7 U0 Y! l* H( @
    : ]% J4 x1 T) w7 C% M% w- N基本上我把MySQL的逻辑架构的东西都简单聊了一遍,当然你去自信了解的话,你会发现其实里面还有很多细节的,我只是说了一些常见的问题,这还是阿里丁奇学长的《MySQL实战》的思路。
    % c! `: {  {$ ~0 I& m- S' N5 Q  V1 z& A
    我自己在MySQL方面更多的可能就是理论知识了,还做不到深入了解的地步,大家如果有机会一定要深入的去学习一下。% V4 K# e/ v% J
    ————————————————
    ' P+ p& p) Z0 t9 A  D- k# Y版权声明:本文为CSDN博主「敖 丙」的原创文章,遵循 CC 4.0 BY 版权协议,转载请附上原文出处链接及本声明。
    & ~8 a* `( ~+ h9 c& [$ t原文链接:https://blog.csdn.net/qq_35190492/article/details/104203466: C6 d' H+ O1 u  A7 Y: m6 E$ w9 B

    * z. _! _3 ?5 ^1 m/ B% Q! G! [) h/ w% n1 d0 [! K( h4 U
    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-10 08:45 , Processed in 0.523738 second(s), 53 queries .

    回顶部