电商平台搭建--订单管理模块开发(一)

Hi,大家好,我是Steafan!项目开发到这里,基本上已经进行了90%了。项目中最关键的也是最难开发的就是今天我要向大家介绍的订单管理模块了。在实际的项目开发中,订单管理模块和支付模块往往是并行开发的,在开发过程中往往会遇到这样的情景:两个后台开发工程师,一个在开发订单管理模块,另一个在开发支付模块,恰好呢,在开发订单管理模块时会使用到支付模块中的部分功能,这时候工程师还没有写完支付模块,这就会产生一个等待的问题。无论让这两个工程师谁去等待谁都是不合理的,对于这样的情景,我们就应该学会抽象开发另一种功能的能力。

一、订单管理模块-概述

      先来看看我们需要在订单管理模块实现哪些功能

    由图可见,订单管理分为前台和后台。先来实现前台的功能。

在订单管理模块前台开发中,最复杂的要数订单号的生成规则和处理支付宝回调信息,在讲解该功能点时我会花费较大的篇幅去讲解,希望大家能够看明白。

二、订单管理模块-涉及到的VO

(1)、OrderItemVo

 
  1. public class OrderItemVo {

  2.  
  3. private Long orderNo;

  4.  
  5. private Integer productId;

  6.  
  7. private String productName;

  8. private String productImage;

  9.  
  10. private BigDecimal currentUnitPrice;

  11.  
  12. private Integer quantity;

  13.  
  14. private BigDecimal totalPrice;

  15.  
  16. private String createTime;

  17.  
  18. public Long getOrderNo() {

  19. return orderNo;

  20. }

  21.  
  22. public void setOrderNo(Long orderNo) {

  23. this.orderNo = orderNo;

  24. }

  25.  
  26. public Integer getProductId() {

  27. return productId;

  28. }

  29.  
  30. public void setProductId(Integer productId) {

  31. this.productId = productId;

  32. }

  33.  
  34. public String getProductName() {

  35. return productName;

  36. }

  37.  
  38. public void setProductName(String productName) {

  39. this.productName = productName;

  40. }

  41.  
  42. public String getProductImage() {

  43. return productImage;

  44. }

  45.  
  46. public void setProductImage(String productImage) {

  47. this.productImage = productImage;

  48. }

  49.  
  50. public BigDecimal getCurrentUnitPrice() {

  51. return currentUnitPrice;

  52. }

  53.  
  54. public void setCurrentUnitPrice(BigDecimal currentUnitPrice) {

  55. this.currentUnitPrice = currentUnitPrice;

  56. }

  57.  
  58. public Integer getQuantity() {

  59. return quantity;

  60. }

  61.  
  62. public void setQuantity(Integer quantity) {

  63. this.quantity = quantity;

  64. }

  65.  
  66. public BigDecimal getTotalPrice() {

  67. return totalPrice;

  68. }

  69.  
  70. public void setTotalPrice(BigDecimal totalPrice) {

  71. this.totalPrice = totalPrice;

  72. }

  73.  
  74. public String getCreateTime() {

  75. return createTime;

  76. }

  77.  
  78. public void setCreateTime(String createTime) {

  79. this.createTime = createTime;

  80. }

  81. }

    这里存储了订单号和商品有关的信息。

(2)、OrderProductVo

 
  1. public class OrderProductVo {

  2. private List<OrderItemVo> orderItemVoList;

  3. private BigDecimal productTotalPrice;

  4. private String imageHost;

  5.  
  6. public List<OrderItemVo> getOrderItemVoList() {

  7. return orderItemVoList;

  8. }

  9.  
  10. public void setOrderItemVoList(List<OrderItemVo> orderItemVoList) {

  11. this.orderItemVoList = orderItemVoList;

  12. }

  13.  
  14. public BigDecimal getProductTotalPrice() {

  15. return productTotalPrice;

  16. }

  17.  
  18. public void setProductTotalPrice(BigDecimal productTotalPrice) {

  19. this.productTotalPrice = productTotalPrice;

  20. }

  21.  
  22. public String getImageHost() {

  23. return imageHost;

  24. }

  25.  
  26. public void setImageHost(String imageHost) {

  27. this.imageHost = imageHost;

  28. }

  29. }

    这里存储了和商品有关的订单信息。

