苍穹外卖 --day09-地址簿--用户下单--订单支

1. 导入地址簿功能代码

1.1 需求分析和设计
1.1.1 产品原型

地址簿,指的是消费者用户的地址信息,用户登录成功后可以维护自己的地址信息。同一个用户可以有多个地址信息,而且可以设置为默认地址,但是默认地址是唯一的

效果图:

对于地址薄管理,想要实现以下功能:

查询地址列表

新增地址

修改地址

删除地址

设置默认地址

查询默认地址

1.1.2 接口设计

根据原型图设计接口,一共包含7个接口

接口设计:

新增地址

查询登录用户所有地址

查询默认地址

根据id修改地址

根据id查询地址

设置默认地址

1). 新增地址

2). 查询登录用户所有地址

3). 查询默认地址

4). 修改地址

5). 根据id删除地址

6). 根据id查询地址

7). 设置默认地址

1.1.3 表设计

地址信息会存到address_book表,即地址簿表中。具体表结构如下:

字段名数据类型说明备注
idbigint主键自增
user_idbigint用户id逻辑外键
consigneevarchar(50)收货人
sexvarchar(2)性别
phonevarchar(11)手机号
province_codevarchar(12)省份编码
province_namevarchar(32)省份名称
city_codevarchar(12)城市编码
city_namevarchar(32)城市名称
district_codevarchar(12)区县编码
district_namevarchar(32)区县名称
detailvarchar(200)详细地址信息具体到门牌号
labelvarchar(100)标签公司,学校,家
is_defaulttinyint(1)是否默认地址1是 0否

这里有一个字段is_default,在设置默认地址的时候,直接更新这个字段就OK了

1:查询当前登录用户的所有地址信息

Controller

 /**
     * 查询当前登录用户的所有地址信息
     *
     * @return
     */

@GetMapper("/list")
@ApiOperation("查询当前登录用户的所有地址信息")
public Result<List<AddressBook>> list(){
    AddressBook addressBook = new AddressBook();
    //获取当前登录的用户
    addressBook.setUserId(BaseContext.getCurrentId());
    addressBookService.list(addressBook);
    return Result.success(list)
}

Service

List<AddressBook> list (AddressBook addressBook)

Impl

public List<AddressBook> list(AddressBook addressBook){
       return addressBookMapper.list(addressBook)
}

Mapper

List<AddressBook> list(AddressBook addressBook);

Mapper.xml

  <select id="list" parameterType="com.sky.entity.AddressBook" resultType="com.sky.entity.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>

2.新增地址

Controller

/**
* 新增地址
*
* @param addressBook
* @return
*/

@PostMapping
@ApiOperation("新增地址")
public Result save(@RequestBody AddressBook addressBook){

           addressBookService.save(addressBook);

           return Result.success();
}

Service

void save(AddressBook addressBook)

Impl

   /**
     * 新增地址
     *
     * @param addressBook
     */

public void save(AddressBook addressBook){
       //获取当前登录用户
       addressBook.setUserId(BaseContext.getCurrentId());
       //设置是否默认0为未默认,1为默认
       addressBook.setIsDefault(0);
       addressBookMapper.insert(addressBook);

}

 Mapper

/**
* 新增
* @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);

3.根据id查询地址

 Controller

@GetMapper("/{id}")
@ApiOperation("根据id查询地址")
public Result <AddressBook> getById(@PathVariable Long id){
AddressBook addressBook  = addressBookService.getById(id);
return Result.success(addressBook);
}

 Service

AddressBook getById(Long id);

Impl

    /**
     * 根据id查询
     *
     * @param id
     * @return
     */
      public AddressBook getById(Long id){
      AddressBook addressBook = addressBookMapper.getById(id);
      return addressBook;
     }

Mapper

    /**
     * 根据id查询
     * @param id
     * @return
     */
    @Select("select * from address_book where id = #{id}")
    AddressBook getById(Long id);

