• 基于Springboot+Vue实现前后端分离商城管理系统


     作者主页:编程指南针

    作者简介:Java领域优质创作者、CSDN博客专家 、掘金特邀作者、多年架构师设计经验、腾讯课堂常驻讲师

    主要内容:Java项目、毕业设计、简历模板、学习资料、面试题库、技术互助

    文末获取源码 

    一,项目简介

     新新商城,一款基于 Springboot+Vue 的电商项目,前后端分离项目。完整的实现了一个商城系统应有的基本功能,包括但不限于以下主要功能模块:

    前端商城用户

    1. 用户注册登陆
    2. 商品信息分类和品牌浏览
    3. 全文搜索
    4. 添加购物车管理
    5. 在线购买商品:使用支付宝沙箱在线支付
    6. 个人信息管理
    7. 个人订单管理
    8. 在线退换货功能
    9. 退款功能

    后台用户管理功能

    1. 商品分类管理
    2. 商品品牌管理
    3. 商品规格管理
    4. 商品采购管理
    5. 供应商管理
    6. 订单管理
    7. 退货退款管理
    8. 轮播图设置管理
    9. 用户管理
    10. 权限角色管理
    11. 个人信息管理

          项目后台基于Springboot+MybatisPlus开发实现,前端使用VUE+Element开发实现,前后端分离开发,前端通过调用后台接口来进行相应的交互处理。

          亮点技术:短信发送验证码、阿里云OSS云存储商品图片、邮箱自动发邮件验证操作权限,Shiro权限管理,数据加密处理,支付宝沙箱技术应用,Redis数据缓存处理。

        项目功能完整,界面优雅大方,人机交互流畅,是一个难得的毕业设计作品。

    二,环境介绍

    语言环境:Java:  jdk1.8

    数据库:Mysql: mysql5.7  Redis:5.0.10

    应用服务器:Tomcat:  tomcat8.5.31

    开发工具:IDEA或eclipse

    技术应用:

    后端技术

    技术

    说明

    官网

    SpringBoot

    容器+MVC框架

    https://spring.io/projects/spring-boot

    Shiro

    认证和授权框架

    Apache Shiro Simple. Java. Security.

    MyBatis

    ORM框架

    http://www.mybatis.org/mybatis-3/zh/index.html

    MySQL

    数据库

    https://www.mysql.com/

    Redis

    分布式缓存

    https://redis.io/

    Druid

    数据库连接池

    https://github.com/alibaba/druid

    前端技术

    技术

    说明

    官网

    Vue

    前端框架

    https://vuejs.org/

    Vue-router

    路由框架

    https://router.vuejs.org/

    Vuex

    全局状态管理框架

    https://vuex.vuejs.org/

    Element

    前端UI框架

    https://element.eleme.io

    Axios

    前端HTTP框架

    https://github.com/axios/axios

    vue-clipboard2

    将内容复制到剪贴板

    https://github.com/Inndy/vue-clipboard2

    vuex-persistedstate

    vuex持久化

    https://www.npmjs.com/package/vuex-persistedstate

    nprogress

    进度条控件

    https://github.com/rstacruz/nprogress

    开发环境

    第三方技术

    工具

    官网

    支付宝沙箱技术

    沙箱环境 | 开放平台

    OSS 存储

    阿里云-为了无法计算的价值

    阿里云短信服务

    阿里云-为了无法计算的价值

    QQ邮箱服务

    登录QQ邮箱

    三,系统展示

    下面展示一下项目的主要核心功能:

    前端用户操作

    用户注册:会对邮箱发验证码验证,要求必须真实有效邮箱

    用户登陆

    首页

    商品购买

    购物车

    结算付款

    我的订单:可以申请退款、收货后可以申请退货等操作

    后台管理用户功能

    后台首页

    商品管理:可以实现商品添加、编辑、删除、下架、查询等,另可以对商品分类、品牌、规则、采购信息进行管理,以及对供应商进行管理,不再一一展示。菜单上的功能都是齐全的。

    订单管理:可以进行发货、退货、退款等相关操作

    营销管理

    主要对前端展示的广告轮播图进行管理

    用户和权限管理:可以对管理员、顾客、角色进行管理操作

    四,核心代码展示

    1. package com.qiu.controller;
    2. import com.alibaba.fastjson.JSON;
    3. import com.qiu.entity.Logistics;
    4. import com.qiu.entity.Order;
    5. import com.qiu.entity.Product;
    6. import com.qiu.service.LogisticsService;
    7. import com.qiu.service.OrderService;
    8. import com.qiu.service.ProductService;
    9. import com.qiu.util.general.CommonResult;
    10. import org.apache.commons.lang3.StringUtils;
    11. import org.springframework.beans.factory.annotation.Autowired;
    12. import org.springframework.data.redis.core.RedisTemplate;
    13. import org.springframework.web.bind.annotation.CrossOrigin;
    14. import org.springframework.web.bind.annotation.RequestMapping;
    15. import org.springframework.web.bind.annotation.RestController;
    16. import java.util.ArrayList;
    17. import java.util.List;
    18. import java.util.Map;
    19. import java.util.concurrent.TimeUnit;
    20. /**
    21. * @author ZNZ
    22. * @email 469603589@qq.com
    23. * @date 2022/12/28 18:11
    24. * @description 订单相关业务
    25. */
    26. @RestController
    27. @CrossOrigin
    28. public class OrderController {
    29. private static final String VIP = "Vip";
    30. private static final String COLLECT_GOODS_STATE = "已收货";
    31. @Autowired
    32. private OrderService orderService;
    33. @Autowired
    34. private ProductService productService;
    35. @Autowired
    36. private LogisticsService logisticsService;
    37. @Autowired
    38. private RedisTemplate<String, String> redisTemplate;
    39. @RequestMapping(value = "/order/findById")
    40. public CommonResult findOrderById(Integer orderId) {
    41. Order order = orderService.selectById(orderId);
    42. if (orderId != null) {
    43. return CommonResult.success("订单信息查询成功", order);
    44. }
    45. return CommonResult.error("订单信息查询失败");
    46. }
    47. @RequestMapping(value = "/order/findOrderInfo")
    48. public CommonResult findOrderInfo(String userAccount) {
    49. List<Map<String, Object>> orderMap = orderService.selectAllOrder(userAccount);
    50. if (orderMap != null) {
    51. return CommonResult.success("订单信息查询成功", orderMap);
    52. }
    53. return CommonResult.error("订单信息查询失败");
    54. }
    55. @RequestMapping(value = "/order/findAll")
    56. public CommonResult findAllOrder() {
    57. List<Order> orders = orderService.selectAll();
    58. if (orders != null) {
    59. return CommonResult.success("订单信息查询成功", orders);
    60. }
    61. return CommonResult.error("订单信息查询失败");
    62. }
    63. @RequestMapping(value = "/order/findCount")
    64. public CommonResult findCount() {
    65. Integer count = orderService.selectCount();
    66. if (count != null) {
    67. return CommonResult.success("订单数量查询成功", count);
    68. }
    69. return CommonResult.error("订单数量查询失败");
    70. }
    71. @RequestMapping(value = "/order/add")
    72. public CommonResult addOrder(Order order) {
    73. if (order != null) {
    74. if (order.getProductNo().contains(VIP)) {
    75. return handleMemberOrders(order);
    76. }
    77. return handleMerchandiseOrders(order);
    78. } else {
    79. return CommonResult.error("订单数据不完整");
    80. }
    81. }
    82. @RequestMapping(value = "/order/cartOrder")
    83. public CommonResult cartOrder(String orderNo, String ordersInfo, String cartIds) {
    84. List<String> cartIdList = JSON.parseArray(cartIds, String.class);
    85. List<Order> orders = JSON.parseArray(ordersInfo, Order.class);
    86. if (orders != null) {
    87. ArrayList<String> orderInfo = new ArrayList<>();
    88. ArrayList<String> productInfo = new ArrayList<>();
    89. for (Order order : orders) {
    90. Product product = productService.selectByKey(order.getProductNo());
    91. Integer productStock = product.getProductStock();
    92. Integer payAmount = order.getPayAmount();
    93. if (productStock >= payAmount) {
    94. Product newProduct = new Product();
    95. newProduct.setProductId(product.getProductId());
    96. int newStock = productStock - payAmount;
    97. newProduct.setProductStock(newStock);
    98. newProduct.setIsStockOut(newStock < product.getLowestStock());
    99. // 如果库存小于等于0,自动下架
    100. newProduct.setIsSale(newStock > 0);
    101. if (productService.updateById(newProduct) && orderService.insertData(order)) {
    102. orderInfo.add(order.getOrderNo());
    103. productInfo.add(order.getProductNo());
    104. }
    105. }
    106. }
    107. if (!orderInfo.isEmpty()) {
    108. String cartIdsInfo = StringUtils.join(cartIdList.toArray(), ",");
    109. String orderNoInfo = StringUtils.join(orderInfo, ",");
    110. String productNoInfo = StringUtils.join(productInfo, ",");
    111. redisTemplate.opsForValue().set(orderNo, orderNoInfo, 24, TimeUnit.HOURS);
    112. redisTemplate.opsForValue().set("cartId" + orderNo, cartIdsInfo, 24, TimeUnit.HOURS);
    113. return CommonResult.success("创建订单成功", productNoInfo);
    114. }
    115. return CommonResult.error("创建订单失败,请查看商品库存是否满足购买数量");
    116. } else {
    117. return CommonResult.error("订单数据不完整");
    118. }
    119. }
    120. @RequestMapping(value = "/order/update")
    121. public CommonResult updateOrder(Order order) {
    122. if (orderService.updateById(order)) {
    123. return CommonResult.success("修改订单成功", order);
    124. }
    125. return CommonResult.error("修改订单失败");
    126. }
    127. @RequestMapping(value = "/order/delete")
    128. public CommonResult deleteOrder(Integer orderId) {
    129. if (orderService.deleteById(orderId)) {
    130. return CommonResult.success("删除订单成功", "订单id:" + orderId);
    131. }
    132. return CommonResult.error("删除订单失败");
    133. }
    134. @RequestMapping(value = "/order/receipt")
    135. public CommonResult updateOrder(Integer orderId) {
    136. Order order = new Order();
    137. order.setOrderId(orderId);
    138. order.setOrderState(COLLECT_GOODS_STATE);
    139. if (orderService.updateById(order)) {
    140. return CommonResult.success("商品收货成功", order);
    141. }
    142. return CommonResult.error("商品收货失败");
    143. }
    144. @RequestMapping(value = "/orderDetail/orderInfo")
    145. public CommonResult orderInfo(String orderNo) {
    146. ArrayList<Object> resultList = new ArrayList<>();
    147. Order order = orderService.selectByKey(orderNo);
    148. Logistics logistics = logisticsService.selectOrderNo(orderNo);
    149. if (order != null) {
    150. resultList.add(order);
    151. }
    152. if (logistics != null) {
    153. resultList.add(logistics);
    154. }
    155. return CommonResult.success("订单详情查询成功", resultList);
    156. }
    157. /**
    158. * 处理会员订单
    159. *
    160. * @param order 订单信息
    161. */
    162. private CommonResult handleMemberOrders(Order order) {
    163. if (orderService.insertData(order)) {
    164. return CommonResult.success("创建订单成功", order);
    165. } else {
    166. return CommonResult.error("创建订单失败");
    167. }
    168. }
    169. /**
    170. * 处理商品订单
    171. *
    172. * @param order 订单信息
    173. */
    174. private CommonResult handleMerchandiseOrders(Order order) {
    175. Product product = productService.selectByKey(order.getProductNo());
    176. Integer productStock = product.getProductStock();
    177. Integer payAmount = order.getPayAmount();
    178. boolean isOk = productStock >= payAmount;
    179. if (isOk) {
    180. Product newProduct = new Product();
    181. newProduct.setProductId(product.getProductId());
    182. int newStock = productStock - payAmount;
    183. newProduct.setProductStock(newStock);
    184. newProduct.setIsStockOut(newStock < product.getLowestStock());
    185. // 如果库存小于等于0,自动下架
    186. newProduct.setIsSale(newStock > 0);
    187. if (productService.updateById(newProduct)) {
    188. if (orderService.insertData(order)) {
    189. redisTemplate.opsForValue().set(order.getOrderNo(), order.getOrderNo(), 24, TimeUnit.HOURS);
    190. return CommonResult.success("创建订单成功", order);
    191. } else {
    192. return CommonResult.error("创建订单失败");
    193. }
    194. } else {
    195. return CommonResult.error("创建订单失败");
    196. }
    197. } else {
    198. return CommonResult.error("商品库存不足");
    199. }
    200. }
    201. }
    1. package com.qiu.controller;
    2. import com.qiu.util.general.CommonResult;
    3. import com.qiu.util.oss.AliyunOssUtil;
    4. import lombok.extern.slf4j.Slf4j;
    5. import org.apache.commons.io.FileUtils;
    6. import org.apache.commons.lang3.StringUtils;
    7. import org.springframework.beans.factory.annotation.Autowired;
    8. import org.springframework.web.bind.annotation.CrossOrigin;
    9. import org.springframework.web.bind.annotation.RequestMapping;
    10. import org.springframework.web.bind.annotation.RequestParam;
    11. import org.springframework.web.bind.annotation.RestController;
    12. import org.springframework.web.multipart.MultipartFile;
    13. import java.io.File;
    14. import java.io.FileOutputStream;
    15. import java.io.IOException;
    16. /**
    17. * @author ZNZ
    18. */
    19. @Slf4j
    20. @CrossOrigin
    21. @RestController
    22. public class OssController {
    23. @Autowired
    24. private AliyunOssUtil ossUtil;
    25. @RequestMapping("/uploadImage")
    26. public CommonResult upload(@RequestParam("name") String folderName,
    27. @RequestParam("file") MultipartFile file) throws IOException {
    28. if (file != null) {
    29. String fileName = file.getOriginalFilename();
    30. if (StringUtils.isNotBlank(fileName)) {
    31. File newFile = new File(fileName);
    32. try (FileOutputStream os = new FileOutputStream(newFile)) {
    33. os.write(file.getBytes());
    34. file.transferTo(newFile);
    35. String path = ossUtil.upload(folderName, newFile);
    36. log.info("文件上传成功,路径:{}", path);
    37. return new CommonResult(200, "上传成功", path);
    38. } catch (Exception e) {
    39. log.error("文件上传失败", e);
    40. return CommonResult.error("上传失败");
    41. } finally {
    42. if (newFile.exists()) {
    43. FileUtils.forceDelete(newFile);
    44. }
    45. }
    46. }
    47. }
    48. return CommonResult.error("文件不存在");
    49. }
    50. }

    1. package com.qiu.controller;
    2. import com.qiu.entity.ShoppingCart;
    3. import com.qiu.service.ShoppingCartService;
    4. import com.qiu.util.general.CommonResult;
    5. import org.springframework.beans.factory.annotation.Autowired;
    6. import org.springframework.web.bind.annotation.CrossOrigin;
    7. import org.springframework.web.bind.annotation.RequestMapping;
    8. import org.springframework.web.bind.annotation.RestController;
    9. import java.util.List;
    10. import java.util.Map;
    11. /**
    12. * @author ZNZ
    13. * @email 469603589@qq.com
    14. * @date 2022/12/31 16:23
    15. * @description 购物车业务类
    16. */
    17. @RestController
    18. @CrossOrigin
    19. public class ShoppingCartController {
    20. @Autowired
    21. private ShoppingCartService shoppingCartService;
    22. /**
    23. * 新增商品到购物车
    24. *
    25. * @param shoppingCart 购物车商品信息
    26. */
    27. @RequestMapping(value = "/shoppingCart/add")
    28. public CommonResult addShoppingCart(ShoppingCart shoppingCart) {
    29. if (shoppingCartService.insertData(shoppingCart)) {
    30. return CommonResult.success("购物车添加成功", shoppingCart);
    31. }
    32. return CommonResult.error("购物车添加失败");
    33. }
    34. /**
    35. * 更新购物车商品
    36. *
    37. * @param shoppingCart 购物车商品信息
    38. */
    39. @RequestMapping(value = "/shoppingCart/update")
    40. public CommonResult updateShoppingCart(ShoppingCart shoppingCart) {
    41. if (shoppingCartService.updateById(shoppingCart)) {
    42. return CommonResult.success("购物车修改成功", shoppingCart);
    43. }
    44. return CommonResult.error("购物车修改失败");
    45. }
    46. /**
    47. * 购物车移除商品
    48. *
    49. * @param cartId 购物车商品编号
    50. */
    51. @RequestMapping(value = "/shoppingCart/deleteById")
    52. public CommonResult deleteShoppingCart(Integer cartId) {
    53. if (shoppingCartService.deleteById(cartId)) {
    54. return CommonResult.success("购物车删除成功", "cartId: " + cartId);
    55. }
    56. return CommonResult.error("购物车删除失败");
    57. }
    58. /**
    59. * 根据用户移除购物车
    60. *
    61. * @param account 用户账户
    62. */
    63. @RequestMapping(value = "/shoppingCart/deleteByUser")
    64. public CommonResult deleteByUser(String account) {
    65. if (shoppingCartService.deleteByUser(account)) {
    66. return CommonResult.success("购物车删除成功", account);
    67. }
    68. return CommonResult.error("购物车删除失败");
    69. }
    70. /**
    71. * 查询用户账号下的购物车信息
    72. *
    73. * @param account 用户账户
    74. */
    75. @RequestMapping(value = "/shoppingCart/findAll")
    76. public CommonResult findAllShoppingCart(String account) {
    77. List<Map<String, Object>> shoppingInfo = shoppingCartService.selectAll(account);
    78. if (shoppingInfo != null) {
    79. return CommonResult.success("购物车查询成功", shoppingInfo);
    80. }
    81. return CommonResult.error("购物车查询失败");
    82. }
    83. /**
    84. * 根据购物车商品编号查询购物车商品信息
    85. *
    86. * @param cartId 购物车商品编号
    87. */
    88. @RequestMapping(value = "/shoppingCart/findById")
    89. public CommonResult findById(Integer cartId) {
    90. ShoppingCart shoppingCart = shoppingCartService.selectById(cartId);
    91. if (shoppingCart != null) {
    92. return CommonResult.success("购物车查询成功", shoppingCart);
    93. }
    94. return CommonResult.error("购物车查询失败");
    95. }
    96. }

    五,项目总结

     项目后台基于Springboot+MybatisPlus开发实现,前端使用VUE+Element开发实现,前后端分离开发,前端通过调用后台接口来进行相应的交互处理。

          亮点技术:短信发送验证码、阿里云OSS云存储商品图片、邮箱自动发邮件验证操作权限,Shiro权限管理,数据加密处理,支付宝沙箱技术应用,Redis数据缓存处理。

        项目功能完整,界面优雅大方,人机交互流畅,是一个难得的毕业设计作品。

  • 相关阅读:
    基于python的c语言学习笔记(2)
    认识border
    大数据-玩转数据-Flink定时器
    web前端期末大作业:基于HTML+CSS+JavaScript实现网上鲜花店网站设计(14页)
    llava1.5-部署
    AndroidJetpack应用指南学习笔记13--DataBinding的简单使用
    (10)(10.9) 术语表(一)
    SpringSecurity简介及其入门案例
    单精度浮点数和双精度浮点数有什么区别
    PHREEQC建模及典型案例解析与高阶拓展应用【反向“编译”、“玩转”后处理技术、GibbsStudio和PhreePlo方法】
  • 原文地址:https://blog.csdn.net/znzbs/article/details/125513912