地址簿模块-导入代码
需求分析和设计
原型:


查询默认地址接口,是应用在用户下单的界面,默认显示 默认地址
接口设计

新增地址

查询登录用户所有地址

查询默认地址

修改地址

根据id删除地址

根据id查询地址

设置默认地址

数据库设计

具体代码开发:
控制层-AddressBookController
@RestController
@RequestMapping("/user/addressBook")
@Api(tags = "C端地址簿接口")
public class AddressBookController {
@Autowired
private AddressBookService addressBookService;
/**
* 查询当前登录用户的所有地址信息
*
* @return
*/
@GetMapping("/list")
@ApiOperation("查询当前登录用户的所有地址信息")
public Result<List<AddressBook>> list() {
AddressBook addressBook = new AddressBook();
addressBook.setUserId(BaseContext.getCurrentId());
List<AddressBook> list = addressBookService.list(addressBook);
return Result.success(list);
}
/**
* 新增地址
*
* @param addressBook
* @return
*/
@PostMapping
@ApiOperation("新增地址")
public Result save(@RequestBody AddressBook addressBook) {
addressBookService.save(addressBook);
return Result.success();
}
@GetMapping("/{id}")
@ApiOperation("根据id查询地址")
public Result<AddressBook> getById(@PathVariable Long id) {
AddressBook addressBook = addressBookService.getById(id);
return Result.success(addressBook);
}
/**
* 根据id修改地址
*
* @param addressBook
* @return
*/
@PutMapping
@ApiOperation("根据id修改地址")
public Result update(@RequestBody AddressBook addressBook) {
addressBookService.update(addressBook);
return Result.success();
}
/**
* 设置默认地址
*
* @param addressBook
* @return
*/
@PutMapping("/default")
@ApiOperation("设置默认地址")
public Result setDefault(@RequestBody AddressBook addressBook) {
addressBookService.setDefault(addressBook);
return Result.success();
}
/**
* 根据id删除地址
*
* @param id
* @return
*/
@DeleteMapping
@ApiOperation("根据id删除地址")
public Result deleteById(Long id) {
addressBookService.deleteById(id);
return Result.success();
}
/**
* 查询默认地址
*/
@GetMapping("default")
@ApiOperation("查询默认地址")
public Result<AddressBook> getDefault() {
//SQL:select * from address_book where user_id = ? and is_default = 1
AddressBook addressBook = new AddressBook();
addressBook.setIsDefault(1);
addressBook.setUserId(BaseContext.getCurrentId());
List<AddressBook> list = addressBookService.list(addressBook);
if (list != null && list.size() == 1) {
return Result.success(list.get(0));
}
return Result.error("没有查询到默认地址");
}
}
业务层-AddressBookServiceImpl
@Service
@Slf4j
public class AddressBookServiceImpl implements AddressBookService {
@Autowired
private AddressBookMapper addressBookMapper;
/**
* 条件查询
*
* @param addressBook
* @return
*/
public List<AddressBook> list(AddressBook addressBook) {
return addressBookMapper.list(addressBook);
}
/**
* 新增地址
*
* @param addressBook
*/
public void save(AddressBook addressBook) {
addressBook.setUserId(BaseContext.getCurrentId());
addressBook.setIsDefault(0);
addressBookMapper.insert(addressBook);
}
/**
* 根据id查询
*
* @param id
* @return
*/
public AddressBook getById(Long id) {
AddressBook addressBook = addressBookMapper.getById(id);
return addressBook;
}
/**
* 根据id修改地址
*
* @param addressBook
*/
public void update(AddressBook addressBook) {
addressBookMapper.update(addressBook);
}
/**
* 设置默认地址
*
* @param addressBook
*/
@Transactional
public void setDefault(AddressBook addressBook) {
//1、将当前用户的所有地址修改为非默认地址 update address_book set is_default = ? where user_id = ?
addressBook.setIsDefault(0);
addressBook.setUserId(BaseContext.getCurrentId());
addressBookMapper.updateIsDefaultByUserId(addressBook);
//2、将当前地址改为默认地址 update address_book set is_default = ? where id = ?
addressBook.setIsDefault(1);
addressBookMapper.update(addressBook);
}
/**
* 根据id删除地址
*
* @param id
*/
public void deleteById(Long id) {
addressBookMapper.deleteById(id);
}
}
数据持久层-AddressBookMapper
@Mapper
public interface AddressBookMapper {
/**
* 条件查询
* @param addressBook
* @return
*/
List<AddressBook> list(AddressBook addressBook);
/**
* 新增
* @param addressBook
*/
@Insert("insert into address_book" +
" (user_id, consignee, phone, sex, province_code, province_name, city_code, city_name, district_code," +
" district_name, detail, label, is_default)" +
" values (#{userId}, #{consignee}, #{phone}, #{sex}, #{provinceCode}, #{provinceName}, #{cityCode}, #{cityName}," +
" #{districtCode}, #{districtName}, #{detail}, #{label}, #{isDefault})")
void insert(AddressBook addressBook);
/**
* 根据id查询
* @param id
* @return
*/
@Select("select * from address_book where id = #{id}")
AddressBook getById(Long id);
/**
* 根据id修改
* @param addressBook
*/
void update(AddressBook addressBook);
/**
* 根据 用户id修改 是否默认地址
* @param addressBook
*/
@Update("update address_book set is_default = #{isDefault} where user_id = #{userId}")
void updateIsDefaultByUserId(AddressBook addressBook);
/**
* 根据id删除地址
* @param id
*/
@Delete("delete from address_book where id = #{id}")
void deleteById(Long id);
}
<select id="list" parameterType="AddressBook" resultType="AddressBook">
select * from address_book
<where>
<if test="userId != null">
and user_id = #{userId}
</if>
<if test="phone != null">
and phone = #{phone}
</if>
<if test="isDefault != null">
and is_default = #{isDefault}
</if>
</where>
</select>
<update id="update" parameterType="addressBook">
update address_book
<set>
<if test="consignee != null">
consignee = #{consignee},
</if>
<if test="sex != null">
sex = #{sex},
</if>
<if test="phone != null">
phone = #{phone},
</if>
<if test="detail != null">
detail = #{detail},
</if>
<if test="label != null">
label = #{label},
</if>
<if test="isDefault != null">
is_default = #{isDefault},
</if>
</set>
where id = #{id}
</update>
用户下单
需求分析与设计