3.根据id修改地址

Controller

    /**
     * 根据id修改地址
     *
     * @param addressBook
     * @return
     */
      @PutMapping
      @ApiOperation("根据id修改地址")
      public Result update(@RequestBody AddressBook addressBook){
      addressBookService.update(addressBook);
      return Result.success();
      }

Service

void update(AddressBook addressBook);

Impl

    /**
     * 根据id修改地址
     *
     * @param addressBook
     */

      public void update(AddressBook addressBook){
      addressBookMapper.update(addressBook);

      }

Mapper

    /**
     * 根据id修改
     * @param addressBook
     */
    void update(AddressBook addressBook);

Mapper.xml

<update id="update" parameterType="com.sky.entity.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>

4.设置默认地址

Controller

    /**
     * 设置默认地址
     *
     * @param addressBook
     * @return
     */
     @PutMapping("/default")
     @ApiOperation("设置默认地址")
     public Result setDefault(@RequestBody AddressBook addressBook){
     addressBookService.setDefault(addressBook);
     return Result.success();
     }

Service

void setDefault(AddressBook addressBook);

Impl

 设置默认地址本质上是更新操作,整体的思路是:先让所有地址都变成非默认,然后在设置一个默认地址。

    /**
     * 设置默认地址
     *
     * @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);
     
      }

Mapper

    
      /**
     * 根据 用户id修改 是否默认地址
     * @param addressBook
     */
    @Update("update address_book set is_default = #{isDefault} where user_id = #{userId}")
    void updateIsDefaultByUserId(AddressBook addressBook);
    /**
     * 根据id修改
     * @param addressBook
     */ 
      void update(AddressBook addressBook)

Impl

    <update id="update" parameterType="com.sky.entity.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>

5.根据id删除地址

Controller

    /**
     * 根据id删除地址
     *
     * @param id
     * @return
     */
      @DeleteMapping
      @ApiOperation("根据id删除地址")
      public Result deleteById(Long id){
      addressBookService.deleteById(id);
      return Result.success();
      }

Service

void deleteById(Lonr id)

Impl

    /**
     * 根据id删除地址
     *
     * @param id
     */
      public void deleteById(Long id){
      addressBookMapper.deleteById(id);
     }

Mapper


    /**
     * 根据id删除地址
     * @param id
     */
     @Delete("delete from address_book where id = #{id}")
     void deleteById(Long id);

6.查询默认地址

    /**
     * 查询默认地址
     */

     @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("没有查询到默认地址");
   }
}

Service

List<AddressBook>list(AddressBook addressBook);

impl

    /**
     * 条件查询
     *
     * @param addressBook
     * @return
     */
     public List<AddressBook> list(AddressBook addressBook){
     return addressBookMapper.list(addressBook);
     }

Mapper

    /**
     * 条件查询
     * @param addressBook
     * @return
     */
     List<AddressBook> list(AddressBook addressBook);

Mapper.xml

    <select id="list" parameterType="com.sky.entity.AddressBook" resultType="com.sky.entity.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>

 2. 用户下单

 2.1 需求分析和设计

 2.1.1 产品原型

用户下单业务说明: 在电商系统中,用户一般是通过下单的方式通知商家,用户已经购买了商品,需要商家进行备货和发货。用户下单后会产生订单相关数据,订单数据需要能够体现如下信息:

用户将菜品或者套餐加入购物车后,可以点击购物车中的  "去结算" 按钮,然后页面跳转到订单确认页面,点击"去支付"按钮规则完成下单操作。

用户点餐业务流程(效果图):

2.1.3表设计

用户下单业务对应的数据表为orders表和order_detail表(一对多关系,一个订单关联多个订单明细):

表名含义说明
orders订单表主要存储订单的基本信息(如:订单号,状态,金额,下单用户,支付方式,下单用户,收货地址等)
order_detail订单明细表主要存储的是订单详情信息