(3)、OrderVo

 
  1. public class OrderVo {

  2.  
  3. private Long orderNo;

  4.  
  5. private BigDecimal payment;

  6.  
  7. private Integer paymentType;

  8.  
  9. private String paymentTypeDesc;

  10. private Integer postage;

  11.  
  12. private Integer status;

  13.  
  14.  
  15. private String statusDesc;

  16.  
  17. private String paymentTime;

  18.  
  19. private String sendTime;

  20.  
  21. private String endTime;

  22.  
  23. private String closeTime;

  24.  
  25. private String createTime;

  26.  
  27. //订单的明细

  28. private List<OrderItemVo> orderItemVoList;

  29.  
  30. private String imageHost;

  31. private Integer shippingId;

  32. private String receiverName;

  33.  
  34. private ShippingVo shippingVo;

  35.  
  36. public Long getOrderNo() {

  37. return orderNo;

  38. }

  39.  
  40. public void setOrderNo(Long orderNo) {

  41. this.orderNo = orderNo;

  42. }

  43.  
  44. public BigDecimal getPayment() {

  45. return payment;

  46. }

  47.  
  48. public void setPayment(BigDecimal payment) {

  49. this.payment = payment;

  50. }

  51.  
  52. public Integer getPaymentType() {

  53. return paymentType;

  54. }

  55.  
  56. public void setPaymentType(Integer paymentType) {

  57. this.paymentType = paymentType;

  58. }

  59.  
  60. public String getPaymentTypeDesc() {

  61. return paymentTypeDesc;

  62. }

  63.  
  64. public void setPaymentTypeDesc(String paymentTypeDesc) {

  65. this.paymentTypeDesc = paymentTypeDesc;

  66. }

  67.  
  68. public Integer getPostage() {

  69. return postage;

  70. }

  71.  
  72. public void setPostage(Integer postage) {

  73. this.postage = postage;

  74. }

  75.  
  76. public Integer getStatus() {

  77. return status;

  78. }

  79.  
  80. public void setStatus(Integer status) {

  81. this.status = status;

  82. }

  83.  
  84. public String getStatusDesc() {

  85. return statusDesc;

  86. }

  87.  
  88. public void setStatusDesc(String statusDesc) {

  89. this.statusDesc = statusDesc;

  90. }

  91.  
  92. public String getPaymentTime() {

  93. return paymentTime;

  94. }

  95.  
  96. public void setPaymentTime(String paymentTime) {

  97. this.paymentTime = paymentTime;

  98. }

  99.  
  100. public String getSendTime() {

  101. return sendTime;

  102. }

  103.  
  104. public void setSendTime(String sendTime) {

  105. this.sendTime = sendTime;

  106. }

  107.  
  108. public String getEndTime() {

  109. return endTime;

  110. }

  111.  
  112. public void setEndTime(String endTime) {

  113. this.endTime = endTime;

  114. }

  115.  
  116. public String getCloseTime() {

  117. return closeTime;

  118. }

  119.  
  120. public void setCloseTime(String closeTime) {

  121. this.closeTime = closeTime;

  122. }

  123.  
  124. public String getCreateTime() {

  125. return createTime;

  126. }

  127.  
  128. public void setCreateTime(String createTime) {

  129. this.createTime = createTime;

  130. }

  131.  
  132. public List<OrderItemVo> getOrderItemVoList() {

  133. return orderItemVoList;

  134. }

  135.  
  136. public void setOrderItemVoList(List<OrderItemVo> orderItemVoList) {

  137. this.orderItemVoList = orderItemVoList;

  138. }

  139.  
  140. public String getImageHost() {

  141. return imageHost;

  142. }

  143.  
  144. public void setImageHost(String imageHost) {

  145. this.imageHost = imageHost;

  146. }

  147.  
  148. public Integer getShippingId() {

  149. return shippingId;

  150. }

  151.  
  152. public void setShippingId(Integer shippingId) {

  153. this.shippingId = shippingId;

  154. }

  155.  
  156. public String getReceiverName() {

  157. return receiverName;

  158. }

  159.  
  160. public void setReceiverName(String receiverName) {

  161. this.receiverName = receiverName;

  162. }

  163.  
  164. public ShippingVo getShippingVo() {

  165. return shippingVo;

  166. }

  167.  
  168. public void setShippingVo(ShippingVo shippingVo) {

  169. this.shippingVo = shippingVo;

  170. }

  171. }

    这里存储了和订单状态有关的所有字段信息。

