微信小程序支付 + 公众号支付 (含源码)

微信支付.jpg

微信小程序支付 + 公众号支付 (含源码)
注:本文来源于:CodeCow · 程序牛 => 微信小程序支付 + 公众号支付 (含源码)


为啥会有这篇文章

最近换了个公司,接手了个别人的老项目,从git上拉取代码,导入idea时,我 TM 快哭了。。。
d57d738fd5ce884416a897485c0ff09f.jpg
我瞅了眼微信支付相关代码,顿时想上去就是 一Jao ! 我看到了这样的代码:
1595131368(1).jpg
我靠,你这么写,倒是一时爽;维护可就爽歪歪咯!!!
怎么办,怎么办。。。
21311e96de616e100210e0bbeb69cde2 - 副本.png

支付模块重构

别的先不说,重构之前我们先楼一眼微信部分支付流程图

1595137707(1) - 副本.jpg

注:上面是部分支付流程图(完整图片,可上官网查看)

官网:
https://pay.weixin.qq.com/wiki/doc/api/index.html

重构项目全貌

微信支付项目全貌.jpg
  • 项目架构:springboot(2.2.2) + mybatis
  • 数据库:mysql
  • 开发环境:jdk8、idea
  • 支付:微信小程序、公众号
重构之后,代码简介、清爽,它不香吗!!

微信支付实操

1、controller层
/**
 * Create By CodeCow on 2020/7/19.
 */
@Slf4j
@RestController
@RequestMapping("/pay")
public class WeiXinPayController {

    @Resource
    private WeiXinPayService weiXinPayService;

    @ApiOperation(value = "微信支付(公众号/小程序)", response = WechatPayMiniRespDTO.class)
    @ApiResponses({@ApiResponse(code = 0, message = "Success"), @ApiResponse(code = 500, message = "failed")})
    @PostMapping(value = "/miniPay", consumes = "application/json", produces = "application/json")
    public RespResult<WechatPayMiniRespDTO> weChatMiniBookOrderPay(@RequestBody @Valid WechatPayReqDTO wechatPayReqDTO) {
        try {
            return weiXinPayService.weChatBookOrderPay(wechatPayReqDTO);
        } catch (Exception e) {
            log.error("weChatBookOrderPay error: ", e);
            return RespResult.error(RespResultEnum.WEIXIN_PAY_ERROR.getCode(),
                    RespResultEnum.WEIXIN_PAY_ERROR.getMessage() + e.getMessage());
        }
    }
}

2、service层
/**
 * Create By CodeCow on 2020/7/19.
 */
public interface WeiXinPayService {

    RespResult weChatBookOrderPay(WechatPayReqDTO wechatPayReqDTO);
}

快快快,拿出小本本,提起纸笔。。。开始装B
骚操作 - 副本.gif
3、serviceImpl层
/**
 * Create By CodeCow on 2020/7/19.
 */
@Slf4j
@Service
public class WeiXinPayServiceImpl implements WeiXinPayService {

    @Resource
    private UserMapper userMapper; //用户mapper

    @Resource
    private WechatConfig wechatConfig; //微信配置(在最后)

    @Resource
    private OrderMapper orderMapper; //订单mapper

    @Resource
    private CommonTemplateService templateService; // 自己封装的restTemplate(在最后)

    @Override
    public RespResult weChatBookOrderPay(WechatPayReqDTO wechatPayReqDTO) {
        UserDO userDO = userMapper.selectById(wechatPayReqDTO.getUserId());
        Long orderCode = TimeUtil.getTimestamp();    //生成随机订单号
        WechatBookOrderReqDTO wechatBookOrderReqDTO  //封装微信下单请求参数(此方法在下面)
                = getBookOrderReqDTO(wechatPayReqDTO, orderCode, userDO);
        String sign =     //生成签名(签名工具类在最后)
                WeChatSignUtil.getSign(wechatBookOrderReqDTO, wechatConfig.getApiKey());
        wechatBookOrderReqDTO.setSign(sign);
        String reqXML = XstreamUtil.objectToXml(wechatBookOrderReqDTO); //把请求参数转化成 xml 格式
        log.info(">>>> reqXML >>>>:\n " + reqXML);  //打印日志
        ResponseEntity<String> responseEntity =     //使用自己封装的restTemplate向微信发起请求
                templateService.executePost(reqXML, wechatConfig.getBookOrderUrl());
        if (responseEntity.getStatusCode().value() != HttpStatus.OK.value()) { //判断请求是否成功
            return RespResult.error(RespResultEnum.ORDER_ERROR.getCode(),
                    RespResultEnum.ORDER_ERROR.getMessage() + responseEntity.getBody());
        }
        WechatBookOrderRespDTO wechatBookOrderRespDTO = // 把微信返回的参数封装到自己DTO中(DTO在最后)
                XstreamUtil.xmlToObject(responseEntity.getBody(), WechatBookOrderRespDTO.class);
        if (wechatBookOrderRespDTO.getXmlReturnCode().equals(WechatRespEnum.WECHAT_RETURN_CODE_FAIL.getCode())) { //判断微信响应结果
            return RespResult.error(RespResultEnum.ORDER_RETURN_ERROR.getCode(),
                    RespResultEnum.ORDER_RETURN_ERROR.getMessage() + wechatBookOrderRespDTO.getXmlReturnMsg());
        } else if (wechatBookOrderRespDTO.getXmlResultCode().equals(WechatRespEnum.WECHAT_RESULT_CODE_FAIL.getCode())) {
            return RespResult.error(RespResultEnum.ORDER_RESULT_ERROR.getCode(),
                    RespResultEnum.ORDER_RESULT_ERROR.getMessage() + wechatBookOrderRespDTO.getXmlErrCodeDes());
        }
        insertRefuelOrder(userDO, wechatPayReqDTO, orderCode); //向数据库保存订单信息(方法在下面)
        String nonceStr = WeChatSignUtil.getNonceStr();
        long timestamp = TimeUtil.getTimestamp();
        String paySign = WeChatPaySignUtil.getPaySign(wechatConfig.getAppId(), nonceStr, // 重新生成签名(前段调起支付需要)
                wechatBookOrderRespDTO.getXmlPrepayId(), wechatConfig.getSignType(), String.valueOf(timestamp), wechatConfig.getApiKey());
        WechatPayMiniRespDTO wechatPayMiniRespDTO =  // 返回前段支付需要的DTO(DTO在最后)
                getPayMiniRespDTO(paySign, nonceStr, timestamp, orderCode, wechatBookOrderRespDTO);
        return RespResult.success(wechatPayMiniRespDTO); // 返回数据
    }