具体的表结构如下:

1). orders订单表

字段名数据类型说明备注
idbigint主键自增
numbervarchar(50)订单号
statusint订单状态1:待付款,2:待接单,3:已接单,4:派送中,5:已完成,,6:已取消
user_idbigint用户id逻辑外键
address_book_idbigint地址id逻辑外键
order_timedatetime下单时间
checkout_timedatetime付款时间
pay_methodint支付方式1微信支付,2支付宝支付
pay_statustinyint支付状态0未支付,1已支付,2退款
amountdecimal(10,2)订单金额
remarkvarchar(100)备注信息
phonevarchar(11)手机号冗余字段
addressvarchar(255)详细地址信息冗余字段
consigneevarchar(32)收货人冗余字段
cancel_reasonvarchar(255)订单取消原因
rejection_reasonvarchar(255)拒单原因
cancel_timedatetime订单取消时间
estimated_delivery_timedatetime预计送达时间
delivery_statustinyint配送状态1:立即送出 0:选择具体时间
delivery_timedatetime送达时间
pack_amountint打包费
tableware_numberint餐具数量
tableware_statustinyint餐具数量状态1按餐量提供 0选择具体数量

2). order_detail订单明细表

字段名数据类型说明备注
idbigint主键自增
namevarchar(32)商品名称冗余字段
imagevarchar(255)商品图片路径冗余字段
order_idbigint订单id逻辑外键
dish_idbigint菜品id逻辑外键
setmeal_idbigint套餐id逻辑外键
dish_flavorvarchar(50)菜品口味
numberint商品数量
amountdecimal(10,2)商品单价

说明:用户提交订单的时候,需要往订单表orders中插入一条记录,并且还需要再order_detail中插入一条或多条记录

2.2 代码开发
2.2.1 DTO设计

根据用户下单接口的参数设计DTO:

在sky-pojo模块,OrdersSubmitDTO

package com.sky.dto;
​
import com.fasterxml.jackson.annotation.JsonFormat;
import lombok.Data;
import java.io.Serializable;
import java.math.BigDecimal;
import java.time.LocalDateTime;
​
@Data
public class OrdersSubmitDTO implements Serializable {
    //地址簿id
    private Long addressBookId;
    //付款方式
    private int payMethod;
    //备注
    private String remark;
    //预计送达时间
    @JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime estimatedDeliveryTime;
    //配送状态  1立即送出  0选择具体时间
    private Integer deliveryStatus;
    //餐具数量
    private Integer tablewareNumber;
    //餐具数量状态  1按餐量提供  0选择具体数量
    private Integer tablewareStatus;
    //打包费
    private Integer packAmount;
    //总金额
    private BigDecimal amount;
}
2.2.2 VO设计

根据用户下单接口的返回结果设计VO:

在sky-pojo模块,OrderSubmitVO

package com.sky.vo;
​
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.io.Serializable;
import java.math.BigDecimal;
import java.time.LocalDateTime;
​
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class OrderSubmitVO implements Serializable {
    //订单id
    private Long id;
    //订单号
    private String orderNumber;
    //订单金额
    private BigDecimal orderAmount;
    //下单时间
    private LocalDateTime orderTime;
}
2.2.3 Controller层

创建OrderController并提供用户下单方法:

@PostMapping(/submit)
@Apioperation("用户下单")
public Result<OrderSubmitVO>orderSubmit(@RequestBody OrdersSubmitDTO ordersSubmitDTO){
log.info("用户下单{}",ordersSubmitDTO); OrderSubmitVO orderSubmitVO = orderService.oderSubmit(ordersSubmitDTO);
return Result.success(orderSubmitVO);
}
2.2.4 Service层接口

创建OrderService接口,并声明用户下单方法:

OrderSubmitVO orderSubmit(OrdersSubmitDTO ordersSubmitDTO);
2.2.5 Service层实现类