(4)、处理OrderVo

 
  1. private OrderVo assembleOrderVo(Order order,List<OrderItem> orderItemList){

  2. OrderVo orderVo = new OrderVo();

  3. orderVo.setOrderNo(order.getOrderNo());

  4. orderVo.setPayment(order.getPayment());

  5. orderVo.setPaymentType(order.getPaymentType());

  6. orderVo.setPaymentTypeDesc(Const.PaymentTypeEnum.codeOf(order.getPaymentType()).getValue());

  7.  
  8. orderVo.setPostage(order.getPostage());

  9. orderVo.setStatus(order.getStatus());

  10. orderVo.setStatusDesc(Const.OrderStatusEnum.codeOf(order.getStatus()).getValue());

  11.  
  12. orderVo.setShippingId(order.getShippingId());

  13. Shipping shipping = shippingMapper.selectByPrimaryKey(order.getShippingId());

  14. if(shipping != null){

  15. orderVo.setReceiverName(shipping.getReceiverName());

  16. orderVo.setShippingVo(assembleShippingVo(shipping));

  17. }

  18.  
  19. orderVo.setPaymentTime(DateTimeUtil.dateToStr(order.getPaymentTime()));

  20. orderVo.setSendTime(DateTimeUtil.dateToStr(order.getSendTime()));

  21. orderVo.setEndTime(DateTimeUtil.dateToStr(order.getEndTime()));

  22. orderVo.setCreateTime(DateTimeUtil.dateToStr(order.getCreateTime()));

  23. orderVo.setCloseTime(DateTimeUtil.dateToStr(order.getCloseTime()));

  24.  
  25.  
  26. orderVo.setImageHost(PropertiesUtil.getProperty("ftp.server.http.prefix"));

  27.  
  28.  
  29. List<OrderItemVo> orderItemVoList = Lists.newArrayList();

  30.  
  31. for(OrderItem orderItem : orderItemList){

  32. OrderItemVo orderItemVo = assembleOrderItemVo(orderItem);

  33. orderItemVoList.add(orderItemVo);

  34. }

  35. orderVo.setOrderItemVoList(orderItemVoList);

  36. return orderVo;

  37. }

    这里面会涉及到几个工具类,在后面会进行补充说明。

(5)、处理OrderItemVo

 
  1. private OrderItemVo assembleOrderItemVo(OrderItem orderItem){

  2. OrderItemVo orderItemVo = new OrderItemVo();

  3. orderItemVo.setOrderNo(orderItem.getOrderNo());

  4. orderItemVo.setProductId(orderItem.getProductId());

  5. orderItemVo.setProductName(orderItem.getProductName());

  6. orderItemVo.setProductImage(orderItem.getProductImage());

  7. orderItemVo.setCurrentUnitPrice(orderItem.getCurrentUnitPrice());

  8. orderItemVo.setQuantity(orderItem.getQuantity());

  9. orderItemVo.setTotalPrice(orderItem.getTotalPrice());

  10.  
  11. orderItemVo.setCreateTime(DateTimeUtil.dateToStr(orderItem.getCreateTime()));

  12. return orderItemVo;

  13. }

这里的处理OrderProductVo是分开处理的,按需分配,需要使用其中的哪一个字段或方法再来处理!

值得一提的是,这些Vo并不是随便定义的,包括在之前的其他模块中的Vo,都是根据实际开发需求来定义的,不用随便定义Vo。