    private WechatBookOrderReqDTO getBookOrderReqDTO(WechatPayReqDTO wechatPayReqDTO, Long orderCode, UserDO userDO) {
        return WechatBookOrderReqDTO.builder()
                .appid(wechatConfig.getAppId())
                .mch_id((wechatConfig.getMchId()))
                .openid(userDO.getOpenid())
                .trade_type(TradeTypeEnum.WECHAT_JSAPI.getType())
                .nonce_str(WeChatSignUtil.getNonceStr())
                .sign_type(wechatConfig.getSignType())
                .body(wechatPayReqDTO.getDescription())
                .out_trade_no(String.valueOf(orderCode))
                .total_fee(wechatPayReqDTO.getTotalFee())
                .spbill_create_ip(wechatPayReqDTO.getIp())
                .time_start(String.valueOf(TimeUtil.getTimestamp()))
                .notify_url(wechatConfig.getNotifyUrl())
                .build();
    }

    private void insertRefuelOrder(UserDO userDO, WechatPayReqDTO wechatPayReqDTO, Long orderCode) {
        OrderDO orderDO = OrderDO.builder()
                .id(TimeUtil.getTimestamp())
                .userId(String.valueOf(userDO.getUserId()))
                .orderCode(String.valueOf(orderCode))
                .status("0")
                .totalFee(new BigDecimal(wechatPayReqDTO.getTotalFee()))
                .payType("1") //支付类型(1-微信,2-支付宝)
                .detail(wechatPayReqDTO.getDescription())
                .createTime(TimeUtil.now())
                .build();
        orderMapper.insert(orderDO);
    }

    private WechatPayMiniRespDTO getPayMiniRespDTO(String paySign, String nonceStr, long timestamp,
                                                   Long orderCode, WechatBookOrderRespDTO wechatBookOrderRespDTO) {
        return WechatPayMiniRespDTO.builder()
                .paySign(paySign)
                .nonceStr(nonceStr)
                .outTradeNo(String.valueOf(orderCode))
                .ppackage(ColumnNames.PARAM_PREPAY_ID + wechatBookOrderRespDTO.getXmlPrepayId())
                .signType(wechatConfig.getSignType())
                .timeStamp(String.valueOf(timestamp))
                .tradeType(wechatBookOrderRespDTO.getXmlTradeType())
                .build();
    }
}

4、WechatPayMiniRespDTO
/**
 * Create By CodeCow on 2020/7/19.
 */
@Data
@Builder
@ApiModel("微信小程序/公众号支付返回参数")
public class WechatPayMiniRespDTO {

    @ApiModelProperty(value = "时间戳从1970年1月1日00:00:00至今的秒数,即当前的时间", required = true)
    private String timeStamp;

    @ApiModelProperty(value = "随机字符串,不长于32位", required = true)
    private String nonceStr;

    @ApiModelProperty(value = "统一下单接口返回的 prepay_id 参数值", required = true)
    @JsonProperty(value = "package")
    private String ppackage;

    @ApiModelProperty(value = "MD5", required = true)
    private String signType;

    @ApiModelProperty(value = "前端调用微信支付功能所需签名", required = true)
    private String paySign;

    @ApiModelProperty(value = "交易类型", required = true)
    private String tradeType;

    @ApiModelProperty(value = "订单号")
    private String outTradeNo;

}
代码实在有点多,写到这,手都酸了。。。
f2a2a13a324e4901bc9ead4264f292c0 - 副本.jpg

好文推荐

小声BB

  • 由于支付代码比较多,小编就不挨个粘贴了
  • 若需要完整“ 源码 ” 可在公众号内回复“ 微信支付,发百度网盘

公众号二维码.jpg

弱弱问一句

  • 看完了,支付还用自己写吗
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352