创建OrderServiceImpl实现OrderService接口:

 @Transactional
    @Override
    public OrderSubmitVO oderSubmit(OrdersSubmitDTO ordersSubmitDTO) {
        //处理各种业务异常(地址是否为空,购物车数据是否为空。)
        //查询地址
        AddressBook addressBook = addressBookMapper.getById(ordersSubmitDTO.getAddressBookId());
        if (addressBook == null){
            //抛出地址异常
            throw new AddressBookBusinessException(MessageConstant.ADDRESS_BOOK_IS_NULL);
        }

        //查询当前用户的购物车数据
        //获取当前用户id
        Long currentId = BaseContext.getCurrentId();
        ShoppingCart shoppingCart = new ShoppingCart();
        shoppingCart.setUserId(currentId);
        List<ShoppingCart> shoppingCarts = shoppingCartMapper.checkShoppingCartList(shoppingCart);
         if (shoppingCarts == null || shoppingCarts.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);
         orders.setStatus(Orders.PENDING_PAYMENT);
         //订单号使用当前时间的时间戳
         orders.setNumber(String.valueOf(System.currentTimeMillis()));
         orders.setUserId(currentId);
        orderMapper.insertOrders(orders);

        ArrayList<OrderDetail> orderDetailArrayList = new ArrayList<>();
        //向订单明细表插入n条数据
        //遍历购物车数据
        for (ShoppingCart cart : shoppingCarts) {
            OrderDetail orderDetail = new OrderDetail();
            BeanUtils.copyProperties(cart,orderDetail);
            orderDetail.setOrderId(orders.getId());//设置当前订单明细关联的订单id
            orderDetailArrayList.add(orderDetail);
        }
        orderDetailMapper.insertBatch(orderDetailArrayList);

        //清空当前用户的购物车数据
        shoppingCartMapper.deleteByUserId(shoppingCart);

        //封装VO返回结果
        OrderSubmitVO orderSubmitVO = OrderSubmitVO.builder()
                .id(orders.getId())
                .orderTime(orders.getOrderTime())
                .orderNumber(orders.getNumber())
                .orderAmount(orders.getAmount())
                .build();
        return  orderSubmitVO;
}
2.2.6 Mapper层

创建OrderMapper接口和对应的xml映射文件

    void insertOrders(Orders orders);