三、订单管理模块-前台-生成订单功能的开发

 
  1. public ServerResponse createOrder(Integer userId,Integer shippingId){

  2.  
  3. //从购物车中获取数据

  4. List<Cart> cartList = cartMapper.selectCheckedCartByUserId(userId);

  5.  
  6. //计算这个订单的总价

  7. ServerResponse serverResponse = this.getCartOrderItem(userId,cartList);

  8. if(!serverResponse.isSuccess()){

  9. return serverResponse;

  10. }

  11. List<OrderItem> orderItemList = (List<OrderItem>)serverResponse.getData();

  12. BigDecimal payment = this.getOrderTotalPrice(orderItemList);

  13.  
  14.  
  15. //生成订单

  16. Order order = this.assembleOrder(userId,shippingId,payment);

  17. if(order == null){

  18. return ServerResponse.createByErrorMessage("生成订单错误");

  19. }

  20. if(CollectionUtils.isEmpty(orderItemList)){

  21. return ServerResponse.createByErrorMessage("购物车为空");

  22. }

  23. for(OrderItem orderItem : orderItemList){

  24. orderItem.setOrderNo(order.getOrderNo());

  25. }

  26. //mybatis 批量插入

  27. orderItemMapper.batchInsert(orderItemList);

  28.  
  29. //生成成功,我们要减少产品的库存

  30. this.reduceProductStock(orderItemList);

  31. //清空一下购物车

  32. this.cleanCart(cartList);

  33.  
  34. //返回给前端数据

  35.  
  36. OrderVo orderVo = assembleOrderVo(order,orderItemList);

  37. return ServerResponse.createBySuccess(orderVo);

  38. }

    生成订单的核心就是从购物车中遍历存在的商品(信息),然后根据商品(信息)生成一串数字并对应该商品(信息)记录,此时就完成了一个商品订单的生成。当用户在前台点击该商品订单的时候,用户会看到该订单商品的详细信息,这就是数字为什么要绑定商品信息的原因。如果订单只有一串数字(订单号)而不包含其他商品信息,这样是不符合要求的。

    该方法通过传递过来的userId来进行用户区分,从购物车中获取该userId(用户)下的所有的商品信息并存放到List集合中,通过getCartOrderItem方法来生成个当前用户的购物车中商品的总价。如果到这里请求都是成功的,接下来就通过shippingId来生成订单,此时生成的订单中就包含当前用户下单的商品信息和下单的商品总价。如果返回的order为null,表示生成定订单错误;如果orderItemList为空,说明在用户购物车中没有商品,否则订单生成成功。然后把生成的订单信息批量插入到数据库中,同时减少相应的商品库存和清空在用户购物车中的对应的商品记录信息,最后将数据返回给前端即可。

 
  1. private ServerResponse getCartOrderItem(Integer userId,List<Cart> cartList){

  2. List<OrderItem> orderItemList = Lists.newArrayList();

  3. if(CollectionUtils.isEmpty(cartList)){

  4. return ServerResponse.createByErrorMessage("购物车为空");

  5. }

  6.  
  7. //校验购物车的数据,包括产品的状态和数量

  8. for(Cart cartItem : cartList){

  9. OrderItem orderItem = new OrderItem();

  10. Product product = productMapper.selectByPrimaryKey(cartItem.getProductId());

  11. if(Const.ProductStatusEnum.ON_SALE.getCode() != product.getStatus()){

  12. return ServerResponse.createByErrorMessage("产品"+product.getName()+"不是在线售卖状态");

  13. }

  14.  
  15. //校验库存

  16. if(cartItem.getQuantity() > product.getStock()){

  17. return ServerResponse.createByErrorMessage("产品"+product.getName()+"库存不足");

  18. }

  19.  
  20. orderItem.setUserId(userId);

  21. orderItem.setProductId(product.getId());

  22. orderItem.setProductName(product.getName());

  23. orderItem.setProductImage(product.getMainImage());

  24. orderItem.setCurrentUnitPrice(product.getPrice());

  25. orderItem.setQuantity(cartItem.getQuantity());

  26. orderItem.setTotalPrice(BigDecimalUtil.mul(product.getPrice().doubleValue(),cartItem.getQuantity()));

  27. orderItemList.add(orderItem);

  28. }

  29. return ServerResponse.createBySuccess(orderItemList);

  30. }

    getCartOrderItem方法需要传递一个userId和Cart类型的List集合。该方法的主要作用是判断用户购物车是否为空

 
  1. private void reduceProductStock(List<OrderItem> orderItemList){

  2. for(OrderItem orderItem : orderItemList){

  3. Product product = productMapper.selectByPrimaryKey(orderItem.getProductId());

  4. product.setStock(product.getStock()-orderItem.getQuantity());

  5. productMapper.updateByPrimaryKeySelective(product);

  6. }

  7. }

    减少商品库存

 
  1. private void cleanCart(List<Cart> cartList){

  2. for(Cart cart : cartList){

  3. cartMapper.deleteByPrimaryKey(cart.getId());

  4. }

  5. }

    清空用户购物车

