! ]8 P! K4 v( G! j( I, ]' j4 L这里需要注意的是,我们数据库的客户端太久没响应,连接器就会自动断开了,这个时间参数是wait_timeout控制住的,默认时长为8小时。 : Q- w; o' ?$ a9 {1 p; L $ b7 {2 S2 u3 P* _. k" O5 `断开后重连的时候会报错,如果你想再继续操作,你就需要重连了。 4 d) {# b" I, c" e; @% u3 D) ~. b 6 }0 ^2 g5 ~( v* G6 x这个有个我看过的书本的案例: 9 ?0 V7 `! U+ y3 H# J. o6 x( i/ i. l , w+ j* G- W& O5 d& z: ?一个在政府里的朋友说,他们的系统很奇怪,每天早上都得重启一下应用程序,否则就提示连接数据库失败,他们都不知道该怎么办。 # L# X/ [( n# _: l4 W8 a, ]. T我分析说,按照这个错误提示,应该就是连接时间过长了,断开了连接。数据库默认的超时时间是8小时,而你们平时六点下班,下班之后系统就没有人用了,等到第二天早上九点甚至十点才上班,这中间的时间已经超过10个小时了,数据库的连接肯定就会断开了。 ; k- J* L# m5 Y: R是的,就是超出了超时时间,然后写代码的人也没注意到这个细节,所以才会出现这个问题。 - u0 I6 ?2 s5 F" l4 D: q& E3 O; M6 l/ Q; J8 C
把超时时间改得长一点,问题就解决了。 4 D4 p+ L! I/ F& o' ` ' T3 D9 Q+ t J6 L这种参数其实我们平时不一定能接触到,但是真的遇到问题的时候,知道每个参数的大概用法,不至于让你变成无头苍蝇。5 O6 ~% ^! m0 {8 `
$ I* ~9 s" Y6 k4 {! X _8 W* @
那除了重新链接,还有别的方式么?因为建立链接还是比较麻烦的。 + |+ p0 ~ f, n3 T使用长连接。 6 L. I/ H% y% W $ L. {0 Z$ f8 d但是这里有个缺点,使用长连接之后,内存会飙得很快,我们知道MySQL在执行过程中临时使用的内存是管理在连接对象里面的。; k: l k* d; }' y3 X' B
; w6 L) b: D9 c4 N只有在链接断开的时候才能得到释放,那如果一直使用长连接,那就会导致OOM(Out Of Memory),会导致MySQL重启,在JVM里面就会导致频繁的Full GC。! g8 A& L+ h* p' N/ o
, `: k* T2 v2 O! v1 l. o
那你会怎么解决? " V% m g: S: u4 x我一般会定期断开长连接,使用一段时间后,或者程序里面判断执行过一个占用内存比较大的查询后就断开连接,需要的时候重连就好了。3 \& g: o# q9 S0 B) t. l; P
( S" E( z. r5 b3 Z% E- m还有别的方法么?你这种感觉不优雅呀小老弟。; J$ j5 N. D' I8 f
执行比较大的一个查询后,执行mysql_reset_connection可以重新初始化连接资源。这个过程相比上面一种会好点,不需要重连,但是会初始化连接的状态。; i1 Z) o; h! p6 D2 o) j5 _ A
* ^7 h' u/ v _" M6 F& T- @* H4 R' W
你了解MySQL的查询缓存么? ! H* V! ^6 a0 o- |MySQL拿到一个查询请求后,会先到查询缓存看看,之前是不是执行过这条语句。6 h0 ]: F& w4 `1 ]5 U: E$ c G5 u
; n7 f- Y, U1 f
大家是不是好奇同一条语句在MySQL执行两次,第一次和后面的时间是不一样的,后者明显快一些,这就是因为缓存的存在。0 K% Y# s0 A2 N
: {; A" h4 y; I# E: i+ s4 y他跟Redis一样,只要是你之前执行过的语句,都会在内存里面用key-value形式存储着。 % k! k" b& a8 w6 w 4 g, Q! E a/ B查询的时候就会拿着语句先去缓存中查询,如果能够命中就返回缓存的value,如果不命中就执行后面的阶段。 3 G( n3 ?) K2 ~2 p" ~4 ^( C4 h U* h/ d& {: z0 L; S8 O; J9 `. K
但是我还是不喜欢用缓存,因为缓存弊大于利。. Y& W9 Y' ~" ^, m8 e& |2 S
* {7 J/ g8 }) K
哦?此话怎讲?/ _1 N* o: z6 Y3 O
缓存的失效很容易,只要对表有任何的更新,这个表的所有查询缓存就会全部被清空,就会出现缓存还没使用,就直接被清空了,或者积累了很多缓存准备用来着,但是一个更新打回原形。( L! Q" ?2 W3 u' e0 O
8 e; W& v9 H5 N
这就导致查询的命中率低的可怕,只有那种只查询不更新的表适用缓存,但是这样的表往往很少存在,一般都是什么配置表之类的。6 L1 C3 x9 G- q5 E" i
0 ?; _" v/ n3 \7 ^那我们查询的时候不想用缓存一般都是怎么操作的,或者是用缓存又怎么操作?# e3 S4 A4 k5 e
可以显示调用,把query_cache_type设置成为DEMAND,这样SQL默认不适用缓存,想用缓存就用SQL_CACHE。 9 f: I7 h2 q2 o: w6 w & k# r) T/ S6 A" J. a0 C& k& _有个小技巧就是,我们之前开发的时候,都会去库里看看sql执行时间,但是可能是有缓存的,一般我们就在sql前面使用SQL_NO_CACHE就可以知道真正的查询时间了。 / v4 C1 r; n F. X9 i- A6 v' _+ H/ o8 x
select SQL_NO_CACHE * from B # [! y" L1 p3 G H3 t% X4 p! v- g# I/ w5 d缓存在MySQL8.0之后就取消了,所以大家现在应该不需要太关注这个问题,主要是我之前用的版本都不高,所以缓存一直有,在《高性能MySQL》书中也看到了一些关于缓存的介绍,就想起来给大家也提一下了。3 e; J1 |: R2 W6 N0 e$ [& g% K# Q