OrderMapper.xml

 <insert id="insertOrders" keyColumn="id" keyProperty="id" parameterType="com.sky.entity.Orders" useGeneratedKeys="true">
        insert into orders
        <trim prefix="(" suffix=")" suffixOverrides=",">
            <if test="id != null">id,</if>
            <if test="number != null">number,</if>
            <if test="status != null">status,</if>
            <if test="userId != null">user_id,</if>
            <if test="addressBookId != null">address_book_id,</if>
            <if test="orderTime != null">order_time,</if>
            <if test="checkoutTime != null">checkout_time,</if>
            <if test="payMethod != null">pay_method,</if>
            <if test="payStatus != null">pay_status,</if>
            <if test="amount != null">amount,</if>
            <if test="remark != null">remark,</if>
            <if test="phone != null">phone,</if>
            <if test="address != null">address,</if>
            <if test="userName != null">user_name,</if>
            <if test="consignee != null">consignee,</if>
            <if test="cancelReason != null">cancel_reason,</if>
            <if test="rejectionReason != null">rejection_reason,</if>
            <if test="cancelTime != null">cancel_time,</if>
            <if test="estimatedDeliveryTime != null">estimated_delivery_time,</if>
            <if test="deliveryStatus != null">delivery_status,</if>
            <if test="deliveryTime != null">delivery_time,</if>
            <if test="packAmount != null">pack_amount,</if>
            <if test="tablewareNumber != null">tableware_number,</if>
            <if test="tablewareStatus != null">tableware_status,</if>
        </trim>
        <trim prefix="values (" suffix=")" suffixOverrides=",">
            <if test="id != null">#{id,jdbcType=BIGINT},</if>
            <if test="number != null">#{number,jdbcType=VARCHAR},</if>
            <if test="status != null">#{status,jdbcType=INTEGER},</if>
            <if test="userId != null">#{userId,jdbcType=BIGINT},</if>
            <if test="addressBookId != null">#{addressBookId,jdbcType=BIGINT},</if>
            <if test="orderTime != null">#{orderTime,jdbcType=TIMESTAMP},</if>
            <if test="checkoutTime != null">#{checkoutTime,jdbcType=TIMESTAMP},</if>
            <if test="payMethod != null">#{payMethod,jdbcType=INTEGER},</if>
            <if test="payStatus != null">#{payStatus,jdbcType=TINYINT},</if>
            <if test="amount != null">#{amount,jdbcType=DECIMAL},</if>
            <if test="remark != null">#{remark,jdbcType=VARCHAR},</if>
            <if test="phone != null">#{phone,jdbcType=VARCHAR},</if>
            <if test="address != null">#{address,jdbcType=VARCHAR},</if>
            <if test="userName != null">#{userName,jdbcType=VARCHAR},</if>
            <if test="consignee != null">#{consignee,jdbcType=VARCHAR},</if>
            <if test="cancelReason != null">#{cancelReason,jdbcType=VARCHAR},</if>
            <if test="rejectionReason != null">#{rejectionReason,jdbcType=VARCHAR},</if>
            <if test="cancelTime != null">#{cancelTime,jdbcType=TIMESTAMP},</if>
            <if test="estimatedDeliveryTime != null">#{estimatedDeliveryTime,jdbcType=TIMESTAMP},</if>
            <if test="deliveryStatus != null">#{deliveryStatus,jdbcType=TINYINT},</if>
            <if test="deliveryTime != null">#{deliveryTime,jdbcType=TIMESTAMP},</if>
            <if test="packAmount != null">#{packAmount,jdbcType=INTEGER},</if>
            <if test="tablewareNumber != null">#{tablewareNumber,jdbcType=INTEGER},</if>
            <if test="tablewareStatus != null">#{tablewareStatus,jdbcType=TINYINT},</if>
        </trim>
    </insert>

创建OrderDetailMapper接口和对应的xml映射文件:

Mapper

  void insertBatch(List<OrderDetail> orderDetailList);

XML

    <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})
    </insert>

3. 订单支付

3.1 微信支付介绍

订单支付,就是完成付款功能,在苍穹外卖项目中,咱们选择的就是微信支付这中支付方式。

要实现微信支付就需要注册微信支付的一个商户号,这个商户号是必须要有一家企业并且有正规的营业执照。只有具备了这些资质之后,才可以去注册商户号,才能开通支付权限。

个人不具备这种资质,所以我们在学习微信支付时,最重要的是了解微信支付的流程,并且能够阅读微信官方提供的接口文档,能够和第三方支付平台对接起来就可以了。

可参考:产品中心 - 微信支付商户平台产品中心 - 微信支付商户平台

微信支付接入流程:

微信小程序支付时序图:

微信支付相关接口:

JSAPI下单: 商户系统调用该接口在微信支付服务后台生成预支付交易单

微信小程序调起支付:通过JSAPI下单接口获取到发起支付的必要参数prepay_id,然后使用微信支付提供的小程序方法调起小程序支付(对应时序图的第10步)

3.2 微信支付准备工作
3.2.1 如何保证数据安全?

完成微信支付有两个关键的步骤:

第一个就是需要在商户系统中调用微信后台的一个下单接口,就是生成预支付交易单。

第二个就是支付成功之后微信后台会给推送消息。

这两个接口数据的安全性,要求非常高

解决:微信提供的方式其实就是对数据进行加密,解密,签名多种方式。要完成数据加密解密,需要提前准备相应的一些文件,比如证书.