Controller层

 
  1. @RequestMapping("create.do")

  2. @ResponseBody

  3. public ServerResponse create(HttpSession session, Integer shippingId){

  4. User user = (User)session.getAttribute(Const.CURRENT_USER);

  5. if(user ==null){

  6. return ServerResponse.createByErrorCodeMessage(ResponseCode.NEED_LOGIN.getCode(),ResponseCode.NEED_LOGIN.getDesc());

  7. }

  8. return iOrderService.createOrder(user.getId(),shippingId);

  9. }

    在生成订单之前需要对用户是否登录做一个校验,如果用户未登录则需要强制登录,否则返回在Service层中的处理方法即可!

四、订单管理模块-前台-取消订单功能的开发

 
  1. public ServerResponse<String> cancel(Integer userId,Long orderNo){

  2. Order order = orderMapper.selectByUserIdAndOrderNo(userId,orderNo);

  3. if(order == null){

  4. return ServerResponse.createByErrorMessage("该用户此订单不存在");

  5. }

  6. if(order.getStatus() != Const.OrderStatusEnum.NO_PAY.getCode()){

  7. return ServerResponse.createByErrorMessage("已付款,无法取消订单");

  8. }

  9. Order updateOrder = new Order();

  10. updateOrder.setId(order.getId());

  11. updateOrder.setStatus(Const.OrderStatusEnum.CANCELED.getCode());

  12.  
  13. int row = orderMapper.updateByPrimaryKeySelective(updateOrder);

  14. if(row > 0){

  15. return ServerResponse.createBySuccess();

  16. }

  17. return ServerResponse.createByError();

  18. }

    取消订单的返回结果是String类型的,所以ServerResponse的泛型为String。通过传递过来的userId和对应的订单号来进行订单的查询。在查询出来以后还要进行是否已付款的校验,如果订单已经付款,就不能取消订单,否则就进行取消订单的操作。

 
  1. @RequestMapping("cancel.do")

  2. @ResponseBody

  3. public ServerResponse cancel(HttpSession session, Long orderNo){

  4. User user = (User)session.getAttribute(Const.CURRENT_USER);

  5. if(user ==null){

  6. return ServerResponse.createByErrorCodeMessage(ResponseCode.NEED_LOGIN.getCode(),ResponseCode.NEED_LOGIN.getDesc());

  7. }

  8. return iOrderService.cancel(user.getId(),orderNo);

  9. }

    处理方法同上,不再赘述!

五、订单管理模块-前台-获取购物车中订单的数量功能的开发

 
  1. public ServerResponse getOrderCartProduct(Integer userId){

  2. OrderProductVo orderProductVo = new OrderProductVo();

  3. //从购物车中获取数据

  4.  
  5. List<Cart> cartList = cartMapper.selectCheckedCartByUserId(userId);

  6. ServerResponse serverResponse = this.getCartOrderItem(userId,cartList);

  7. if(!serverResponse.isSuccess()){

  8. return serverResponse;

  9. }

  10. List<OrderItem> orderItemList =( List<OrderItem> ) serverResponse.getData();

  11.  
  12. List<OrderItemVo> orderItemVoList = Lists.newArrayList();

  13.  
  14. BigDecimal payment = new BigDecimal("0");

  15. for(OrderItem orderItem : orderItemList){

  16. payment = BigDecimalUtil.add(payment.doubleValue(),orderItem.getTotalPrice().doubleValue());

  17. orderItemVoList.add(assembleOrderItemVo(orderItem));

  18. }

  19. orderProductVo.setProductTotalPrice(payment);

  20. orderProductVo.setOrderItemVoList(orderItemVoList);

  21. orderProductVo.setImageHost(PropertiesUtil.getProperty("ftp.server.http.prefix"));

  22. return ServerResponse.createBySuccess(orderProductVo);

  23. }

