/**
* 實現使用者的登出操作 要求刪除cookie 和redis中的資料(key)
* 步驟: 通過cookie獲取ticket資訊.
* url: http://www.jt.com/user/logout.html
* 引數: 暫時沒有
* 返回值: 重定向到系統首頁
*/
@RequestMapping("/logout")
public String logout(HttpServletRequest request,HttpServletResponse response){
Cookie[] cookies = request.getCookies();
if(cookies !=null && cookies.length >0){
for(Cookie cookie : cookies){
if("JT_TICKET".equals(cookie.getName())){
//獲取value之後刪除cookie
String ticket = cookie.getValue();
jedisCluster.del(ticket); //刪除redis中的資料
//刪除cookie時 必須與原來的cookie資料保持一致
cookie.setDomain("jt.com");
cookie.setPath("/");
cookie.setMaxAge(0);
response.addCookie(cookie);
break;
}
}
}
return "redirect:/";
}
package com.jt.util;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class CookieUtil {
/**
* 該工具API主要的任務
* 1.根據cookie的名稱 返回cookie物件
* 2.根據cookie的名稱 返回valve的值
* 3.新增cookie方法
* 4.刪除cookie方法
*/
public static Cookie getCookie(String cookieName, HttpServletRequest request){
Cookie[] cookies = request.getCookies();
if(cookies !=null && cookies.length >0) {
for (Cookie cookie : cookies) {
if (cookieName.equals(cookie.getName())) {
return cookie;
}
}
}
return null ;
}
public static String getCookieValue(String cookieName,HttpServletRequest request){
Cookie cookie = getCookie(cookieName, request);
return cookie ==null?null:cookie.getValue();
}
public static void addCookie(String cookieName, String cookieValue, String path,
String domain, int maxAge, HttpServletResponse response){
Cookie cookie = new Cookie(cookieName,cookieValue);
cookie.setPath(path);
cookie.setDomain(domain);
cookie.setMaxAge(maxAge);
response.addCookie(cookie);
}
public static void deleteCookie(String cookieName,String path,
String domain,HttpServletResponse response){
addCookie(cookieName,"",path, domain, 0, response);
}
}
/**
* 實現使用者的登出操作 要求刪除cookie 和redis中的資料(key)
* 步驟: 通過cookie獲取ticket資訊.
* url: http://www.jt.com/user/logout.html
* 引數: 暫時沒有
* 返回值: 重定向到系統首頁
*/
@RequestMapping("/logout")
public String logout(HttpServletRequest request,HttpServletResponse response){
String ticket = CookieUtil.getCookieValue("JT_TICKET", request);
if(!StringUtils.isEmpty(ticket)){
//1.刪除redis
jedisCluster.del(ticket);
//2.刪除cookie
CookieUtil.deleteCookie("JT_TICKET", "/", "jt.com", response);
}
return "redirect:/";
/* Cookie[] cookies = request.getCookies();
if(cookies !=null && cookies.length >0){
for(Cookie cookie : cookies){
if("JT_TICKET".equals(cookie.getName())){
//獲取value之後刪除cookie
String ticket = cookie.getValue();
jedisCluster.del(ticket); //刪除redis中的資料
//刪除cookie時 必須與原來的cookie資料保持一致
cookie.setDomain("jt.com");
cookie.setPath("/");
cookie.setMaxAge(0);
response.addCookie(cookie);
break;
}
}
}*/
}
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<artifactId>jt-cart</artifactId>
<parent>
<artifactId>jt</artifactId>
<groupId>org.example</groupId>
<version>1.0-SNAPSHOT</version>
</parent>
<!--新增依賴項-->
<dependencies>
<dependency>
<groupId>org.example</groupId>
<artifactId>jt-common</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
</dependencies>
<!--新增外掛-->
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
@TableName("tb_cart")
@Data
@Accessors(chain = true)
public class Cart extends BasePojo{
@TableId(type = IdType.AUTO)
private Long id; //主鍵自增
private Long userId; //使用者id
private Long itemId; //商品id
private String itemTitle;
private String itemImage;
private Long itemPrice;
private Integer num;
}
server:
port: 8094
servlet:
context-path: /
spring:
datasource:
#引入druid資料來源
#type: com.alibaba.druid.pool.DruidDataSource
driver-class-name: com.mysql.cj.jdbc.Driver
url: jdbc:mysql://127.0.0.1:3306/jtdb?serverTimezone=GMT%2B8&useUnicode=true&characterEncoding=utf8&autoReconnect=true&allowMultiQueries=true
username: root
password: root
#提供了MVC的支援
mvc:
view:
prefix: /WEB-INF/views/
suffix: .jsp
#mybatis-plush設定
mybatis-plus:
type-aliases-package: com.jt.pojo
mapper-locations: classpath:/mybatis/mappers/*.xml
configuration:
map-underscore-to-camel-case: true
logging:
level:
com.jt.mapper: debug
#關於Dubbo設定
dubbo:
scan:
basePackages: com.jt #指定dubbo的包路徑
application: #應用名稱
name: provider-cart #一個介面對應一個服務名稱
registry: #zk叢集 主機中的資訊與從機中的資訊一致的 從zk中獲取資料的時候連結的從機 主機的作用就是監控叢集
address: zookeeper://192.168.126.129:2181?backup=192.168.126.129:2182,192.168.126.129:2183
protocol: #指定協定
name: dubbo #使用dubbo協定(tcp-ip) web-controller直接呼叫sso-Service
port: 20881 #每一個服務都有自己特定的埠 不能重複.
1). 頁面url分析
2).業務說明
當使用者點選購物車按鈕時,應該根據userId查詢購物車資料資訊,之後在cart.jsp中展現列表資訊.
@Controller
@RequestMapping("/cart")
public class CartController {
@Reference(check = false,timeout = 3000)
private DubboCartService cartService;
/**
* 1.購物車列表資料展現
* url地址: http://www.jt.com/cart/show.html
* 引數: 暫時沒有
* 返回值: 頁面邏輯名稱 cart.jsp
* 頁面取值: ${cartList}
* 應該將資料新增到域物件中 Request域 model工具API操作request物件
*
*/
@RequestMapping("/show")
public String show(Model model){
//1.暫時將userId寫死 7L
Long userId = 7L;
List<Cart> cartList = cartService.findCartListByUserId(userId);
model.addAttribute("cartList",cartList);
return "cart";
}
}
package com.jt.service;
import com.alibaba.dubbo.config.annotation.Service;
import com.baomidou.mybatisplus.core.conditions.query.Query;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.jt.mapper.CartMapper;
import com.jt.pojo.Cart;
import org.springframework.beans.factory.annotation.Autowired;
import java.util.List;
@Service
public class DubboCartServiceImpl implements DubboCartService{
@Autowired
private CartMapper cartMapper;
@Override
public List<Cart> findCartListByUserId(Long userId) {
QueryWrapper<Cart> queryWrapper = new QueryWrapper<>();
queryWrapper.eq("user_id", userId);
return cartMapper.selectList(queryWrapper);
}
}
$(".increment").click(function(){//+
var _thisInput = $(this).siblings("input");
_thisInput.val(eval(_thisInput.val()) + 1);
$.post("/cart/update/num/"+_thisInput.attr("itemId")+"/"+_thisInput.val(),function(data){
TTCart.refreshTotalPrice();
});
});
/**
* 業務說明: 完成購物車數量的更新操作
* url地址: http://www.jt.com/cart/update/num/562379/9
* 引數: 暫時沒有
* 返回值: void
*/
@RequestMapping("/update/num/{itemId}/{num}")
@ResponseBody //ajax結束的識別符號.
public void updateCartNum(Cart cart){ //如果{name} 與屬性的名稱一致,則可以自動的賦值.
Long userId = 7L;
cart.setUserId(userId);
cartService.updateCartNum(cart);
}
/**
* 更新購物車的數量 num
* @param cart
* Sql: update tb_cart set num = #{num},updated = now()
* where user_id = #{user} and item_id = #{itemId}
*/
@Override
public void updateCartNum(Cart cart) { //itemId,userId
Cart cartTemp = new Cart();
cartTemp.setNum(cart.getNum());
UpdateWrapper<Cart> updateWrapper = new UpdateWrapper<>();
updateWrapper.eq("user_id", cart.getUserId())
.eq("item_id", cart.getItemId());
cartMapper.update(cartTemp,updateWrapper);
}
說明: 根據itemId和userId刪除購物車資料.重定向到購物車列表頁面即可.
/**
* 刪除購物車資料
* url地址:http://www.jt.com/cart/delete/562379.html
* 引數: 562379
* 返回值: 購物車列表頁面
*/
@RequestMapping("/delete/{itemId}")
public String deleteCart(Cart cart){
Long userId = 7L;
cart.setUserId(userId);
cartService.deleteCart(cart);
return "redirect:/cart/show.html";
}
@Override
public void deleteCart(Cart cart) { //itemId/userId
//根據物件中不為null的元素充當where條件
cartMapper.delete(new QueryWrapper<>(cart));
}
說明: 如果使用者點選購入購物車時,應該將使用者的提交資料插入到tb_cart表中,並且重定向到購物車列表頁面.
注意事項: 如果使用者重複提交資料,則應該數量更新.
2).引數說明
<form id="cartForm" method="post">
<input class="text" id="buy-num" name="num" value="1" onkeyup="setAmount.modify('#buy-num');"/>
<input type="hidden" class="text" name="itemTitle" value="${item.title }"/>
<input type="hidden" class="text" name="itemImage" value="${item.images[0]}"/>
<input type="hidden" class="text" name="itemPrice" value="${item.price}"/>
</form>
/**
* 業務需求: 完成購物車入庫操作
* url地址: http://www.jt.com/cart/add/562379.html
* 引數: form表單提交的資料/itemId
* 返回值: 重定向到購物車列表頁面
*/
@RequestMapping("/add/{itemId}")
public String addCart(Cart cart){
Long userId = 7L;
cart.setUserId(userId);
cartService.addCart(cart);
return "redirect:/cart/show.html";
}
/**
* 思路:
* 1.先查詢資料庫 user_id/item_id
* 2.判斷返回值是否有結果
* true: 只做數量的更新
* false: 直接入庫即可
*/
@Override
public void addCart(Cart cart) {
QueryWrapper<Cart> queryWrapper = new QueryWrapper<>();
queryWrapper.eq("user_id", cart.getUserId());
queryWrapper.eq("item_id", cart.getItemId());
Cart cartDB = cartMapper.selectOne(queryWrapper);
if(cartDB == null){
//新增入庫
cartMapper.insert(cart);
}else{
//更新數量 MP方式
int num = cart.getNum() + cartDB.getNum(); //數量求和
cartDB.setNum(num).setUpdated(new Date());
cartMapper.updateCartNum(cartDB);
}
}
public interface CartMapper extends BaseMapper<Cart> {
@Update("update tb_cart set num=#{num},updated=#{updated} where user_id=#{userId} and item_id =#{itemId}")
void updateCartNum(Cart cartDB);
}
說明: 新增dubbo的設定即可
說明:當使用者點選某個商品時,需要跳轉到商品的詳情頁面中.根據itemId展現商品資訊.並且在item.jsp中展現資料.
@Controller
public class ItemController {
@Reference(check = false,timeout = 3000)
private DubboItemService itemService;
/**
* 業務: 根據itemId查詢商品資訊(item/itemDesc)
* url地址: http://www.jt.com/items/562379.html
* 引數: 注意restFul風格即可
* 返回值: 頁面邏輯名稱 item.jsp
* 頁面取值: ${item.title }/${itemDesc.itemDesc }
* 5分鐘完成.
* */
@RequestMapping("/items/{itemId}")
public String findItemById(@PathVariable Long itemId, Model model){
Item item = itemService.findItemById(itemId);
ItemDesc itemDesc = itemService.findItemDescById(itemId);
model.addAttribute("item", item);
model.addAttribute("itemDesc", itemDesc);
return "item";
}
}
package com.jt.web.service;
import com.alibaba.dubbo.config.annotation.Service;
import com.jt.mapper.ItemDescMapper;
import com.jt.mapper.ItemMapper;
import com.jt.pojo.Item;
import com.jt.pojo.ItemDesc;
import com.jt.service.DubboItemService;
import org.springframework.beans.factory.annotation.Autowired;
@Service //dubbo的註解
public class DubboItemServiceImpl implements DubboItemService {
@Autowired
private ItemMapper itemMapper; //商品資訊
@Autowired
private ItemDescMapper itemDescMapper; //商品詳情資訊.
@Override
public Item findItemById(Long itemId) {
return itemMapper.selectById(itemId);
}
@Override
public ItemDesc findItemDescById(Long itemId) {
return itemDescMapper.selectById(itemId);
}
}