获取微信支付平台证书,商户私钥文件:

在后续的开发中,会使用到这两个文件,需要提前把这两个文件准备好

3.2.2 如何调用到商户系统?

微信后台会调用到商户系统推送支付的结果,在这里,我们就会遇到一个问题,就是微信后台怎么能调用到我们这个商户系统呢?因为这个调用过程,其实本质上也是一个Http请求

目前,商户系统它的IP地址就是当前自己电脑的IP地址,只是一个局域网内的ip地址,微信后台无法调用到。

解决:内网穿透。通过cpolar软件可以获得一个临时域名,而这个临时域名是一个公网IP,这样微信后台就可以请求到商户系统了。

cpolar软件的使用:

1).下载与安装

下载地址:cpolar - secure introspectable tunnels to localhost

2). cpolar指定authtoken

执行命令:

3).获取临时域名

执行命令:

获取域名:

4). 验证临时域名有效性

访问接口文档

使用临时域名访问、

这样就可以访问了。

用员工登录接口测试:

3.3 代码导入

导入资料中的微信支付功能代码即可

3.31 微信支付相关配置

application-dev.yml

sky:
  wechat:
    appid: wxcd2e39f677fd30ba
    secret: 84fbfdf5ea288f0c432d829599083637
    mchid : 1561414331
    mchSerialNo: 4B3B3DC35414AD50B1B755BAF8DE9CC7CF407606
    privateKeyFilePath: D:\apiclient_key.pem
    apiV3Key: CZBK51236435wxpay435434323FFDuv3
    weChatPayCertFilePath: D:\wechatpay_166D96F876F45C7D07CE98952A96EC980368ACFC.pem
    notifyUrl: https://www.weixin.qq.com/wxpay/pay.php
    refundNotifyUrl: https://www.weixin.qq.com/wxpay/pay.php

application.yml

sky:
  wechat:
    appid: ${sky.wechat.appid}
    secret: ${sky.wechat.secret}
    mchid : ${sky.wechat.mchid}
    mchSerialNo: ${sky.wechat.mchSerialNo}
    privateKeyFilePath: ${sky.wechat.privateKeyFilePath}
    apiV3Key: ${sky.wechat.apiV3Key}
    weChatPayCertFilePath: ${sky.wechat.weChatPayCertFilePath}
    notifyUrl: ${sky.wechat.notifyUrl}
    refundNotifyUrl: ${sky.wechat.refundNotifyUrl}

WeChatProperties.java:读取配置

package com.sky.properties;
​
import lombok.Data;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;
 
@Component
@ConfigurationProperties(prefix = "sky.wechat")
@Data
public classWeChatProperties {
    private String appid; //小程序的appid
    private String secret; //小程序的秘钥
    private String mchid; //商户号
    private String mchSerialNo; //商户API证书的证书序列号
    private String privateKeyFilePath //商户私钥文件
    private String apiV3Key; //证书解密的密钥
    private String weChatPayCertFilePath; //平台证书
    private String notifyUrl; //支付成功的回调地址
    private String refundNotifyUrl; //退款成功的回调地址
}

3.3.2 Mapper层

在OrderMapper.java中添加getByNumberAndUserId和update两个方法


    /**
     * 根据订单号和用户id查询订单
     * @param orderNumber
     * @param userId
     */
    @Select("select * from orders where number = #{orderNumber} and user_id = #{userId}")
    Orders getByNumberAndUserId(String OrderNumber, Long userId);


    /**
     * 修改订单信息
     * @param orders
     */
    void update(Orders orders);

在OrderMapper.xml中添加

