干貨 | 京東購(gòu)物車的Java架構(gòu)實(shí)現(xiàn)及原理!
今天來(lái)寫(xiě)一下關(guān)于購(gòu)物車的東西, 這里首先拋出四個(gè)問(wèn)題:
1)用戶沒(méi)登陸用戶名和密碼,添加商品, 關(guān)閉瀏覽器再打開(kāi)后 不登錄用戶名和密碼
問(wèn):購(gòu)物車商品還在嗎?
2)用戶登陸了用戶名密碼,添加商品,關(guān)閉瀏覽器再打開(kāi)后 不登錄用戶名和密碼
問(wèn):購(gòu)物車商品還在嗎?
3)用戶登陸了用戶名密碼,添加商品, 關(guān)閉瀏覽器,然后再打開(kāi),登陸用戶名和密碼
問(wèn):購(gòu)物車商品還在嗎?
4)用戶登陸了用戶名密碼,添加商品, 關(guān)閉瀏覽器 外地老家打開(kāi)瀏覽器 登陸用戶名和密碼
問(wèn):購(gòu)物車商品還在嗎?
上面四個(gè)問(wèn)題都是以京東為模板, 那么大家猜猜結(jié)果是什么呢?
1)在
2)不在了
3)在
4)在
如果你能夠猜到答案, 那么說(shuō)明你真的很棒, 那么關(guān)于這四點(diǎn)是怎么實(shí)現(xiàn)的呢??。ㄈ绻胁徽J(rèn)可的小伙伴可以用京東實(shí)驗(yàn)一下)
下面我們就來(lái)講解下購(gòu)物車的原理,***再來(lái)說(shuō)下具體的code實(shí)現(xiàn).
1)用戶沒(méi)有登錄, 添加商品, 此時(shí)的商品是被添加到了瀏覽器的Cookie中, 所以當(dāng)再次訪問(wèn)時(shí)(不登錄),商品仍然在Cookie中, 所以購(gòu)物車中的商品還是存在的.
2)用戶登錄了,添加商品, 此時(shí)會(huì)將Cookie中和用戶選擇的商品都添加到購(gòu)物車中, 然后刪除Cookie中的商品. 所以當(dāng)用戶再次訪問(wèn)(不登錄),此時(shí)Cookie中的購(gòu)物車商品已經(jīng)被刪除了, 所以此時(shí)購(gòu)物車中的商品不在了.
3)用戶登錄, 添加商品,此時(shí)商品被添加到數(shù)據(jù)庫(kù)做了持久化存儲(chǔ), 再次打開(kāi)登錄用戶名和密碼, 該用戶選擇的商品肯定還是存在的, 所以購(gòu)物車中的商品還是存在的.
4)理由3)
這里再說(shuō)下 沒(méi)登錄 保存商品到Cookie的優(yōu)點(diǎn)以及保存到Session和數(shù)據(jù)庫(kù)的對(duì)比:
1:Cookie: 優(yōu)點(diǎn): 保存用戶瀏覽器(不用浪費(fèi)我們公司的服務(wù)器) 缺點(diǎn):Cookie禁用,不提供保存
2:Session:(Redis : 浪費(fèi)大量服務(wù)器內(nèi)存:實(shí)現(xiàn)、禁用Cookie) 速度很快
3:數(shù)據(jù)庫(kù)(Mysql、Redis、SOlr) 能持久化的就數(shù)據(jù)庫(kù) 速度太慢
那么我今天要講的就是:
- 用戶沒(méi)登陸:購(gòu)物車添加到Cookie中
- 用戶登陸: 保存購(gòu)物車到Redis中 (不用數(shù)據(jù)庫(kù))
整體的思路圖解:
接下來(lái)就是代碼實(shí)例來(lái)實(shí)現(xiàn) 購(gòu)物車的功能了:
首先我們看下購(gòu)物車和購(gòu)物項(xiàng)兩個(gè)JavaBean的設(shè)計(jì):
購(gòu)物車: buyerCart.java
這里使用了@JsonIgonre注解是因?yàn)橄旅嫘枰獙uyerCart 轉(zhuǎn)換成Json格式, 而這幾個(gè)字段只有g(shù)et 方法, 所以不能轉(zhuǎn)換, 需要使用忽略Json.
下面是購(gòu)物項(xiàng): buyerItem.java
1、將商品加入購(gòu)物車中
這里傳入的參數(shù)是skuId(庫(kù)存表的主鍵, 庫(kù)存表保存的商品id,顏色,尺碼,庫(kù)存等信息), 購(gòu)買數(shù)量amount.
接著我們來(lái)看Controller是如何來(lái)處理的:
這里設(shè)計(jì)一個(gè)知識(shí)點(diǎn): 將對(duì)象轉(zhuǎn)換成json字符串/json字符串轉(zhuǎn)成對(duì)象
我們?cè)谶@里先寫(xiě)一個(gè)小的Demo來(lái)演示json和對(duì)象之間的互轉(zhuǎn), 這里使用到了springmvc中的ObjectMapper類.
執(zhí)行結(jié)果:
這里我們使用了Include.NON_NULL, 如果TestTb 中屬性為null 的就不給轉(zhuǎn)換成Json, 從對(duì)象-->Json字符串 用的是 objectMapper.writeValue(). 從Json字符串-->對(duì)象使用的是objectMapper.readValue().
回歸上面我們項(xiàng)目中的代碼, 只有未登錄 添加商品時(shí)才會(huì)將此商品添加到Cookie中.
我們debug 可以看到:
這里已經(jīng)將對(duì)象購(gòu)物車對(duì)象buyerCart轉(zhuǎn)換成了Json格式.
將商品添加到購(gòu)物車, 不管是登錄還是未登錄, 都要先取出Cookie中的購(gòu)物車, 然后將當(dāng)前選擇的商品追加到購(gòu)物車中.
然后登錄的話 就把Cookie中的購(gòu)物車清空, 并將購(gòu)物車的內(nèi)容添加到Redis中做持久化保存.
如果未登錄, 將選擇的商品追加到Cookie中.
將購(gòu)物車追加到Redis中的代碼:insertBuyerCartToRedis(這里面包含了判斷添加的是否是同款)
判斷用戶是否登錄: String username =
sessionProviderService.getAttributterForUsername(RequestUtils.getCSessionId(request, response));
2、購(gòu)物車展示頁(yè)面
*** 重定向到購(gòu)物車展示頁(yè): return "redirect:/shopping/toCart"; 這里進(jìn)入結(jié)算頁(yè)有兩種方式:
1) 在商品詳情頁(yè) 點(diǎn)擊加入購(gòu)物車.
2) 直接點(diǎn)擊購(gòu)物車按鈕 進(jìn)入購(gòu)物車結(jié)算頁(yè).
下面來(lái)看下結(jié)算頁(yè)的代碼:
這里 就是 購(gòu)物車詳情展示頁(yè)面, 這里需要注意, 如果是同一件商品連續(xù)添加, 是需要合并的.
購(gòu)物車詳情展示頁(yè)面就包括兩大塊, 1) 商品詳情 2)總計(jì)(商品總額,運(yùn)費(fèi))
其中1)商品詳情又包括 商品尺碼,商品顏色, 商品購(gòu)買數(shù)量, 是否有貨.
取出Redis中的購(gòu)物車: buyerCart = cartService.selectBuyerCartFromRedis(username);
將購(gòu)物車裝滿, 前面只是將skuId裝進(jìn)購(gòu)物車, 這里還需要查出sku詳情: List<BuyerItem> items = buyerCart.getItems();
buyerItem.setSku(cartService.selectSkuById(buyerItem.getSku().getId()));
接著就返回"cart.jsp", 這個(gè)就是購(gòu)物車詳情展示頁(yè)面了.
3、去結(jié)算頁(yè)面
到了這里就說(shuō)明用戶必須要 登錄, 而且購(gòu)物車中必須要有商品.
所以這里我么你需要利用springmvc的過(guò)濾功能, 用戶點(diǎn)擊結(jié)算的時(shí)候必須要先登錄, 如果沒(méi)有登錄的話就提示用戶需要登錄.
取出 所指定的購(gòu)物車, 因?yàn)槲覀兘Y(jié)算之前在購(gòu)物車詳情頁(yè)面會(huì)勾選 我們 需要購(gòu)買的商品, 所以這里是根據(jù)所勾選的商品去結(jié)算的.
BuyerCart buyerCart = cartService.selectBuyerCartFromRedisBySkuIds(skuIds, username);
從購(gòu)物車中取出指定商品:
1) 當(dāng)我們購(gòu)買的商品只要有一件是無(wú)貨的狀態(tài), 那么刷新購(gòu)物車詳情頁(yè)面, 回顯無(wú)貨的商品狀態(tài).
2)當(dāng)購(gòu)物車中午商品時(shí), 刷新當(dāng)前頁(yè)面.
購(gòu)物車就這么多東西, 可能有講解不到或者錯(cuò)誤的地方, 歡迎大家指出來(lái).如果對(duì)你有幫助的話也請(qǐng)點(diǎn)個(gè)贊支持一下,謝謝~