, _ M6 f1 v8 y2 I; t6 b c" p9 Q3 Q4 a三、页面缓存优化 ( s9 g2 g% _6 Q3 x; ~) z9 T 1 F( O' [0 v# D7 d: c& y* s页面缓存你优化主要使用redis完成了页面缓存、URL缓存和对象缓存,并且使得频繁访问的商品详情页面和订单详情页面静态化,成为纯html页面,使用ajax异步的方式与后端交互,实现前后端分离。另外还有静态资源优化和CDN优化没有实现。 % ]+ j6 [3 [( g1 r0 m8 \- ~ 1 j! ~7 q: k" ?. P2 A- F1、页面缓存 9 k3 h& i& k( L+ i; E. G: X* {& X! @! H
这里是一个商品列表的展示,可以看到前后代码的区别,刚开始只是直接去数据库中获取商品列表并返回到前台的页面,前台页面获取其中的信息进行填充。而下面的代码则是对页面做了一个缓存,流程大概是这样的:从redis中取缓存->从数据库中取数据->手动渲染模板->返回html。* B) M5 d4 w4 N
首先从redis中取相应的缓存,如果有直接返回,没有则到数据库中取商品列表的数据,并且进行手动渲染,渲染后将html页面存进redis中,并且返回。在redis的保存方法中设置了过期时间,默认是60秒。4 j0 Z- f5 S7 J, A6 {8 G
/ E( x g' O3 @/ X@RequestMapping(value="/to_list") E8 n+ H) z$ h* h; d3 e" M9 L public String list(Model model,MiaoshaUser user) {9 _7 ?' {9 l/ B) e2 B+ B! h2 S" `
model.addAttribute("user", user); * k: H w8 S _4 Y/ G. z" L7 v5 x+ m: F List<GoodsVo> goodsList = goodsService.listGoodsVo();0 @0 W3 }- P! ]' o
model.addAttribute("goodsList", goodsList);! s R/ H$ `8 e& k
return "goods_list"; - u& }/ d) s- c/ d! l2 b } $ W# T6 R+ j$ a7 V! y1 1 T1 g5 L e$ l, l4 q6 V- b: P2" c, |7 K$ j) Y D1 n) _
3( T' o$ g7 o& y
4 ' s( k7 J% e# e5 4 W) n. m. \" z! J; }6 $ {; F, T% u& d: F2 G$ }4 N7 / |# U0 d4 s5 h; R/** 5 F1 R' g! Q0 N2 J f8 t9 ` * 页面缓存:取缓存、手动渲染模板和结果输出2 P5 C+ M( T' ]; m! j
* @param model" y$ L, Q0 _, f) O% }
* @param user* \4 c1 s8 N0 E( {) O9 N6 `
* @return ! A$ `. z8 y% a. k" O3 h */0 b+ D5 |% k1 x# ]# N& R: h* h
@RequestMapping(value = "/to_list",produces = "text/html")* W7 @7 r$ [9 T7 A( Y" h
@ResponseBody P2 l' V% z8 `: m* F. I& p$ M public String toList(HttpServletRequest request, HttpServletResponse response,( s0 F) l$ w) z6 v1 L* l+ T
Model model, KillsUser user){ * }' E, O( o O/ h) i; u //无cookie返回登录界面0 o% M% q/ x7 X$ f/ R: {" Z+ l
model.addAttribute("user",user); 7 g- h3 R- Q" f //取缓存,如果redis中有相应的缓存,直接取出使用 ; k6 t q6 p; @) {( ~0 j9 y String html = redisService.get(GoodsKey.getGoodsList, "", String.class);: @& j; V, o' S: i8 U2 e
if(!StringUtils.isEmpty(html)){, _* H( e+ v+ _. h% F
return html; ' \% Q5 u7 e8 Q+ E1 g5 ` }, r1 W' ]7 l4 x8 `5 a
//查询商品列表3 U: D( e! F# o
List<GoodsVo> goodsList = goodsService.listGoodsVo();, k0 Z6 J( V5 ^ y* j
model.addAttribute("goodsList",goodsList);2 ^9 h2 e0 n' m" z1 w0 } [' T
//return "goods_list";6 ^3 l! e! {) `1 O+ n8 g2 ?
/ c) s( N, G/ l3 b* p6 q' t0 u SpringWebContext ctx=new SpringWebContext(request,response,request.getServletContext(), 3 Y& W" x3 m& U# ]8 E# B request.getLocale(),model.asMap(),applicationContext);" q/ N) F4 b3 Y$ I& f+ k/ }
//手动渲染模板并且存入redis,框架渲染的具体方式也是如此 2 N# J/ Z; E: p3 U/ B html=thymeleafViewResolver.getTemplateEngine().process("goods_list",ctx);& k S; o6 w# i, E& E
if(!StringUtils.isEmpty(html)){; V2 k- `5 y B5 d4 u. \
redisService.set(GoodsKey.getGoodsList,"",html);; @' Z' ~+ U2 ^$ J! @9 [) a
}! B" }; Z" L# z! S# s# {7 e- o, `
//结果返回; C/ g0 ~( c* L! j% D k6 f
return html;3 b" K* O. U3 n: i. k, M0 P( s
}! r# h: p* z; P" _. {& _2 [
1 + J2 U; f# X) f; n4 @2 I2 * C) }+ J! p6 F% c' l3 , b7 p1 P: }$ a4 7 N7 \; X# m/ u/ H8 F5 6 b3 J4 n6 E2 g& l- z6 , O) a" m( Y+ z8 x, _77 Z7 J; w, q _) F4 n8 ?5 M9 F
8' a/ k- a! V& E! N, {
9 % n. W, f" ?! }8 M" ]10+ k- C8 F6 { ]' T" u' I. `
11( i, z6 j7 b. h: D4 i
12 : v$ ?' N# U/ k% S7 V: v/ p13 ) \( W" B. V, A+ g" V14; K: b6 u. y' u) U1 y0 Y3 |2 K
15 y- D4 Q5 q0 a- U0 M: X" n
16 7 I) r! n5 F# N" E17 ) B' O& Q6 P$ X+ y: _18; G3 G9 V1 v8 \5 ~+ R c6 Y x
19 ! ]+ h, H8 _2 e/ ^7 i$ z208 i% w9 T" o9 o/ C4 e5 @/ e2 f
21! x D, E3 w6 n8 X" C, @
22 ' Z# ~ n9 u* E2 v' Z7 J23 / a) ~1 @& l6 H& c# i242 m+ }2 F; q" C f3 d- S
25 h* g( a$ G9 I+ x8 ~% h4 m& n
264 y ~0 i' K3 ^; f
27+ W5 H* ?1 T& |
28& G' f% _* E; K
299 r2 x7 P7 |+ L' p7 _6 t* X
301 k! k5 q I: d0 o' P
31 ( Q% j* O8 J. L* I' _% P( q! M R32 ( f( h' a& u" H) u ' m6 I5 k/ c* b$ g0 z2 O: Q7 V5 h8 G 0 P7 h6 f* j4 j" L/ z- X! g2、对象缓存 v0 e6 {+ t5 K% ?% r
0 `! c# ]4 Q7 {& I/ g
根据token对秒杀对象进行缓存,如果缓存中还有秒杀对象的信息,则不去数据库进行取出。 . R- y7 { ^) R 0 f% n! }4 |1 D: g t + e) W' y9 ]: A: k3、页面静态化5 e, P) j* ^! H' x! P
! J$ J, f) g U% Y
页面静态化常用的技术为AngularJS、Vue.js等等,此处使用js简单模拟该过程,实现前后端分离。页面静态化,其实就是把用户经常访问的页面做成一个静态的页面,避免每次都要从数据库取出数据并且渲染到对应的模板文件中,从而减少服务器的压力。) x1 H7 G. L I' n$ B4 z
页面静态化是将动态渲染生成的画面保存为html文件,放到静态文件服务器中。用户访问的是处理好的静态文件,而需要展示不同数据内容时,则可以在请求完静态化的页面后,在页面中向后端发送请求,获取属于用户的特殊数据。4 v$ O' T/ s# [- O- `$ Q
页面静态化是一个以空间换时间的操作,虽然页面增多时会占用更多的空间,但提高了网站的访问速度,提升了用户的体验,还是非常值得的。 ! j! Y6 ?4 ~7 Y- M关于原来的页面,正常的前端访问后端接口,获取html页面,展示,而页面静态化,是利用ajax异步传递数据,在前台进行数据的填充,实现了前后端分离。 * j3 S5 c g, @: @, B; C/ e* r- x2 l* l3 ]" z6 F
@RequestMapping(value = "/to_detail/{goodsId}",produces = "text/html"): H2 E3 D( X4 r/ U
@ResponseBody 9 U; I. j" l8 o3 i public String detail(HttpServletRequest request, HttpServletResponse response, 1 D/ g$ ?9 U& f/ X( p+ w" V Model model,KillsUser user,7 r3 \! I6 l% V( G$ o5 l2 {, |9 ]
@PathVariable("goodsId")long goodsId){6 \6 W6 `2 I* q+ g; X l
model.addAttribute("user",user);( R5 \* s1 |! q! h1 k
0 ]' s% F) i+ j: | //取缓存,如果redis中有相应的缓存,直接取出使用 $ |* W: H- W9 r8 l9 a String html = redisService.get(GoodsKey.getGoodsDetail, ""+goodsId, String.class); $ S5 g% W+ N" I% ]9 I if(!StringUtils.isEmpty(html)){* Q3 F; S+ d2 S" h3 O1 U' ^% E
return html; % d! i+ N( O# D7 I h% q8 u' ? }; f) s5 \, a/ f: z$ h& x
+ [. P- i0 B c
GoodsVo goods=goodsService.getGoodsVoByGoodsId(goodsId);- g! C' G5 v$ v a
model.addAttribute("goods",goods); ' f2 W: O1 _+ r4 X4 P; ]( J1 P F# c: P+ v* q/ D
long startAt=goods.getStartDate().getTime();+ w! a' _) A1 h$ x9 T! R. {" w
long endAt=goods.getEndDate().getTime();' o W$ Z# [$ p" J! ^
long now=System.currentTimeMillis(); , m& @" g7 N+ C0 w2 f 2 D% `* E" Y0 `) m: l% D3 B int KillStatus=0;//表示还未秒杀 ) v2 ]: b6 N4 A& h9 k6 r int remainSeconds=0;4 n; C- M0 x8 P. Q: a& z
# p) I+ _/ Q6 L% X' f+ O9 k if(now < startAt){//秒杀未开始1 j* m, K5 P. l
KillStatus=0;2 P7 s5 T4 r. b% Z- N8 O
remainSeconds=(int)((startAt-now)/1000);, q: b$ A3 `; }) m( o
}else if(now > endAt){//秒杀已结束 2 r" E& t1 }6 [: v# j; G9 a KillStatus=2; . I' O, ^% x8 l( M remainSeconds=-1; % {6 K2 v; F/ q$ Q, B" ~ }else{//秒杀进行中 * P6 ?5 c- U& ^1 S/ g1 e KillStatus=1;* L9 o I, q4 L; T
remainSeconds=0;1 e K# `3 j6 f+ ?" g. H
}/ |# H* n" \' N0 p- B
model.addAttribute("KillStatus",KillStatus); ! @! B+ {- L6 I( A3 Z6 x. @9 } model.addAttribute("remainSeconds",remainSeconds);& v; {6 e' O/ `, R* {) ^% ?
3 r1 q, @! V$ U
& q, W- W5 V# d I( t; E) T9 a
SpringWebContext ctx=new SpringWebContext(request,response,request.getServletContext(), 3 L( ?: V0 E5 \1 q request.getLocale(),model.asMap(),applicationContext); . |+ P( b0 J# |" I1 i //手动渲染模板并且存入redis,框架渲染的具体方式也是如此- M/ U# ^2 c( t0 w; @) ^
html=thymeleafViewResolver.getTemplateEngine().process("goods_detail",ctx);% [ }1 C: @( a5 J/ ?" G! _* ?1 J
if(!StringUtils.isEmpty(html)){) D2 \! X/ `8 J+ e& J9 E& L
redisService.set(GoodsKey.getGoodsDetail,""+goodsId,html);* `5 g4 F6 G4 J# @: T
}8 N# q# a- G/ k7 `! m4 u+ y
//结果返回& L1 ]7 Q3 z" o, H
return html; 5 S' d" k# H# h7 | //return "goods_detail";: g! r# x! Y7 C
} 6 N! c* F& ^# w6 z# w8 l2 l: _11 v$ B' C5 k$ a9 G: u$ R T
2$ H& z+ b4 Z7 G# y' M% Q+ p& T
3$ \9 e4 E5 u. s8 i, Q9 w W
4: k/ a5 D5 N" Q H
51 c) S& }5 I6 l0 w* z* n
66 A$ m: U$ i' L" {
7& u5 @5 u. Y0 H4 m3 i
8, z( V$ r- F# \
9 " v# U) T/ t* |" k# D7 I( c- q10 ; J% J, R" Q/ T3 v11% X8 b7 m* ^1 |: ?
12 9 m1 }4 Y: x9 b: S' H9 b13 5 C1 P3 P" ?$ N* w: c- ~14" l. D# z4 g; o: `' P, @( S
15+ A$ w! W: n# D, D/ a
16 6 X& J* S3 Y( E2 z176 ~( a1 l& a- \# x1 |
18 6 R# @( |& x0 j# y, n195 `1 P- u) q5 u4 M9 M( H5 B$ ~
20: X- W( N* b; G5 i4 u/ q% P
21 & ?+ c* j7 i: z- w: |22 U5 A9 R0 |2 m7 l/ u23 + z3 g* ?: @5 L) Y$ W' W24 0 C8 S' _) D. I1 S25 * H6 U$ h: i1 S# I: _- P26 " ^( f2 I2 j% Y* p2 J U* n27; A: `+ V' q; _: t7 [6 d
28 % Y7 u3 e) Z7 ^# Y: _29 : K1 [* i, l! D6 g305 O( q5 a/ t/ _( y
31 : I3 w% W" j8 a5 B" l7 O32 9 l- X2 {: K, ~0 _) O# I33 3 v& K U/ @& W# D+ V6 ]% S" B34' _7 N9 P! v: f, h( o$ O# U; \9 F$ J
35 $ |0 n% }5 H: z% T1 I36& @7 J0 `6 F F, a5 o# X
37 S( }8 x3 e- W: C1 @+ U38 , a4 f. d3 `5 n39, Z& h0 x8 N3 _1 G& j2 N# z0 g
405 n1 A9 q3 _# q+ ]" T% Q% I& [
419 u+ c# c. |. M+ S7 p. Y
421 u7 R! i4 e `. m$ `
431 v. G% _1 q6 ]( O3 @
44 4 T" U2 T0 e1 {* Y* u453 E; J3 S7 W6 J
467 |6 S* \) l: \/ s+ d& t
47" U5 E% N3 }. j* E/ j! D, ^
48+ H1 C# r9 Z. }3 C
前端代码通过ajax异步请求后端的接口返回数据,并在前台取出数据对页面进行填充。 : ^2 x7 U9 l! j3 c: d/ H2 s % J8 P) G* b3 I3 X7 z//前端 1 l; h: b5 D8 V& R" F6 M7 n$(function () { . E/ h' N) d4 @+ N Z. C* C4 W //countDown();& v% t+ T t" t* t e3 D6 _
getDetail(); ) j- Z( K1 ]( Z! |9 \5 p}); 3 E Z3 l: M8 Efunction getDetail(){ " E7 l7 z0 H5 Q9 @" c; b var goodsId=g_getQueryString("goodsId"); ! {4 V( r: N6 J/ D8 Z1 T $.ajax({! @0 C$ q# O3 J7 Z) G6 I
url:"/goods/detail2/"+goodsId, 4 W! ?- n4 G1 Y) m- H type:"GET",- ?3 a- |1 m7 w
success:function(data){ 3 h6 x7 B6 R3 t$ P+ _6 r if(data.code==0){ # V8 O$ Y6 ^% ~& v; h1 _: i render(data.data); 8 A; A. N5 Z- V+ l }else{9 ^( x4 a8 n2 H( p: L; h
layer.msg(data.msg);& T ~7 w+ C$ W2 ]
} $ J: ?8 k- J3 l },3 g! Y. E% \# u1 s0 c. {# R
error:function(){ 6 E# ?+ H! w" {3 o h8 x: s, Q! M+ F layer.msg("客户端请求有误") ; O2 n# w( V/ _% q& C) S } / x! d& U! W1 N# H8 J( T& f2 E+ ~* Y$ g& V x
}); 1 |8 y9 q$ k% v* @# w( b5 ^}! J$ J+ R, B, V! T+ i
/*text() :设置或返回匹配元素的内容) e# l2 x9 B3 k5 }4 N+ q
* val() :设置或返回匹配元素的值" w7 g' |- A9 r3 a I5 \
* */ - J+ v2 z5 O- N# F' Yfunction render(detail){ 2 S$ |, d' q$ B8 }: e var goods=detail.goods; ' X# @$ O( z/ }* D) e1 j0 L; M var KillStatus=detail.KillStatus; 1 U/ J* A; `5 ` var remainSeconds=detail.remainSeconds; 3 G1 K5 d5 O* U6 n" I var user=detail.user;4 a/ c8 ^, e. f/ [. P# m2 E4 N: E
if(user){. u, u( E0 J! ^8 G$ _& N# X0 M
$("#userTip").hide();3 ]8 \$ Y6 R8 z/ H- O( u
} 4 l6 ~! L6 ?. ^) j( w2 I: C% M; ~ $("#goodsName").text(goods.goodsName); - y; J! o- a' [- u, w- a( j( I $("#goodsImg").attr("src",goods.goodsImg);, T, y0 r6 @9 X; ^* N1 E& Y$ s" T" o! u: T% W
$("#startDate").text(new Date(goods.startDate).format("yyyy-MM-dd hh:mm:ss"))3 O+ Q6 v* e1 D5 a9 C, ]. H4 j- o
$("#remainSeconds").val(remainSeconds); : E' z% g7 @& }, w7 W0 U4 @ $("#goodsId").val(goods.id);0 O- r4 l0 d4 @* T! v
$("#goodsPrice").text(goods.goodsPrice); " X) ^ I- ^) a1 W" L, p9 r $("#miaoshaPrice").text(goods.miaoshaPrice); $ T. {7 n3 g1 \, z \ $("#stockCount").text(goods.stockCount); 9 \' `5 h( k$ E countDown(); ' y- C1 \* j$ ?) T0 R& t}7 C h2 v6 G+ h9 r" l% c
//后台4 z& B8 w# d0 e) Q
@RequestMapping(value = "/detail2/{goodsId}") * o$ d1 \$ [1 X. k+ b' J @ResponseBody ' r5 B) q0 { {# N5 k public Result<GoodsdetailVo> detail2(HttpServletRequest request, HttpServletResponse response, 6 m& U1 f2 M9 \2 l% z Model model, KillsUser user,$ Z- k( m0 ]% l T
@PathVariable("goodsId")long goodsId){ " [$ l9 u" u, ]$ f GoodsVo goods=goodsService.getGoodsVoByGoodsId(goodsId);6 `% f3 R2 f+ E
: f- \/ c5 f+ D" M; v long startAt=goods.getStartDate().getTime();; y# p0 a. X& I' X( i4 `) O, V+ I
long endAt=goods.getEndDate().getTime(); 3 U. | W* M, u1 p# @ long now=System.currentTimeMillis();8 { ]# x6 y# l1 d* H& C% i
9 A2 R: i4 V3 r! `' l7 N
int KillStatus=0;//表示还未秒杀 : S/ M1 h/ H$ K9 J' ~4 w int remainSeconds=0;$ K( U1 b" V" {+ P8 H* T, n
, s( V/ w- K% D- n8 |( D- X if(now < startAt){//秒杀未开始 2 x$ p* m+ P5 v KillStatus=0; 5 E2 X$ P2 X7 U6 N remainSeconds=(int)((startAt-now)/1000);- K6 s! G2 T" o* t. S
}else if(now > endAt){//秒杀已结束 + g( v$ V% V4 _+ S E KillStatus=2; , c2 A" t# \, @6 O4 W remainSeconds=-1; * L# d1 Z1 _9 Q/ c+ f! K2 O% m! B7 D1 { }else{//秒杀进行中# C s6 _; L0 h" a* s& J0 n
KillStatus=1;, T) {. O& `, g2 G) \
remainSeconds=0; 9 a( D& D& S( u" a* T } + t3 s q! m! }* d; x W: u C- Z& _2 }5 x: s$ G0 D0 Y
GoodsdetailVo vo=new GoodsdetailVo();: C9 q+ R& n, h. T" ^8 d# Q
vo.setGoods(goods); : E% {6 e. [6 S+ H; |$ T vo.setKillStatus(KillStatus);0 I1 s, x# f. a+ {* }0 u( _
vo.setRemainSeconds(remainSeconds); 4 S' T$ I7 ]% x6 y P# Y vo.setUser(user); 2 K9 \. Y1 K4 }; L9 `, f$ T1 V# X return Result.success(vo);6 U9 }$ n( X3 i( P
} % Q+ G! r4 S- E, F1( U9 A4 c1 Q. W) W
2 2 O/ o0 G/ @, M4 y. N$ t1 _31 u0 w8 g* a2 I2 R2 G
4 6 X9 V7 n' [7 M" H# |! U4 D+ u6 a6 V. Y7 W/ v5) k5 b |' y T6 D
6 : i/ Y+ G& j$ t1 Z. d7 3 E, w6 c8 N) n5 f. J8 3 { a2 A1 L5 w+ S9 H' \1 l% D$ M9 . p$ W Y) v+ \' r# G103 E( C2 p. c2 G+ l; P% |
11 5 A% g& C0 a! ^12 4 _& p) z& {" `2 A+ I13, f* s3 y* C) {7 x
14: c. C/ T0 ]) D0 |9 s
15 . P, g2 w/ B' \% d: l( Z169 N. X+ I: L, m5 p5 X1 y6 \
17 5 a& ~' m2 O% X4 v m0 [% \2 K18$ m0 j) b- u! v: b
19 d0 S: X0 R! P; ~6 Q' D7 f$ i
206 o! R# S$ i* Z+ I: k% [+ L+ j
21 ! `* O& O% s1 f5 H* @. G5 @228 h( y' k. t2 O% E: h
23 ! Z. \, {' [* e4 S$ J24 0 d" E* {( A |; A) [25 9 q, L2 `/ v$ p4 l26 : P$ p4 t. c6 A1 f4 _! i27 5 A# V; I0 t. Y1 K/ H28# s/ b" x* v) \) {1 a0 F
291 b, N/ Q& e9 u: E9 ]) e5 w! |
30 + L. k2 v0 x3 X; A2 o) J3 ]% y31 1 w d# ?( m& S5 P k! y32 1 `# v( i3 d; F) R* S33/ y: F) V0 y" \. Z
34 9 Q6 }. A) _9 Z3 `+ A35! u) E) ^( k( T" X0 @; O
369 i' N* c5 A- y) L& G* S3 X
37 " U% a; c! {+ z- ~ W38) J) H9 T# i3 g+ x" }; c6 v
39" A8 c7 F2 z) e3 \ I( R
40 z1 z( K7 m) h+ b- v7 y
41; P$ G- ~3 l% F- [
42# h! Z1 i* K! p$ Q! |- r$ }! `
43 % d# S2 v9 P/ V: B% {% J5 d44 ! g! x7 Q& a4 x1 X \5 {# {) f) x3 ~45 ( A7 n% d! U/ j7 O' V) o46 d9 n; W- i% W" N47 ' ~, b. W+ {8 L; h% o/ _# r u5 L48 s4 O& l5 u' G0 b1 ^$ [493 l( A* Y: |4 f% [4 G
50) J% ] J4 A- e1 N* t1 g( T
51! @) v- j& f# }
52 " I$ e6 U9 ]% t2 r539 q/ O) t4 B4 z2 C; l& ?% H
54 ) ?, g+ Q/ S) {4 A% n55$ V) g1 X1 b- K- T0 d: S4 V
56+ e9 G* w1 c6 r/ Y
57 + W0 }3 D0 p% {8 {2 V58 5 D1 [9 ?' e) R6 P) t' t4 \0 P59 " t5 F( r% y4 D( \0 b9 c1 m6 ]" B607 i+ Z: \8 g# d7 e7 b! b
61 ( e+ [" w, `5 d/ v" d62 0 y4 k6 a |6 D3 I J) G63/ v$ k" J% A8 t& b7 m
64 3 d8 O. b% @5 T! V" D# X65! P$ {5 e: \, i, `% P0 p/ O
664 ?- _* G& G) t4 c6 X
67& X7 r# ]" I" U, d8 I% B. p
68 ) W* z( p W* D, f+ X9 M* S69 + K$ B& {; k& b% ]+ `70 % n# H1 O9 b2 t: U71 - c: A* P- M# }4 I( b$ \72, U7 x4 N1 H% z! ]) T
730 E. Q$ [! S! c' N# J% Q
74% ]# ^! p5 r L$ b2 M! s* r
753 E) f8 ~$ p! E4 Z- ^9 q6 ^
76 : l2 f( Y# a4 d9 O' `9 j6 G77. [' p2 u- g6 A& O: q. X
四、接口优化 ' [6 A5 M; m5 x% d5 M : x# _( Z1 ?1 t5 g接下来是对于秒杀接口的优化,这是最初始的秒杀接口: 8 N2 j Y5 t- y6 i; D) ~! O刚开始的秒杀接口的业务流程是这样的:判断用户状态->获取商品库存并判断->获取秒杀订单并判断->减库存。这个过程需要调用数据库共3次。& l! ?3 e4 s7 O" }' x
秒杀需要面对一个很严峻的问题:如何防止超卖或者少卖?此项目中总共用了两种方式,第一种是在减库存的sql语句中加上对库存的判断,来防止库存变成负数;第二种是对数据库中的秒杀订单表中的用户id列和商品id列加上唯一索引,防止用户重复购买。8 F4 n" H; \+ z* f( b1 v
% i: }" T+ I9 Y r% ~8 b' v5 Q
@RequestMapping("/do_miaosha")/ y! B2 Y$ C7 H3 m: w
public String miaosha(Model model, KillsUser user, 2 z8 ]/ i3 o y; v, X @RequestParam("goodsId")long goodsId){3 d$ u- P9 s) p; p
model.addAttribute("user",user); 5 @% |/ O: M! r( i% s if(user==null){$ C! j: b3 e! M) ~; |1 @* d
return "login";" m+ l' `) \7 j. T! u
}4 B; Z) {) \! t6 ^" g- W
GoodsVo goodsVo = goodsService.getGoodsVoByGoodsId(goodsId); . |) w0 G5 J2 B& L int stockCount = goodsVo.getStockCount(); & f8 r5 L* @# V. E+ V- \ //库存不足 ' a7 t0 j; s9 ^# ~+ h S5 Q if(stockCount<=0){ - e2 S; W' ^" m! I8 J* m, ] model.addAttribute("errormsg", CodeMsg.MIAO_SHA_OVER.getMsg()); ) N" q, b# G" D& T return "miaosha_fail"; / P0 b! U6 ?8 ~. [, a0 t }( ~% U [" V( l7 i0 a8 }% k0 U
//判断是否已经秒杀到了 5 G+ P0 Q6 _' H! Q% [, P; Z; D# l MiaoshaOrder order = orderService.getMiaoshaOrderByUserIdGoodsId(user.getId(), goodsId);& _1 T7 G3 L3 y- Q& N3 q9 Q
if(order!=null){ 3 g' h$ q0 F5 H) k: Z" {% a model.addAttribute("errormsg",CodeMsg.REPEATE_MIAOSHA.getMsg()); 7 S2 {1 D2 |. f! D7 ] return "miaosha_fail";( E; u* S; n3 m* f5 m+ t' r
}: t, ^1 ~. J, u7 m3 K1 q
//减库存,下单,写入秒杀订单 5 u3 g$ m: {5 W' P+ y$ w" k OrderInfo orderInfo = miaoshaService.miaosha(user, goodsVo);/ d r( m4 q$ I4 H/ M# @$ Q
model.addAttribute("orderInfo",orderInfo);' i, H4 ]( _: ?; L, U1 B1 C$ H: L
model.addAttribute("goods",goodsVo); ' z5 B# S( d* S/ }" [% Y return "order_detail"; 5 {( _# @; y8 _6 ~ } ' o3 A! e" H8 j) w ~! `4 n/ g! m6 t1" k% a' g5 D% k/ N* D
2 * q/ G% ]8 e1 o8 [0 x0 u2 c3 7 j9 X- V1 l& F; c9 Q4% v1 w( Q2 E1 \' C" b* ~" l8 S
5. y9 ]( c$ [9 Z4 w( q- N0 j
6. J: I+ Y) V0 h
7 ; t% M: [: ~, {# P3 j- q2 b5 |8& W3 B {3 J0 k H% _1 L0 Y6 i
9' G4 U) t. t; K! h, f( L5 c( O2 k
10 ( b1 C5 M6 |) v& C11 - w( _; A2 c) F; ^! I; x8 w12 7 [# w+ k5 h% m5 ?' B) s135 T- y; {/ C( j
14 $ ` E# b$ N/ ~/ F15' _0 J' v0 M" H2 L( {/ [
16 0 A. i& f9 g$ F4 Q; k17. W8 y- V& X% g- U1 K/ s* i
18 ; ` Z2 G$ M( n" M/ c19 2 A' [- f2 G( ]# F/ A/ B& m# x$ L20, q" Y% ` w7 B" p! {
21 + E" N$ u, n% l$ Y22 5 }4 M1 K( Z: L" F1 Y2 s23( l: c/ r0 \: g' x# k) X
245 C) _3 d- g* m
25 7 W6 W9 Y1 n/ \" @26 7 w* O$ l- t0 s0 \0 f优化:秒杀静态化,通过ajax异步请求调用秒杀接口并进行轮询查看是否成功,redis预减库存减少对数据库的访问,内存标记减少redis访问,RabbitMQ队列缓冲,异步下单,增强用户体验,同时减少服务器压力。首先是使用redis预减库存,我们使用一个HashMap来存储某商品是否售空,这里用到一个方法afterPropertiesSet(),该方法会在Bean的所有属性初始化完成后调用,在这个方法中,我们从数据库中查出商品列表,将每个商品的库存存入redis中,并将库存不为0的商品在HashMap中设置为false,意为未售空。(此处应加上对商品库存是否大于0的判断) ; D: i6 ^$ ]# @# Y2 p在秒杀方法中,我们首先获取mao中的布尔值来判断是否售空决定是否进行下一步操作。使用redis的decr()来预减库存,减完判断是否小于0,如果小于0就设置map中的值为true,并且返回,否则进行下一步操作。 - m& a, J: _0 \. Q, ?; E5 \关于RabbitMQ的操作,我个人感觉还是有点迷的,毕竟真正的秒杀不会让你一直等待吧?(没事,咱也没秒杀到过东西,问题应该不是很大,也可能等待的时间很短人们感受不出来?这里请懂的朋友评论区指导指导呗) 4 [: j& s0 m% Z' z' V/ ]此项目中使用RabbitMQ将秒杀请求入队,前端进行轮询确认是否下单成功来实现异步下单的操作。关于RabbitMQ,RabbitMQ是一个开源的消息代理和队列服务器,用来通过普通协议在完全不同的应用之间共享数据,或者简单的将作业排队以便让分布式服务器进行处理。% B" ~- h4 V7 N5 }
RabbitMQ有四种交换机模式,分别是Direct模式、Topic模式、Fanout模式和Headers模式。Direct模式需要将一个队列绑定到交换机上,要求该信息与一个特定的路由键完全匹配;Topic模式是将路由键和某模式进行匹配,队列需要绑定到一个模式上;Fanout模式则是不处理路由键,只需要将队列绑定到交换机上,一个发送到该交换机的信息都会被转发到与该交换机绑定的所有队列上,类似于子网传播;Headers则是以key-value形式来匹配。 ) N5 j% j, N" x- l* V( d& Z# y" J队列和交换机绑定使用的是BindingBuilder.bind(队列).to(交换机)' {, _$ a0 G) R! _; i
' G3 y- A) A5 i' _! p注意此时秒杀操作并不在秒杀功能中处理,而是在RabbitMQ中的receive()方法中处理(见第三段代码). Q! b8 }* a8 ^" i
7 [$ p" A4 r( V2 ^
//用于保存商品的id和是否售空: Z# ^6 a8 |/ t6 ~5 H
private HashMap<Long, Boolean> localOverMap = new HashMap<Long, Boolean>(); 4 c9 Q; ^7 F# q ; V6 J$ r% n @( P( m; W: [ /**6 _6 t) W/ r+ S5 I. H2 H$ D1 x+ g
* 系统初始化 " v. L! a8 {6 `* ^: f * *// v* M* @9 a. W0 L( K
public void afterPropertiesSet() throws Exception {& h1 `4 C5 J; w3 M7 S) b3 n0 K
List<GoodsVo> goodsList = goodsService.listGoodsVo();1 |! p* l, o! H: P. O: }
if(goodsList == null) {& O U, |, O5 j" ?4 g
return; ) {( L: z. L ^* i* `; u } + n# A9 I6 g, N+ D6 m A+ S for(GoodsVo goods : goodsList) { # O+ C; ]* f. r" C2 I! ` redisService.set(GoodsKey.getMiaoshaGoodsStock, ""+goods.getId(), goods.getStockCount()); ( R" o2 y, t7 z' m% |% @% l localOverMap.put(goods.getId(), false);5 G; @3 W% o1 d, B
}% i) \+ j8 n: d! v
}0 e p$ f% a; h, O0 |* k; J
* h; [* x2 r" r
+ n" F9 G9 t! T) b @RequestMapping(value="/do_miaosha", method=RequestMethod.POST)" R4 e4 L* }9 ]# g+ S' x; C4 p* v
@ResponseBody# u5 s0 K' F- B1 U. p2 W
public Result<Integer> miaosha(Model model,MiaoshaUser user, ! C2 t# ? z0 I/ M @RequestParam("goodsId")long goodsId) { 7 j) j% x- T+ j G model.addAttribute("user", user); 1 x) i* Z4 ?3 X' P6 t# B8 ? if(user == null) {3 I# P9 t- N9 `* X1 `0 c5 s8 k
return Result.error(CodeMsg.SESSION_ERROR);: x* `$ {* u; g+ Q! p, J. l# w
} - V4 J3 x, U& [ //内存标记,减少redis访问 1 H; P/ B0 J5 | boolean over = localOverMap.get(goodsId);: O. {: y1 W; W; F
if(over) { 0 y6 B1 @0 ]: N return Result.error(CodeMsg.MIAO_SHA_OVER); 6 E1 q$ ~# ]8 i) U1 m, x* `6 S' L* N }5 _* Q% W( ~) y+ p N" {. V
//预减库存4 g3 ]6 ]& O6 e1 u, Y6 q
long stock = redisService.decr(GoodsKey.getMiaoshaGoodsStock, ""+goodsId);//10+ q {: ?- O' F* d% t4 D( R- l7 n
if(stock < 0) {6 \& m" D# b S4 ^
localOverMap.put(goodsId, true); 6 {: W2 \9 W. F3 V2 D6 [ return Result.error(CodeMsg.MIAO_SHA_OVER);. q8 j9 A' d7 x
}; [* J5 a3 [, p y# f9 n, k, O
//判断是否已经秒杀到了 8 e* Q6 _1 V9 n- w {% d& v& F MiaoshaOrder order = orderService.getMiaoshaOrderByUserIdGoodsId(user.getId(), goodsId); 2 k- a9 B. V2 J9 f2 ^/ o if(order != null) {0 \ c5 d! F8 Y9 _
return Result.error(CodeMsg.REPEATE_MIAOSHA);+ ]: C. [9 O5 P' l1 X
} " h9 D- O0 @. q6 I //入队: h4 r4 y2 ~' ^# X
MiaoshaMessage mm = new MiaoshaMessage();5 Q- `% n" D" K8 g- e4 S3 {" I
mm.setUser(user);) i4 x2 s5 _/ S' B* }" a
mm.setGoodsId(goodsId);( e$ k/ Y5 K6 j2 K* [3 F5 r2 z; Z
sender.sendMiaoshaMessage(mm);+ b/ V2 K4 M& I
return Result.success(0);//排队中 6 X; a. m' ]( S# g- i( a* g }+ G+ T8 o2 y2 `% l, }! Z
& M. E# d( q8 F& I, l0 x; y /**MQReceiver类中的方法( D2 b: Z# K3 ?$ T2 w3 Y
* 此时到达这里的请求大大减少 , }3 @5 }! J8 ^! c * 因为有redis库存的拦截; f; ?/ P7 }+ E. `
* @param message7 m; d. M& v% b9 ]2 {# l% E& f7 k
*/; @ ?' \4 X4 p" R$ m* N
@RabbitListener(queues = MQConfig.MIAOSHA_QUEUE) 5 q& Z) A! o7 {& J/ U+ c4 P# ? public void receive(String message){! M4 P" `$ O& d9 Y
log.info("receive message:"+message);9 y, V0 P1 }1 z- T6 E k
MiaoshaMessage mm = redisService.stringToBean(message, MiaoshaMessage.class); 9 i" ?( y8 P4 f# e KillsUser user = mm.getUser();5 N2 m2 ?6 {6 ~! Y1 Y: C3 \
long goodsId = mm.getGoodsId(); ) j( P1 H0 {' Q7 g5 ]5 q& E' i1 J: Y //判断库存 9 {: f; V4 }! {& X; w1 Z# f GoodsVo goodsVo = goodsService.getGoodsVoByGoodsId(goodsId); * x- ~' E- e, Q7 } H int stockCount = goodsVo.getStockCount();5 E5 j2 ? [ D1 c! w
//库存不足 1 |1 h1 X( _- f! m: J7 e# ` if(stockCount<=0){ j- f3 w1 |* T1 p return;6 _% M) x& K* E# i' U: Y' c
}' _, X, ^( R: K8 b8 L. y
//判断是否已经秒杀到了,这里可以不去数据库查找是否有订单 ) R9 s, ]3 Z' O% p2 b c MiaoshaOrder order = orderService.getMiaoshaOrderByUserIdGoodsId(user.getId(), goodsId); 5 ?( K! L- W& B; ^3 X( k if(order!=null){ ; x2 T' C! u; V: p return; : }2 j7 M* J9 W4 p0 \& ~/ `/ f } & p) z$ l! I5 U //减库存,下单,写入秒杀订单* l: v/ t* u% p2 D7 I" z
OrderInfo orderInfo = miaoshaService.miaosha(user, goodsVo);0 F# U% [. m& k/ }$ u. G
}* h$ ]% n% N3 i! A( x& j
10 m1 a( U( T. t- L# W0 j
2 # ~; I8 C1 q' Z E5 w3 G+ D% \% B- w& Y2 ]2 e/ Y
4! ?' `1 }7 [" G# z/ O9 S8 M8 p
5 5 Q2 K8 _ U' p8 o2 k$ I0 \6& s- w" {$ z, d4 V% r
70 T. s9 T; P2 V3 q
8 : H2 f8 \* i& r$ \; k4 v9 ! ]5 [: N( B7 c* o: L- S f10, _4 v4 d, Y- d8 E9 Y/ e, B1 F! m
11) P8 f, x8 R+ L: p& z
128 R! A3 K- h% v( @
13 - j/ F) q3 n/ w6 W# @' u144 n2 j* t/ X& w( m5 y
15 / h) P I- O4 ^0 [9 X16 : N8 i+ }& Z# Y0 s9 N) t17' b9 Z+ ?! q+ B$ ^
18 ( b1 I" o5 E9 u6 P7 A$ i0 o192 {% w7 U- l/ z$ F3 `
20% j. I8 ~' q" D8 r5 g. Y- w4 ]
21 5 h4 N, Y$ u3 ^; \" R! g& m22 ) y5 D% }: _4 c. |9 }23 " H9 o& Q- w' b! l! i( Y; {24 . F* m# r6 h8 H& F8 B9 F( E8 g25- [8 G+ V: k3 A9 u4 m* k
26 % S( y/ H l% N# M27 " h' i. e: A8 ?+ Y. s( |8 z: [7 W T28! h3 K1 @0 D2 j
29 4 H8 z# U5 B9 m) R9 N30+ @* r" p/ J6 B
314 w! T6 F* `3 ]$ X3 G' c* ~/ V5 \
321 ?- J, ?9 {) o& W, D
335 W% K2 p8 b; @) c' ^9 I4 F
34 * Y; O8 v. O* J+ f- W$ j35 ) T9 _1 Q6 `9 k4 m36& ]5 {1 H# ?5 T( k: {
37 : w" q& x' ^7 o) _- e$ t1 ]" g38 - A% U) h0 W0 r; ~4 B/ \$ |392 {) H5 F" g) g
40( {" y) S3 ?2 R, X* ~4 o
41 * r( E' \- M& ^42 ! N( D/ r; c7 X! U% Z( E. d: B43 % s' S5 `2 P4 }* p9 z445 b' G6 T! }3 }# x& w! ~
45" q" \2 ^; m1 \5 Y6 p; O
46 5 A8 H5 L k% x6 z0 [477 `. v' b, u. f! [$ n, R- F3 q! @
48$ v. T" l8 x: s) t. n
49( |: K3 r3 m, V! @' j8 G) ^6 z2 q. C
50 * \6 i4 c* j2 ~' d; [+ R0 d51 ; C: g! T5 B' e0 W z4 T+ J* q52 9 B; I9 w; z. m' b53 9 `& p( C A2 @" U54 ; a* u9 S9 w: d$ [8 V2 M% {4 f553 p/ [" }7 E4 d
56% m/ Y/ A9 f# z" V/ s
57# }. S$ {. k9 {$ ?, B/ H2 z
58" Q! D& j4 ?1 d' C& W! M
59 $ g. x- k, O ~/ k- n; C. q60 2 \4 r; L6 p5 ~% P" H, J& u61% N' q8 D6 |" P
62/ z; y, Y7 W$ h4 x6 f$ i/ T0 w5 o
63 , Y4 _2 g! r0 F- s' x0 {$ B64/ o7 n+ T/ |' ]/ c
659 ?7 T! G, O1 o& X' `
66: I; S1 D& T3 Q: u$ \# w7 j+ Z
67 9 q1 e: k: m" Z" P0 ?68 5 s+ J# ^( v% a! O ^$ R0 R6 W69" K/ j& s. M, D" A6 O* R- F
706 b2 k0 q0 j3 T, b3 j
71 : G; P4 M& l" y72% c5 Q) u" R2 V* g% y- w; ]& M- V' ]
739 g" t- |+ {# @6 C2 s7 o- e
74 : ^% d5 B, l) S J7 u756 e$ ?' u+ i/ ?8 @
76- |! |* r3 p+ J. _
五、安全优化; x* Z: a+ v9 h
|, v( X. ~' w1、秒杀接口地址隐藏& c2 Q, F# J3 I9 D' M4 c
3 w/ r: C9 k% M8 v& I- s在秒杀真正开始之前,我们的秒杀地址不应该是可以访问的,这样会造成极大的不公平,如果你获得秒杀地址就可以在秒杀开始之前买到商品。因此我们可以在地址中间加上一个随机的path,可以利用随机生成随机码,并把用户id和商品id作为key,把path存入redis中,在真正秒杀之前会再度把传入的path和redis中的path作比较。5 d; k$ u) W" o7 h; ^/ n4 U+ k
8 g$ @4 D$ T7 @0 q K$ n3 L
2、数学公式验证码4 G$ l0 n% q( j9 K, u% R
% j, X4 {5 F0 l4 j0 j6 s9 C) @* p
为了减少真正进入秒杀接口的用户,在商品详情页面进行一个验证码的输入,对算式进行计算并把值存进redis,在秒杀方面之前对用户 输入的验证码和redis中的值进行比较,从而判断是否进入秒杀。 # Q: u; t; o# `$ C3 {0 N$ A0 J9 Y9 C% m- S) ^6 W3 ?; y8 T9 K: {1 Q
3、接口防刷 8 B& J4 b p/ s0 x. l: V; Z# z9 R1 n2 R2 y
接口防刷是防止用户短时间内多次访问同一页面,可以避免某些脚本的攻击。原理是使用redis缓存,假如在k秒内访问超过5次就拦截,key值是固定前缀加上用户id,有效时间是k秒,而value则是从1开始,在不超过5次的前提下,调用redis的incr方法。如果超过限制,就拒绝访问,在拦截器的preHandle()方法中返回false,拒绝访问。' m: q+ |* y% Z0 h$ d
但是可能会有很多的接口需要限制,所以会出现冗余代码很多的情况,并且这样也会使得接口方法变得十分复杂,代码可读性也会降低。因此我们可以声明一个相关的注解,如下: " b$ F* j2 H0 g; |' }5 }- f$ n% ]* a' T1 @, U/ e
$ Q* q: I3 K* f3 y$ `/ x7 I0 a
@Override 3 o2 H* c; F" t' x- i6 @% ] public boolean preHandle(HttpServletRequest request, HttpServletResponse response,Object handler) 6 G3 v' m2 w- v: J8 |% w throws Exception{$ m: T3 }; e$ T1 t2 K
if(handler instanceof HandlerMethod){ + P0 J3 H# i; F. h: u0 A KillsUser user=getUser(request,response); + s- u, v$ r% M //将user对象存入ThreadLocal中! m/ ]/ [% m9 V6 c- ~* p! E0 h
UserContext.setUser(user);# s7 P$ Y. k F S& m
5 k" G! K! ]& |2 e6 H' H( w HandlerMethod hm=(HandlerMethod)handler; G- j# d9 {3 e* R AccessLimit accessLimit=hm.getMethodAnnotation(AccessLimit.class);$ I# X) d- P- ^* a
if(accessLimit==null){ ( E; x! u8 O* R) U9 g _( Y1 S( [ //说明没有这个注解 跳过% D- w3 b) A+ Q0 h' S
return true;1 ]8 J- C# p& K2 W
}2 o0 E, f7 h+ H% K9 X; F
8 G5 r) s5 l4 z5 m! W) n3 v int seconds=accessLimit.seconds();0 m2 U( ~9 i! ^9 v( S& p4 c( S
int maxCount=accessLimit.maxCount(); ; ~7 ^4 v3 a/ q( k boolean needLogin=accessLimit.needLogin(); & w% Y5 v' f4 |- Z String key=request.getRequestURI(); ! {$ W. h( u$ _ //这里判断获取的user是否为null cookie决定 8 O0 J3 U8 ^; E1 C% n" i if(needLogin){! z: v: E5 A. ?1 R# p. [, F# I
if(user==null){3 s* Q5 z9 X6 X' @2 D8 M3 {; S
render(response, CodeMsg.SESSION_ERROR);! I% s' J% s3 l, X; N5 k) x( T
return false; 4 |' q w6 P3 r5 G. n# u }. o0 @: ~" f7 F& C7 \4 V0 @# X
}else{/ e) m" T9 T) u
//do nothing / v5 _+ |; Z. Q }6 A1 X% `+ Y' v7 [0 D1 B# u' f- _
key+="_"+user.getId(); ' v7 A5 P2 d) i: F* W AccessKey ak=AccessKey.withExpire(seconds); : e3 Y5 N1 n% J+ q8 |7 o. W Integer count=redisService.get(ak,key,Integer.class); 3 H- v# B: R6 I* |9 s# H3 e if(count==null){ 0 `# P/ R, K; F1 @ redisService.set(ak,key,1); / o+ k j+ I3 G% Z: U7 v( k6 l }else if(count<maxCount){7 c) { ^5 Q! F/ j$ x J
redisService.incr(ak,key); , u' s6 k% A, ^- {/ ~4 p: s }else{//短时间内频繁访问 / u0 V( J b1 m$ ?0 Z( P3 T render(response,CodeMsg.FREQUENCY_ACCESS);) ], Q9 q6 a1 S
return false;: Z' e/ q# n" Z
} / ^- W5 p/ E, `$ H* M, } } 5 h d( b0 F2 q; E& B6 b return true;6 m g$ n/ `" O+ X1 `9 x) i3 k
} 8 m* n6 l. i/ w. ^& e% d, r5 k- V( e% z7 W) b
4、用户存储 ( M* C+ B, m" n6 C 9 Z0 l Z( v }2 M1 |2 a在多个方法中都需要使用秒杀用户作为参数,判断用户是否为空而决定需不需要跳转到登录页面。因此每次都要从数据库中查找相应的对象来返回。我们可以使用一个ThreadLocal来对User对象进行保存,而ThreadLocal的有效期就是在当前线程内,所以在用户访问的过程中都可以从本地拿到一个秒杀用户对象而不用访问数据库。3 W* T& l3 \6 ]
) X1 _' {+ ^7 N8 g* mpublic class UserContext { , E6 [8 x; u N$ Y7 i( s" Y7 k) ] //将user存到当前线程中 5 k, B# I+ ?9 u: T private static ThreadLocal<KillsUser> userHolder=new ThreadLocal<>(); * y+ X/ `' t3 n7 }, k0 U% _& z
public static void setUser(KillsUser user){ & \1 }$ y+ A! D9 U" q8 @- T userHolder.set(user); % e5 [$ V8 v2 k. y. [ } " W/ a7 a+ l: C# ?6 s 3 ~, ?) u% ]/ p+ \. l' D public static KillsUser getUser(){! p6 Y# R6 } K( d, c A" }& E
return userHolder.get(); # B* J/ a! Y. d/ w' s } 1 |4 M/ L# `) a! M. g} 4 f2 I0 \4 t: Y! E+ J- J$ M$ o: U; J0 Z# i1 G
六、个人感受 3 A9 j: m5 W* S9 f! Z; y0 q B# ` " R8 m |( j& a y. i对于高并发秒杀的解决方案,其实是有很多种的,通过这个项目让我这个小菜鸡初窥门径,了解到一些相关的解决方式,但其实里面有一些做法还是有待商榷的。6 p) W0 j; [% j; K! G% P
通过这个项目,还是认识到了redis的强大,有很大一部分优化都是通过redis缓存来实现的,关于redis还是应该多多学习。我自己对于此项目中的很多东西都是比较陌生的,比如Thymeleaf、JMeter和RabbitMQ都是刚刚接触的,如果问我有关的问题,我还是比较懵的,因此对于这些东西还是应该多了解。1 B% O; T) L! U" L
最后,找了两个多月的实习,我依然没有上岸,可能是我太菜了吧,希望大家跟我一起加油趴! 0 f. ]; N s3 i/ a2 p% i2 E———————————————— * Z; O* V5 W5 q, Q版权声明:本文为CSDN博主「fucccck_ly」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 . M" {1 P: |, a/ X5 ?' T& C( u/ ~原文链接:https://blog.csdn.net/fucccck_ly/article/details/106022784 % Z+ e3 }0 B8 H- b 9 V6 U0 e; F5 j/ T % P' E1 A, m$ ^