<update id="update" parameterType="com.skhy.entity.Orders">

      update orders
      <set>
          <if test="cancelReason != null and cancelReason!= '' ">
              cancel_reason=#{cancelReason},
          </if>

          <if test ="rejectionReason != null and rejectionReason!='' ">

              rejection_reason =#{rejectionReason},
          </if>

          <if test="cancelTime != null">
                cancel_time=#{cancelTime},
          </if>

          <if test="payStatus != null">

              pay_status = #{payStatus},
          </if>
          <if test="payMethod != null">
                pay_method=#{payMethod},
            </if>
            <if test="checkoutTime != null">
                checkout_time=#{checkoutTime},
            </if>
            <if test="status != null">
                status = #{status},
            </if>
            <if test="deliveryTime != null">
                delivery_time = #{deliveryTime}
            </if>
        </set>
        where id = #{id}
</update>

     3.3.3 Service层

在OrderService.java中添加payment和paySuccess两个方法定义

@Autowired
private UserMapper userMapper;

@Autowired
private WeChatPayUtil weChatPayUtil;

   /**
     * 订单支付
     *
     * @param ordersPaymentDTO
     * @return
     */

     public OrderPaymentVO payment(OrdersPaymentDTO ordersPaymentDTO) throws Exception {
       // 当前登录用户id
       Long userId = BaseContext.getCurrentId();
       User  user  = userMapper.getById(userId);
       //调用微信支付接口,生成预支付交易单

        

}

在OrderServiceImpl.java中实现payment和paySuccess两个方法

@Autowired
private UserMapper usedrMapper;
@Autowired
Private WeChatPayUtil weChatPayUtil;
 /**
     * 订单支付
     *
     * @param ordersPaymentDTO
     * @return
     */
public OrderPaymentVO payment(OrdersPaymentDTO ordersPaymentDTO) throws Exception {

     //当前登录用户的id
     Long userId  =  BaseContext.getCurrentId();
     User user    =  userMapper.getById(userId);

     //调用微信支付接口,生成预支付交易单
     JSONObject  jsonObject  = weChatPayUtil.pay(
     ordersPaymentDTO.getOrderNumber(),//获取商户订单号
     new BigDecimal(0.01), //支付金额, 单位 元
     "苍穹外卖订单",// 商品描述
     user.getOpenid() // 微信用户的openid

     );
     if(jsonObject.getString("code") != null && 
     jsonObject.getString("code").equals("ORDERPAID")){

     throw new OrderBusinessException("该订单已支付")
}
     OrderPaymentVO  vo =  jsonObject.toJavaObject(OrderPaymentVO.class);
     return vo;

}
 /**
     * 支付成功,修改订单状态
     *
     * @param outTradeNo
     */
public void paySuccess(String outTradeNo) {

      //当前登录用户id
      Long userId = BaseContext.getCurrentId();

      //根据当前订单号查询当前用户的订单
      Orders ordersDB = orderMapper.getByNumberAndUserId(outTradeNo, userId);

      //根据订单id更新订单的状态,支付方式,支付状态,结账时间
      Orders orders  = Orders.builder()
              .id(OdersDB.getId())
              .status(Orders.TO_BE_CONFIRMED)
              .payStatus(Orders.PAID)
              .checkoutTime(LocalDateTime.now())
              .build();
      orderMapper.update(orders);

}
3.3.4 Controller层

在OrderController.java中添加payment方法

   /**
     * 订单支付
     *
     * @param ordersPaymentDTO
     * @return
     */
    @PutMapping("/payment")
    @ApiOperation("订单支付")

    public Result<OrderPaymentVO> payment(@RequestBody OdersPaymentDTO ordersPaymentDTO) throws Exception {
       log.info("订单支付: {}",ordersPaymentDTO);
       OrderPaymentVO oderPaymentVO = orderService.payment(odersPaymentDTO);
       log.info("生成预支付交易单:{}",orderPaymentVO);
       return Result.success(orderPaymentVO);

}



PayNotifyController.java

package com.sky.controller.notify;