Controller层

 
  1. @RequestMapping("get_order_cart_product.do")

  2. @ResponseBody

  3. public ServerResponse getOrderCartProduct(HttpSession session){

  4. User user = (User)session.getAttribute(Const.CURRENT_USER);

  5. if(user ==null){

  6. return ServerResponse.createByErrorCodeMessage(ResponseCode.NEED_LOGIN.getCode(),ResponseCode.NEED_LOGIN.getDesc());

  7. }

  8. return iOrderService.getOrderCartProduct(user.getId());

  9. }

    处理方法同上,不再赘述!

六、订单管理模块-前台-获取订单列表功能的开发

 
  1. public ServerResponse<PageInfo> getOrderList(Integer userId,int pageNum,int pageSize){

  2. PageHelper.startPage(pageNum,pageSize);

  3. List<Order> orderList = orderMapper.selectByUserId(userId);

  4. List<OrderVo> orderVoList = assembleOrderVoList(orderList,userId);

  5. PageInfo pageResult = new PageInfo(orderList);

  6. pageResult.setList(orderVoList);

  7. return ServerResponse.createBySuccess(pageResult);

  8. }

    获取订单列表的操作和之前获取列表的操作是一样的,不再赘述!

 
  1. @RequestMapping("list.do")

  2. @ResponseBody

  3. public ServerResponse list(HttpSession session, @RequestParam(value = "pageNum",defaultValue = "1") int pageNum, @RequestParam(value = "pageSize",defaultValue = "10") int pageSize){

  4. User user = (User)session.getAttribute(Const.CURRENT_USER);

  5. if(user ==null){

  6. return ServerResponse.createByErrorCodeMessage(ResponseCode.NEED_LOGIN.getCode(),ResponseCode.NEED_LOGIN.getDesc());

  7. }

  8. return iOrderService.getOrderList(user.getId(),pageNum,pageSize);

  9. }

    处理方法同上,不再赘述!

七、时间和字符串类型之间的转换工具类的补充(joda-time)

 
  1. public class DateTimeUtil {

  2.  
  3. //joda-time

  4.  
  5. //str->Date

  6. //Date->str

  7. public static final String STANDARD_FORMAT = "yyyy-MM-dd HH:mm:ss";

  8.  
  9. public static Date strToDate(String dateTimeStr,String formatStr){

  10. DateTimeFormatter dateTimeFormatter = DateTimeFormat.forPattern(formatStr);

  11. DateTime dateTime = dateTimeFormatter.parseDateTime(dateTimeStr);

  12. return dateTime.toDate();

  13. }

  14.  
  15. public static String dateToStr(Date date,String formatStr){

  16. if(date == null){

  17. return StringUtils.EMPTY;

  18. }

  19. DateTime dateTime = new DateTime(date);

  20. return dateTime.toString(formatStr);

  21. }

  22.  
  23. public static Date strToDate(String dateTimeStr){

  24. DateTimeFormatter dateTimeFormatter = DateTimeFormat.forPattern(STANDARD_FORMAT);

  25. DateTime dateTime = dateTimeFormatter.parseDateTime(dateTimeStr);

  26. return dateTime.toDate();

  27. }

  28.  
  29. public static String dateToStr(Date date){

  30. if(date == null){

  31. return StringUtils.EMPTY;

  32. }

  33. DateTime dateTime = new DateTime(date);

  34. return dateTime.toString(STANDARD_FORMAT);

  35. }

  36. }

由于时间关系,剩下的功能我们下期再见!

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/qq_36314960/article/details/79395708

猜你喜欢

转载自blog.csdn.net/varyall/article/details/81460937