QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2489|回复: 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, c5 H/ Q$ R3 f4 {- b) P
    1 c0 D. s2 f) r& j3 ?0 B0 K
    前言$ |- S9 R( \9 m: W4 e6 T& P
    8 N0 Y# m8 p7 a  i- Z8 |6 u
    我是个标题党,所有文章的名字只是我的噱头,我们应该有一颗谦逊的心,所以希望大家怀着空杯心态好好学,一起进步。- T' n4 M0 P9 H& c

    9 B% Q6 R' [& V: n, U数据库我想大家应该一点都不陌生吧,我想不管你写啥的,数据库就算没用过也听过了,是我们项目体系里面不可或缺的一环。$ \2 R% O/ m0 f

    ; x. s+ D6 t$ L4 Z' y问了一圈,身边朋友公司要么用自研,要么就清一色的MySQL,所以想看Oracle的朋友可能要失望了,但是不影响你了解数据库的通用知识。2 ?* Q5 N4 L3 |
    ' g5 }3 x, N4 q! m/ _; e
    正文
    ( n$ p, K( }! P2 Z: Z' u' W  K
    % `! u+ C/ e  y: G1 K$ i你知道MySQL的基本架构么?你能在纸上给我大致画出这个示意图么?5 e/ n  o2 e( |7 g9 W
    1.jpg " a0 L" z, X3 _8 J# I
    好的那我们按照顺序了解下,连接器是啥?: m3 U- x$ j& x. }. T
    我们要进行查询,第一步就是先去链接数据库,那这个时候就是连接器跟我们对接。$ i* S$ T7 f2 j9 X
    6 i/ m$ g( N* B
    他负责跟客户端建立链接、获取权限、维持和管理连接。
    ' L6 p% E! _& h/ Q' K7 K3 \) o- P$ ?. \" O
    链接的时候会经过TCP握手,然后身份验证,然后我们输入用户名密码就好了。
    0 [  u8 a8 m9 Z9 e, Y7 L. |5 m
    2 ^8 v2 y5 b3 y验证ok后,我们就连上了这个MySQL服务了,但是这个时候我们处于空闲状态。
    & M: D" I( a) S% k, `% E7 i
    : k0 p4 w3 n3 s- F" A* n, |怎么查看空闲连接列表?
    6 y: L. O2 d7 D! @show processlist,下图就是我在自己的数据库表执行命令的结果,其中的Command列显示为Sleep的这一行,就表示现在系统里面有一个空闲连接。9 L; M$ h4 \8 @
    2.jpg
    0 }+ M/ g$ a4 V8 T) T这里需要注意的是,我们数据库的客户端太久没响应,连接器就会自动断开了,这个时间参数是wait_timeout控制住的,默认时长为8小时。
    " G/ x2 V* S( v' y2 h1 b* @" u. T+ q, k% x) W' N7 H
    断开后重连的时候会报错,如果你想再继续操作,你就需要重连了。$ a3 p. ?. T% a5 d7 n9 D

    ) T# r& x. @8 A% g这个有个我看过的书本的案例:$ E* L9 H& e& k8 |& ~. r2 ^/ D( J
    5 D9 K9 q; v0 ~: h9 \0 \( W
    一个在政府里的朋友说,他们的系统很奇怪,每天早上都得重启一下应用程序,否则就提示连接数据库失败,他们都不知道该怎么办。( O# E0 T2 {+ R1 z3 N: g( I' w
    我分析说,按照这个错误提示,应该就是连接时间过长了,断开了连接。数据库默认的超时时间是8小时,而你们平时六点下班,下班之后系统就没有人用了,等到第二天早上九点甚至十点才上班,这中间的时间已经超过10个小时了,数据库的连接肯定就会断开了。
    . H9 Q1 k( Q& K( C是的,就是超出了超时时间,然后写代码的人也没注意到这个细节,所以才会出现这个问题。
      B; y2 L  a# @# z* ?  s! f' r7 G6 \: W! H0 P
    把超时时间改得长一点,问题就解决了。
    # s9 r# ?2 ?: V! J% M8 v' z' q+ {4 d! O$ c! a, F
    这种参数其实我们平时不一定能接触到,但是真的遇到问题的时候,知道每个参数的大概用法,不至于让你变成无头苍蝇。8 F% i$ Z: s; J; r$ T) Y% {0 ^7 M9 n
    . r+ M5 E, D7 o  [9 f- Z8 T
    那除了重新链接,还有别的方式么?因为建立链接还是比较麻烦的。
    / P; r$ e% y; w* [8 k使用长连接。5 n' ]; n* N# h2 y
    - U) V! A* m2 Z
    但是这里有个缺点,使用长连接之后,内存会飙得很快,我们知道MySQL在执行过程中临时使用的内存是管理在连接对象里面的。
    * b: A! r% V1 k4 [
    3 Y$ V8 S7 n& R; o7 z7 C! [( w只有在链接断开的时候才能得到释放,那如果一直使用长连接,那就会导致OOM(Out Of Memory),会导致MySQL重启,在JVM里面就会导致频繁的Full GC。
    1 v1 o. i, D5 f! i* B% i4 B8 k  p% V1 u$ Q: X! g- v
    那你会怎么解决?
    " |  }7 L3 p  y5 g/ _3 X1 ~$ ?我一般会定期断开长连接,使用一段时间后,或者程序里面判断执行过一个占用内存比较大的查询后就断开连接,需要的时候重连就好了。
    " @/ [; b- p" e: @+ d- W- ~, u- r% c; I* w
    还有别的方法么?你这种感觉不优雅呀小老弟。9 s7 ]# g1 I( c; h" a; L  q( g
    执行比较大的一个查询后,执行mysql_reset_connection可以重新初始化连接资源。这个过程相比上面一种会好点,不需要重连,但是会初始化连接的状态。1 B4 _7 F: M% c' G0 v1 f
    3 E; n' W# o7 x: b9 z2 S1 x
    你了解MySQL的查询缓存么?
    . W$ H5 q8 y0 \9 G* h: a' W& cMySQL拿到一个查询请求后,会先到查询缓存看看,之前是不是执行过这条语句。5 ^  H# R9 \) i/ C% Q: U  V
    " f$ H6 h0 Z/ s/ P; U
    大家是不是好奇同一条语句在MySQL执行两次,第一次和后面的时间是不一样的,后者明显快一些,这就是因为缓存的存在。
    ( q3 m+ S% l/ f6 V0 I/ t4 f  V
    他跟Redis一样,只要是你之前执行过的语句,都会在内存里面用key-value形式存储着。- N# c( d9 F+ s0 |2 S- N3 |
    1 R- r+ \; U) C
    查询的时候就会拿着语句先去缓存中查询,如果能够命中就返回缓存的value,如果不命中就执行后面的阶段。
    1 D/ w8 u5 q9 S$ [3 M
    ) ]/ O1 }% A, P- u" J5 r但是我还是不喜欢用缓存,因为缓存弊大于利。
    3 ~" \) t5 H( r  D4 e  A' x+ R4 i( b4 p9 j! H) j: a
    哦?此话怎讲?
    3 b4 T8 u$ S  N, p缓存的失效很容易,只要对表有任何的更新,这个表的所有查询缓存就会全部被清空,就会出现缓存还没使用,就直接被清空了,或者积累了很多缓存准备用来着,但是一个更新打回原形。! o" ]% Q# ^/ j

    / J% r+ I0 V, Y6 G% _0 F; _这就导致查询的命中率低的可怕,只有那种只查询不更新的表适用缓存,但是这样的表往往很少存在,一般都是什么配置表之类的。9 @4 g/ e3 i6 o+ E0 X& t

    / X5 y9 k! Z/ d那我们查询的时候不想用缓存一般都是怎么操作的,或者是用缓存又怎么操作?
    , a* u& K% H$ o& U- K) w可以显示调用,把query_cache_type设置成为DEMAND,这样SQL默认不适用缓存,想用缓存就用SQL_CACHE。
    6 M9 [9 u8 e- }# i1 N; ]" _5 X% i. ?8 Q  m2 H7 L" J, \  O# l
    有个小技巧就是,我们之前开发的时候,都会去库里看看sql执行时间,但是可能是有缓存的,一般我们就在sql前面使用SQL_NO_CACHE就可以知道真正的查询时间了。
    5 m7 F8 Y  D! h! T/ `9 T& Y6 f; D! S' O- H4 r* R1 F. |9 A
    select SQL_NO_CACHE * from B. |! c9 ]- [( n8 a
    缓存在MySQL8.0之后就取消了,所以大家现在应该不需要太关注这个问题,主要是我之前用的版本都不高,所以缓存一直有,在《高性能MySQL》书中也看到了一些关于缓存的介绍,就想起来给大家也提一下了。
    ; J+ U* s- }7 [, x* c5 r( Y: k% G# g6 ^4 f
    缓存查询完了应该做啥呢?. z4 s# K$ z, ~2 A$ w
    在缓存没有命中的情况下,就开始执行语句了,你写的语句有没有语法错误,这是接下来MySQL比较关心的点。. `  i/ Z/ |6 e  G
    $ W) s, v* Y0 N) C- a: r. i
    那他会怎么做呢?会先做词法分析,你的语句有这么多单词、空格,MySQL就需要识别每个字符串所代表的是什么,是关键字,还是表名,还是列名等等。
    : D' `+ P# S- A. R
    8 ]2 C, L" J) Z( ]* g. X; C然后就开始语法分析,根据词法分析的结果,语法分析会判断你sql的对错,错了会提醒你的,并且会提示你哪里错了。
    ( s* a1 \! X7 j) ?  E* M: T" q" | 3.jpg
    , l( k9 ]) `' X分析没错之后就进入下一步,优化器。
    4 R' k, p8 L, `8 p: t& Y6 t
    6 o8 H& v9 `5 c4 \0 u主要是优化什么呢?- }% b+ @8 g3 H: w! x9 T* a! S
    优化就比较简单了,因为我们建立表可能会建立很多索引,优化有一步就是要确认使用哪个索引,比如使用你的主键索引,联合索引还是什么索引更好。( R$ Z# v  }2 X  Z% N; C2 W
    ' ?6 p% T4 M0 d% K0 f7 s/ a* K
    还有就是对执行顺序进行优化,条件那么多,先查哪个表,还是先关联,会出现很多方案,最后由优化器决定选用哪种方案。
    6 J1 o/ r% g7 k: p8 G9 d2 \$ h  E$ p# w# ~
    最后就是执行了,执行就交给执行器去做。; Z% p  C0 y& q6 |
    % b# \; t5 B( G# \
    第一步可能就是权限的判断,其实这里我不确定的一个点就是,我接触的公司很多都是自研的线上查询系统,我们是不能用Navicat直连线上库,只能去网页操作,那表的权限是在MySQL层做的,还是系统做的,我猜应该是系统层做的,MySQL可能默认就全开放了,只是我们 不知道ip。
    & [4 D1 r5 f1 A9 R* X# |8 R/ s3 E8 l
    有知道的小伙伴也可以在公众号【三太子敖丙】去加我微信跟我说。1 ]; l! a- e* E2 j5 U) ]) }* K
    / V$ p  A! n. q: @& t5 Z
    执行的时候,就一行一行的去判断是否满足条件,有索引的执行起来可能就好点,一行行的判断就像是接口都提前在引擎定义好了,所以他比较快。
    . H7 P+ r, N" X" k+ i: W6 c; `' }+ w3 U) S: V" S. {& U4 u+ |+ |1 [& w
    数据库的慢日志有个rows_examined字段,扫描多少行可以看到,还有explain也可以看到执行计划,我们扫描了多少行。' ]. A3 ~7 [. ~4 G$ Y+ h
    ' t: m6 z* l1 E
    可以小伙子,基础大致框架还是了解得很清楚的,我们下次深入了解下,索引和部分机制。
    ' [# e% a2 @" [3 F5 e好的,我们下次见。
    ; k8 |0 P' I: r' }+ b1 T' w" D
    * k1 `: _6 g. V4 g  ?' ^总结
    : F+ N: S5 u6 }+ I8 _
    ! f2 Z) X, h  }% V. u7 n; k基本上我把MySQL的逻辑架构的东西都简单聊了一遍,当然你去自信了解的话,你会发现其实里面还有很多细节的,我只是说了一些常见的问题,这还是阿里丁奇学长的《MySQL实战》的思路。( T3 }" [5 i  x' V% J# j4 k" @) Y

    + ]) P$ L- e( I6 Q. I  u我自己在MySQL方面更多的可能就是理论知识了,还做不到深入了解的地步,大家如果有机会一定要深入的去学习一下。
    ' [2 I, e- s! V————————————————
    & R8 O& k! v" F- g) g* |版权声明:本文为CSDN博主「敖 丙」的原创文章,遵循 CC 4.0 BY 版权协议,转载请附上原文出处链接及本声明。! O% q! I7 [% v- Z
    原文链接:https://blog.csdn.net/qq_35190492/article/details/104203466- z( I5 u2 a' ~1 c4 g( O

    - j8 u7 ^9 K- {9 Q" U, A4 G$ T) [% Y% Y6 k/ V* v' T! a
    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-16 08:51 , Processed in 0.418745 second(s), 54 queries .

    回顶部