import com.alibaba.druid.support.json.JSONUtils;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.sky.annotation.IgnoreToken;
import com.sky.properties.WeChatProperties;
import com.sky.service.OrderService;
import com.wechat.pay.contrib.apache.httpclient.util.AesUtil;
import lombok.extern.slf4j.Slf4j;
import org.apache.http.entity.ContentType;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.BufferedReader;
import java.nio.charset.StandardCharsets;
import java.util.HashMap;



/**
 * 支付回调相关接口
 */
@estController
@RequestMapping("/notify")
@SLf4j

public class PayNotifyController {
@Autowired
private OrderService orderService;
@Autowired
private WeChatProperties weCchatProperties;


/**
 * 支付成功回调
 */

@RequestMapping("/paySuccess")

public void paySuccessNotify(HttpServletRequest request, HttpServletResponse response) throws Exception {
}

    //读取数据
    String body = readData(request);
    log.info("支付成功回调:{}",body);

    //数据解密
    String plainText = decryptData(body);
    log.info("解密后的文本:{}, plainText")
    JSON.parseObject(plainText);
    String outTradeNo = jsonObject.getString("out_trade_no");//商户平台订单号
    String transactionId = jsonObject.getString("transaction_id");//微信支付交易号
    log.info("商户平台订单号:{}", outTradeNo);
    log.info("微信支付交易号:{}", transactionId);

    //业务处理,修改订单状态,来单提醒
    orderService.paySuccess(outTradeNo);

    //给微信响应
    responseToWeixin(response);

}
    /**
     * 读取数据
     *
     * @param request
     * @return
     * @throws Exception
     */

private String readData(HttpServletRequest request)  throws Exception {

        BufferedReader reader = request.getReader();
        StringBuilder result = new StringBuilder();
        String line = null;
        while ((line = reader.readLine()) != null){

            if (result.length() > 0){

                result.append("\n");
            }
            result.append(line);
        }
       return result.toString();
}


    /**
     * 数据解密
     *
     * @param body
     * @return
     * @throws Exception
     */

private String decryptData(String body) throws Exception {

   JSONObject resultObject  = JSON.parseObject(body);
   JSONObject resource = resultObject.getJSONObject("resource");
   String ciphertext = resource.getString("ciphertext");
   String nonce = resource.getString("nonce");
   String associatedData = resource.getString("associated_data");
   new AesUtil(weChatProperties.getApiV3Key().getBytes(StandardCharsets.UTF_8));
   //密文解密
   String plainText = 
   aesUtil.decryptToString(associatedData.getBytes(StandardCharsets.UTF_8),
   nonce.getBytes(StandardCharsets.UTF_8),ciphertext);

   return plainText;
}
    /**
     * 给微信响应
     * @param response
     */
private void responseToWeixin(HttpServletResponse response) throws Exception{
    response.setStatus(200);
    HashMap<Object,object> map = new HashMap<>();
     response.setHeader("Content-type", ContentType.APPLICATION_JSON.toString());
       response.getOutputStream().write(JSONUtils.toJSONString(map).getBytes(StandardCharsets.UTF_8));
        response.flushBuffer();
    }
}

3.35 步骤理解

1.小程序端,会向后端发送请求,请求OrderController中的payment方法,参数是订单的订单号还有付款方式;这两个封装成了一个OrdersPaymentDTO;然后调用Service中的payment方法。然后根据id得到用户对象

2.调用wechatPayUtil.pay方法实现调用微信支付接口,生成预支付交易单(将appid,description,mchid等等参数)传入微信指定后台,通过HttpClient实现

3.在weChatPayUtil中通过String prepayid = jsonObject.getString("prepay_id")来获得prepayid,然后进行一系列的操作对数据进行封装,加密和签名

4.在Controller返回支付参数给微信小程序,用于调起微信支付·

5.用户确认支付

6.用户确认支付就会调起微信支付(这里后面)都是小程序和微信后台之间的交互

7.返回支付结果

8.显示支付结果

9.假如支付成功,就会通过回调地址返回给商家,回调地址信息在第二步告诉微信后台

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值