1 数据库创建

CREATE TABLE t_cart (
	cid INT AUTO_INCREMENT COMMENT '购物车数据id',
	uid INT NOT NULL COMMENT '用户id',
	pid INT NOT NULL COMMENT '商品id',
	price BIGINT COMMENT '加入时商品单价',
	num INT COMMENT '商品数量',
	created_user VARCHAR(20) COMMENT '创建人',
	created_time DATETIME COMMENT '创建时间',
	modified_user VARCHAR(20) COMMENT '修改人',
	modified_time DATETIME COMMENT '修改时间',
	PRIMARY KEY (cid)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

 2 创建实体类 

public class Cart extends BaseEntity{
    private Integer cid;
    private Integer uid;
    private Integer pid;
    private Long price;
    private Integer num;

3 持久层 

3.1 规划需要执行的SQL语句

1.向购物车表中插入数据

insert into t_catr values (值列表)

2.当当前的商品已经在购物车中存在,则直接更新num的数量即可

update t_cart set num=? where cid=? 

3.在插入或者更新具体执行哪个语句,取决于数据库中是否有当前这个购物车商品的数据,得去查询才能确定。

select * from t_cart where cid=? and uid=? 

3.2 设计接口和抽象方法

创建一个CartMapper接口持久层的文件。 

package com.cy.store.mapper;

import com.cy.store.entity.Cart;

import java.util.Date;

public interface CartMapper {
    /**
     * 插入购物车数据
     * @param cart 购物车数据
     * @return 受影响的行数
     */
    Integer insert(Cart cart);

    /**
     * 更新购物车某件商品的数据
     * @param cid 购物id
     * @param num 更新的数量
     * @param modifiedUser 修改者
     * @param modifiedTime 修改事件
     * @return 受影响行数
     */
    Integer updateNumByCid(Integer cid, Integer num , String modifiedUser, Date modifiedTime);

    /**
     * 根据用户id和商品的id来查询购物车中的数据
     * @param uid 用户id
     * @param pid 商品id
     * @return 
     */
    Cart findByUidAndPid(Integer uid,Integer pid);
}

3.3 SQL映射 

创建一个CartMapper.xml映射文件。添加以上三个抽象方法的SQL语句的映射。

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<!-- namespace属性:用于指定当前的映射文件和哪个接口进行映射,需要指定接口的文件路径,需要标注包的完整路径 -->
<mapper namespace="com.cy.store.mapper.CartMapper">
    <resultMap id="CartEntityMap" type="com.cy.store.entity.Cart">
        <id column="cid" property="cid"/>
        <result column="created_user" property="createdUser"/>
        <result column="created_time" property="createdTime"/>
        <result column="modified_user" property="modifiedUser"/>
        <result column="modified_time" property="modifiedTime"/>
    </resultMap>

    <insert id="insert" useGeneratedKeys="true" keyProperty="cid">
        insert into t_cart (uid,pid,price,num,created_user, created_time, modified_user,
                            modified_time)
        values (#{uid},#{pid},#{price},#{num}, #{createdUser},
                #{createdTime}, #{modifiedUser}, #{modifiedTime})
    </insert>

    <update id="updateNumByCid">
        update t_cart
        set num=#{num},modified_use#{modified_use},modified_time={#modified_time}
        where 
            cid=#{cid}
    </update>
    
    <select id="findByUidAndPid" resultMap="CartEntityMap">
        select * from t_cart where  uid=#{uid} and pid=#{pid}
    </select>
    


    
</mapper>

3.4 测试

package com.cy.store.mapper;

import com.cy.store.entity.Address;
import com.cy.store.entity.Cart;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;

import java.util.Date;
import java.util.List;

//@SpringBootTest:表示标注当前的类i是一个测试类,不会随打包发送
@SpringBootTest
//@RunWith:表示启动这个单元测试类(是不能够运行的),需要传递一个参数,必须时SpringRunner的实例类型
@RunWith(SpringRunner.class)
public class CartMapperTests {
    @Autowired
    private CartMapper cartMapper;
    @Test
    public void test()
    {
        Cart cart=new Cart();
        cart.setUid(8);
        cart.setPid(10000001);
        cart.setNum(3);
        cart.setPrice(10000L);
        cartMapper.insert(cart);
    }
    @Test
    public void updateNumByCid(){
        cartMapper.updateNumByCid(2,4,"张三",new Date());
    }

    @Test
    public void findByUidAndPid(){
        Cart cart=cartMapper.findByUidAndPid(9,10000001);
        System.out.println(cart);
    }




}

4 业务层 

4.1 规划异常

1.插入数据时可能产生异常:InsertException

2.更新数据时产生异常:updateException

4.2 接口和抽象方法

创建一个ICartService接口文件。

public interface ICartService {
    /**
     * 将商品添加到购物车中
     * @param uid 用户id
     * @param pid 商品id
     * @param amount 新增数量
     * @param username 用户名
     */
    void addToCart(Integer uid,Integer pid,Integer amount,String username);
}

4.3 实现接口

创建一个CartServiceImpl实现类 

package com.cy.store.service.impl;

import com.cy.store.entity.Address;
import com.cy.store.entity.Cart;
import com.cy.store.entity.Product;
import com.cy.store.mapper.AddressMapper;
import com.cy.store.mapper.CartMapper;
import com.cy.store.mapper.ProductMapper;
import com.cy.store.service.IAddressService;
import com.cy.store.service.ICartService;
import com.cy.store.service.IDistrictService;
import com.cy.store.service.ex.*;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;

import java.util.Date;
import java.util.List;

@Service
/**购物车的实现类**/
public class CartServiceImpl implements ICartService {
   /**购物车的业务层依赖于购物车的持久层以及商品的持久层**/
    @Autowired
    private CartMapper cartMapper;
    @Autowired
    private ProductMapper productMapper;

    @Override
    public void addToCart(Integer uid, Integer pid, Integer amount, String username) {
       //查询当前要添加的这个购物车是否在表中已存在
        Cart result= cartMapper.findByUidAndPid(uid, pid);
        Date date = new Date();
        if(result==null){
            //表示商品从来没有添加到购物车中,进行新增操作
            //需要创建一个cart对象
            Cart cart = new Cart();
            //补全数据:参数传递的数据
            cart.setUid(uid);
            cart.setPid(pid);
            cart.setNum(amount);
            //价格:来自于商品中的数据
            Product product = productMapper.findById(pid);
            cart.setPrice(product.getPrice());
            //补全日志
            cart.setCreatedUser(username);
            cart.setCreatedTime(date);
            cart.setModifiedUser(username);
            cart.setModifiedTime(date);
            //执行输入操作
            Integer rows = cartMapper.insert(cart);
            if(rows!=1){
                throw new InsertException("插入失败");
            }
        }else{//反之表示当前的商品在购物车中,更新num值
            Integer num=result.getNum()+amount;
            Integer rows = cartMapper.updateNumByCid(result.getCid(),
                    num,
                    username, date);
            if(rows!=1){
                throw new UpdateException("更新异常");
            }
        }
    }
}

 创建对应测试类CartServiceTest.

package com.cy.store.service;

import com.cy.store.entity.Product;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;

import java.util.List;

@SpringBootTest
@RunWith(SpringRunner.class)
public class CartServiceTests {
    @Autowired
    private ICartService cartService;
    @Test

    public void addToCart(){
        //已存在的一件购物车数据的更新操作
        cartService.addToCart(9,10000001,50000,"张明");
    }


}

5 控制层

 1.没有需要处理的异常

2,设计请求处理

/carts/add_to_cart

get

pid amount session

JsonResult<Void>

3.完成请求处理方法的编写。创建一个CartController类。

@RestController
@RequestMapping("carts")
public class CartController extends BaseController{
    @Autowired
    private ICartService cartService;

    @RequestMapping("add_to_cart")
    public JsonResult<Void> addToCart(Integer pid, Integer amount, HttpSession session){
        cartService.addToCart(getUidFromSession(session),pid,amount,getUsernameFromSession(session));
        return new JsonResult<>(OK);
    }

}

5 前端页面 

 在product.html页面给加入购物车按钮添加点击事件,并发送ajax请求。

$("#btn-add-to-cart").click(function () {
				$.ajax({
					url:"/carts/add_to_cart",
					type:"POST",
					data:{
						"pid":id,
						"amount": $("#num").val()
					},
					dataType:"JSON",
					success:function (json) {
						if(json.state==200){
							console.log(json.data)
							alert("加入购物车成功")
						}else{
							alert("加入购物车失败")
						}
					},
					error:function (xhr) {
						alert("加入购物车时产生未知的异常"+xhr.message)
					}
				})
			});

在ajax函数中data参数的数据设置方式:

·data:$("form表单选择").seriaize().当参数过多并且在同一个表单中,字符串的提交等

·data:new FormData($("form表单选择"))[0]只适用提交文件。

·data:“username=tom".适合参数值固定并且参数值列表有限,可以进行手动拼接。

·适用json格式提交数据

data:{
”username”=“tom”,
“age”=18
}