接口设计

数据库设计


具体代码开发
控制层-OrderController
@RestController
@RequestMapping("/user/order")
@Slf4j
@Api(tags = "C端-用户下单接口")
public class OrderController {
@Autowired
private OrderService orderService;
/**
* 用户下单
*
* @param ordersSubmitDTO
* @return
*/
@PostMapping("/submit")
@ApiOperation("用户下单")
public Result<OrderSubmitVO> submit(@RequestBody OrdersSubmitDTO ordersSubmitDTO){
log.info("用户下单,参数为:{}",ordersSubmitDTO);
OrderSubmitVO orderSubmitVO= orderService.submitOrder(ordersSubmitDTO);
return Result.success(orderSubmitVO);
}
}
业务层-OrderServiceImpl
@Service
public class OrderServiceImpl implements OrderService {
@Autowired
private AddressBookMapper addressBookMapper;
@Autowired
private ShoppingCartMapper shoppingCartMapper;
@Autowired
private OrderMapper orderMapper;
@Autowired
private OrderDetailMapper orderDetailMapper;
public OrderSubmitVO submitOrder(OrdersSubmitDTO ordersSubmitDTO) {
//处理各种业务异常 (地址簿为空,购物车数据为空)
AddressBook addressBook = addressBookMapper.getById(ordersSubmitDTO.getAddressBookId());
if (addressBook == null){ //判断地址簿是否为空
//抛出业务异常
throw new AddressBookBusinessException(MessageConstant.ADDRESS_BOOK_IS_NULL);
}
// 查询当前用户的购物车数据
Long userId = BaseContext.getCurrentId();
ShoppingCart shoppingCart = new ShoppingCart();
shoppingCart.setUserId(userId); //需要userId才可以查购物车
List<ShoppingCart> shoppingCartList = shoppingCartMapper.list(shoppingCart);
if (shoppingCartList == null || shoppingCartList.size() == 0 ){
throw new ShoppingCartBusinessException(MessageConstant.SHOPPING_CART_IS_NULL);
}
//向订单表插入一条数据
Orders orders = new Orders();
BeanUtils.copyProperties(ordersSubmitDTO,orders);
orders.setOrderTime(LocalDateTime.now()); //设置订单创建时间
orders.setPayStatus(Orders.UN_PAID); //设置订单支付状态,这里是0未支付
orders.setStatus(Orders.PENDING_PAYMENT); //设置订单状态 1待付款
orders.setNumber(String.valueOf(System.currentTimeMillis())); //生成订单号,通过系统时间戳来生成
orders.setPhone(addressBook.getPhone()); //通过地址对象拿到手机号,然后封装进去
orders.setConsignee(addressBook.getConsignee()); //通过地址对象拿到订单联系人,然后封装进去
orders.setUserId(userId); //设置userId
orderMapper.insert(orders);
//创建订单明细列表对象,方便批量插入如订单明细
List<OrderDetail> orderDetailList = new ArrayList<>();
//向订单明细表插入n条数据
for (ShoppingCart cart : shoppingCartList) { //遍历购物车商品数据
OrderDetail orderDetail = new OrderDetail();
BeanUtils.copyProperties(cart,orderDetail); //将购物车商品拷贝给订单明细对象cart
orderDetail.setOrderId(orders.getId()); //设置当前订单明细 所关联的订单 id
orderDetailList.add(orderDetail);
}
//批量插入 订单明细 数据
orderDetailMapper.insertBatch(orderDetailList);
// 清空当前用户的购物车数据
shoppingCartMapper.deleteByUserId(userId);
//封装VO返回结果
OrderSubmitVO orderSubmitVO = OrderSubmitVO.builder()
.id(orders.getId())
.orderTime(orders.getOrderTime())
.orderNumber(orders.getNumber())
.orderAmount(orders.getAmount())
.build();
return orderSubmitVO;
}
}
数据持久层-OrderMapper 、OrderMapper .xml、OrderDetailMapper、OrderDetailMapper.xml
@Mapper
public interface OrderMapper {
/**
* 向订单表插入一条数据
* @param orders
* @return
*/
void insert(Orders orders);
}
<insert id="insert" useGeneratedKeys="true" keyProperty="id" >
insert into orders(number,status,user_id,address_book_id,order_time,checkout_time,pay_method,pay_status,
amount,remark,phone,address,consignee,estimated_delivery_time,delivery_status,
pack_amount,tableware_number,tableware_status)
values
(#{number},#{status},#{userId},#{addressBookId},#{orderTime},#{checkoutTime},#{payMethod},
#{payStatus},#{amount},#{remark},#{phone},#{address},#{consignee},#{estimatedDeliveryTime},
#{deliveryStatus},#{packAmount},#{tablewareNumber},#{tablewareStatus})
</insert>
@Mapper
public interface OrderDetailMapper {
/**
* 向订单明细表批量插入数据
* @param orderDetailList
* @return
*/
void insertBatch(List<OrderDetail> orderDetailList);
}
<insert id="insertBatch">
insert into order_detail(name,image,order_id,dish_id,setmeal_id,dish_flavor,number,amount)
values
<foreach collection="orderDetailList" item="od" separator=",">
(#{od.name},#{od.image},#{od.orderId},#{od.dishId},#{od.setmealId},#{od.dishFlavor},#{od.number},#{od.amount})
</foreach>
</insert>
微信支付
介绍

准备工作-开发思路

1、商户调用微信支付后台 预支付 接口

2、微信小程序 调起支付

实现功能时的问题:

1、调用过程如何保证数据安全-使用加密和签名-官网获取文件(如下图)

2、微信后台如何调用到商户系统-内网穿透,给服务端一个随机的公网ip

网址:cpolar - secure introspectable tunnels to localhost
https://dashboard.cpolar.com/login
使用内网穿透工具,获取Authtoken,并执行命令开启服务

在端口8080上启动HTTP隧道

获得公网ip

成功通过公网ip访问

代码
配置文件

导入代码

没有企业资质解决方案:暂时跳过
可以将OrderServiceImpl中的payment方法里的调用微信支付接口注释掉,重新new一个空的JSONObject对象

然后在OrderController里直接调用paySuccess方法

在前端程序中的pay中直接修改代码如下重新编译即可

苍穹外卖Day8地址簿与下单模块
544

被折叠的 条评论
为什么被折叠?



