尚品汇商城

一、支付宝介绍

1.1 支付宝简介

  1. [支付宝(中国)网络技术有限公司](https://baike.baidu.com/item/%E6%94%AF%E4%BB%98%E5%AE%9D%EF%BC%88%E4%B8%AD%E5%9B%BD%EF%BC%89%E7%BD%91%E7%BB%9C%E6%8A%80%E6%9C%AF%E6%9C%89%E9%99%90%E5%85%AC%E5%8F%B8/23241708) [1] 是国内的第三方支付平台,致力于提供“简单、安全、快速”的支付解决方案 [2] 。支付宝公司从2004年建立开始,始终以“信任”作为产品和服务的核心。旗下有“支付宝”与“支付宝钱包”两个独立品牌。自2014年第二季度开始成为当前全球最大的[移动支付](https://baike.baidu.com/item/%E7%A7%BB%E5%8A%A8%E6%94%AF%E4%BB%98/565488)厂商。<br /> <br /> 当用户提交订单会跳转到选择支付渠道页面!<br />![](https://cdn.nlark.com/yuque/0/2021/png/25621085/1639539202718-5a4c813c-96df-4453-9578-4b1546980c8a.png#height=173&id=TDsp3&originHeight=508&originWidth=1219&originalType=binary&ratio=1&status=done&style=none&width=414)<br /> 当用户点击立即支付时生成支付的二维码<br /> <br />![](https://cdn.nlark.com/yuque/0/2021/png/25621085/1639539203555-7c090324-d35e-4552-8470-a51cfca559fa.png#height=255&id=a2OaX&originHeight=614&originWidth=998&originalType=binary&ratio=1&status=done&style=none&width=415)<br />使用支付宝app 进行扫码支付<br /> <br />![](https://cdn.nlark.com/yuque/0/2021/png/25621085/1639539204402-69c308e3-c709-4778-82bf-613d8dfb44ea.png#height=162&id=SFJRf&originHeight=270&originWidth=646&originalType=binary&ratio=1&status=done&style=none&width=388)<br /> <br />![](https://cdn.nlark.com/yuque/0/2021/png/25621085/1639539205869-340fe571-166a-4ba5-beee-0fdc11026676.png#height=170&id=QTSVO&originHeight=170&originWidth=416&originalType=binary&ratio=1&status=done&style=none&width=415)

1.2 过程分析

15 支付宝支付 - 图1

1.3 对接支付宝的准备工作

1、申请条件

1. 企业或个体工商户可申请;
2. 提供真实有效的营业执照,且支付宝账户名称需与营业执照主体一致;
3. 网站能正常访问且页面信息有完整商品内容;
4. 网站必须通过ICP备案,个体户备案需与账户主体一致。
(团购类网站不支持个体工商户签约)

支付手续费
15 支付宝支付 - 图2

1.4 申请步骤:

1、 支付宝商家中心中申请 https://www.alipay.com/
2、 https://b.alipay.com/signing/productSetV2.htm?mrchportalwebServer=https%3A%2F%2Fmrchportalweb.alipay.com
15 支付宝支付 - 图3
一个工作日后登录到蚂蚁金服开发者中心中:
15 支付宝支付 - 图4
可以查看到一个已经签约上线的应用。 其中非常重要的是这个APPID,需要记录下来之后的程序中要用到这个参数。
点击查看
15 支付宝支付 - 图5
到此为止,电商网站可以访问支付宝的最基本的准备已经完成。

接下来搭建支付模块

二、支付模块搭建

支付宝开发手册:https://docs.open.alipay.com/270/105900/

2.1 搭建service-payment

2.1.1 搭建service-payment

搭建方式如service-order

2.1.2 修改pom.xml

<?xml version=”1.0” encoding=”UTF-8”?>
<project xmlns=”http://maven.apache.org/POM/4.0.0“ xmlns:xsi=”http://www.w3.org/2001/XMLSchema-instance
xsi:schemaLocation=”http://maven.apache.org/POM/4.0.0
https://maven.apache.org/xsd/maven-4.0.0.xsd”
> <modelVersion>4.0.0</modelVersion> <parent> <groupId>com.atguigu.gmall</groupId> <artifactId>service</artifactId> <version>1.0</version> </parent>
<version>1.0</version> <artifactId>service-payment</artifactId> <packaging>jar</packaging> <name>service-payment</name> <description>service-payment</description>
<dependencies> <dependency> <groupId>com.atguigu.gmall</groupId> <artifactId>service-order-client</artifactId> <version>1.0</version> </dependency> </dependencies>
<build> <finalName>service-payment</finalName> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </build>
</project>

2.1.3 添加配置

bootstrap.properties

spring.application.name=service-payment
spring.profiles.active
=dev
spring.cloud.nacos.discovery.server-addr
=192.168.200.129:8848
spring.cloud.nacos.config.server-addr
=192.168.200.129:8848
spring.cloud.nacos.config.prefix
=${spring.application.name}
spring.cloud.nacos.config.file-extension
=yaml
spring.cloud.nacos.config.shared-configs[0].data-id
=common.yaml

2.1.4 启动类

package com.atguigu.gmall.payment;


@SpringBootApplication
@ComponentScan({“com.atguigu.gmall”})
@EnableDiscoveryClient
@EnableFeignClients(basePackages= {“com.atguigu.gmall”})
public class ServicePaymentApplication {

public static void main(String[] args) {
SpringApplication.run(ServicePaymentApplication.class, args);
}

}

2.2 导入sdk包

https://mvnrepository.com/artifact/com.alipay.sdk/alipay-sdk-java


<dependency> <groupId>com.alipay.sdk</groupId> <artifactId>alipay-sdk-java</artifactId> <version>4.8.73.ALL</version> </dependency>

三、显示付款页面信息

支付页面信息展示流程
15 支付宝支付 - 图6

3.1 在根据订单id获取订单信息接口

3.1.1 在service-order添加接口

OrderService接口/
根据订单Id 查询订单信息
@param **
orderId
__
_ @return
**
**
/
_OrderInfo getOrderInfo(Long orderId);
实现类
@Override
public OrderInfo getOrderInfo(Long orderId) {
OrderInfo orderInfo = orderInfoMapper.selectById(orderId);
QueryWrapper queryWrapper = new QueryWrapper<>();
queryWrapper.eq(“order_id”, orderId);
List orderDetailList = orderDetailMapper.selectList(queryWrapper);
orderInfo.setOrderDetailList(orderDetailList);
return orderInfo;
}
OrderApiController
/
内部调用获取订单
@param **
orderId
__
@return
**
**
/
@GetMapping(“inner/getOrderInfo/{orderId}”)public OrderInfo getOrderInfo(@PathVariable(value = “orderId”) Long orderId){
return orderService.getOrderInfo(orderId);
}

3.1.2 在service-order-client添加接口

OrderFeignClient/
获取订单
@param **
orderId
__
@return
**
**
/
@GetMapping(“/api/order/inner/getOrderInfo/{orderId}”)
OrderInfo getOrderInfo(@PathVariable(value = “orderId”) Long orderId);
OrderDegradeFeignClient@Override
public OrderInfo getOrderInfo(Long orderId) {
return null;
}

3.2 server-gateway模块网关配置

- id: web-payment
uri: lb://web-all
predicates:
- Host=payment.gmall.com
- id: service-payment
uri: lb://service-payment
predicates:
- Path=//payment/* # 路径匹配


3.3 创建支付控制器PaymentController

Web-all模块

package com.atguigu.gmall.payment.controller; @Controller
public class PaymentController {

@Autowired
private OrderFeignClient orderFeignClient;

/
支付页
@param **
request
__
@return
**
**
/
@GetMapping(“pay.html”)
public String success(HttpServletRequest request, Model model) {
String orderId = request.getParameter(“orderId”);
OrderInfo orderInfo = orderFeignClient.getOrderInfo(Long.parseLong(orderId));
model.addAttribute(“orderInfo”, orderInfo);
return “payment/pay”;
}
}

3.4 页面渲染

页面资源: \templates\payment\pay.html

<div class=”checkout-tit”> <h4 class=”tit-txt”><span class=”success-icon”></span><span class=”success-info”>订单提交成功,请您及时付款,以便尽快为您发货~~</span></h4> <div class=”paymark”> <span class=”fl”>请您在提交订单<em class=”orange time”>4小时</em>之内完成支付,超时订单会自动取消。订单号:<em th:text=”${orderInfo.id}”>00</em></span> <span class=”fr”><em class=”sui-lead”>应付金额:</em><em class=”orange money” th:text=”‘¥’+${orderInfo.totalAmount}”>¥1</em></span> </div> </div>

四、支付功能实现


15 支付宝支付 - 图7

支付宝有了同步通知为什么还需要异步通知?

同步回调两个作用
第一是从支付宝的页面上返回自己的网站继续后续操作;
第二是携带支付状态的get参数;让自己的网站用于验证;
同步通知后;还需要异步通知主要是为了防止出现意外情况;
因为涉及到金钱;这是一个对安全和稳定要求比较严格的场景;
如果同步通知的过程中;用户不小心关闭了浏览器;或者浏览器卡死了;
异步也能收到通知;记录支付状态;
即便是用户端没问题;万一自己的服务器网络异常了一下呢?
如果自己的服务器没有正确返回接受到通知的状态;
支付宝的服务器会在一段时间内持续的往自己的服务器发送异步通知
一直到成功;
顺便去确认了下;这个一段时间是:25 小时以内完成 8 次通知(通知的间隔频率一般是:4m,10m,10m,1h,2h,6h,15h)

4.1 思路分析

  1. 将支付数据保存到数据库,以便跟支付宝进行对账
    2. 生成要支付的二维码
    生成二维码需要的参数列表请参考官方文档
    https://opendocs.alipay.com/open/270/105899

    4.2 保存支付信息的表结构

    表结构 payment_info
    15 支付宝支付 - 图8
id 主键自动生成
out_trade_no 订单中已生成的对外交易编号。订单中获取
order_id 订单编号
payment_type 支付类型(微信与支付宝)
trade_no 交易号,回调时生成
total_amount 订单金额。订单中获取
subject 交易内容。利用商品名称拼接。
payment_status 支付状态,默认值未支付。
create_time 创建时间,当前时间。
callback_time 回调时间,初始为空,支付宝异步回调时记录
callback_content 回调信息,初始为空,支付宝异步回调时记录


4.3 编写接口,实现类PaymentService

4.3.1 接口 PaymentService

| package com.atguigu.gmall.payment.service;
import com.atguigu.gmall.model.order.OrderInfo;
public interface PaymentService {
/
保存交易记录
@param **
orderInfo
__
*@param paymentType 支付类型(1:微信 2:支付宝)
*/
void savePaymentInfo(OrderInfo orderInfo, String paymentType);

} | | —- |

4.3.2 定义PaymentMapper接口

PaymentInfoMapper

package com.atguigu.gmall.payment.mapper;
@Mapper
public interface PaymentInfoMapper extends BaseMapper {
}

4.3.3 实现类PaymentServiceImpl

package com.atguigu.gmall.payment.service.impl;

@Service
public class PaymentServiceImpl implements PaymentService {

@Autowired
private PaymentInfoMapper paymentInfoMapper;


@Override
public void savePaymentInfo(OrderInfo orderInfo, String paymentType) {
QueryWrapper queryWrapper = new QueryWrapper<>();
queryWrapper.eq(“order_id”, orderInfo.getId());
queryWrapper.eq(“payment_type”, paymentType);
Integer count = paymentInfoMapper.selectCount(queryWrapper);
if(count > 0) return;

// 保存交易记录
_PaymentInfo paymentInfo = new PaymentInfo();
paymentInfo.setCreateTime(new Date());
paymentInfo.setOrderId(orderInfo.getId());
paymentInfo.setPaymentType(paymentType);
paymentInfo.setOutTradeNo(orderInfo.getOutTradeNo());
paymentInfo.setPaymentStatus(PaymentStatus.**_UNPAID
.name());
paymentInfo.setSubject(orderInfo.getTradeBody());
_//paymentInfo.setSubject(“test”);
_paymentInfo.setTotalAmount(orderInfo.getTotalAmount());

paymentInfoMapper**.insert(paymentInfo);
}
}

4.4 编写支付宝支付接口

4.4.1 制作AlipayClient工具类

创建配置类

package com.atguigu.gmall.payment.config;


@Configuration public class AlipayConfig {

@Value(“${alipay_url}”)
private String alipay_url;

@Value(“${app_private_key}”)
private String app_private_key;

@Value(“${app_id}”)
private String app_id;


public final static String format=“json”;
public final static String charset=“utf-8”;
public final static String sign_type=“RSA2”;

public static String return_payment_url;

public static String notify_payment_url;

public static String return_order_url;

public static String alipay_public_key;

@Value(“${alipay_public_key}”)
public void setAlipaypublic_key(String alipay_public_key) {
AlipayConfig._alipay_public_key
= alipaypublic_key;
}

@Value(“${return_payment_url}”)
public void setReturn_url(String return_payment_url) {
AlipayConfig._return_payment_url
= returnpayment_url;
}

@Value(“${notify_payment_url}”)
public void setNotify_url(String notify_payment_url) {
AlipayConfig._notify_payment_url
= notifypayment_url;
}

@Value(“${return_order_url}”)
public void setReturn_order_url(String return_order_url) {
AlipayConfig._return_order_url
= returnorder_url;
}

//
@Bean
public AlipayClient alipayClient(){
// AlipayClient alipayClient = new DefaultAlipayClient(“https://openapi.alipay.com/gateway.do“, APPID, APP_PRIVATE_KEY, FORMAT, CHARSET, ALIPAY_PUBLIC_KEY, SIGN_TYPE); //获得初始化的AlipayClient
_AlipayClient alipayClient=new DefaultAlipayClient(alipay_url,app_id,app_private_key,**_format
,charset, alipay_public_key,sign_type );
return **alipayClient;
}

}

4.4.2 编写支付宝下单

4.4.2.1 接口

package com.atguigu.gmall.payment.service;


public interface AlipayService {

String createaliPay(Long orderId);
}

4.4.2.2 实现类

package com.atguigu.gmall.payment.service.impl;


@Service
public class AlipayServiceImpl implements AlipayService {

@Autowired
private AlipayClient alipayClient;

@Autowired
private OrderFeignClient orderFeignClient;

@Autowired
private PaymentService paymentService;


@Override
public String createaliPay(Long orderId) {
// 根据订单Id 获取orderInfo
_OrderInfo orderInfo = orderFeignClient.getOrderInfo(orderId);
if (“PAID”.equals(orderInfo.getOrderStatus()) || “CLOSED”.equals(orderInfo.getOrderStatus())){
return “该订单已经完成或已经关闭!”;
}
// 调用保存交易记录方法! paymentService.savePaymentInfo(orderInfo, PaymentType.**_ALIPAY.name());

String form =
“”;
_// 创建 AlipayClient 对象! AlipayClient 这个对象注入到 spring 容器中! // AlipayClient alipayClient = new DefaultAlipayClient( “https://openapi.alipay.com/gateway.do“ , APP_ID, APP_PRIVATE_KEY, FORMAT, CHARSET, ALIPAY_PUBLIC_KEY, SIGN_TYPE); //获得初始化的AlipayClient
_AlipayTradePagePayRequest alipayRequest =
new AlipayTradePagePayRequest(); //创建API对应的request
// 同步回调
http://api.gmall.com/api/payment/alipay/callback/return
_ _alipayRequest.setReturnUrl(AlipayConfig._return_payment_url);
_// 异步回调 _alipayRequest.setNotifyUrl(
http://domain.com/CallBack/notify_url.jsp); _//在公共参数中设置回跳和通知地址 // 封装业务参数 _HashMap map = new HashMap<>();
_// 第三方业务编号! _map.put(
“outtrade_no”,orderInfo.getOutTradeNo());
map.put(
“product_code”,“FAST_INSTANT_TRADE_PAY”);
map.put(
“total_amount”,“0.01”);
map.put(
“subject”**,orderInfo.getTradeBody());
// 设置二维码过期时间 map.put(“timeout_express”,“10m”);
alipayRequest.setBizContent(JSON._toJSONString
(map));
try {
form = alipayClient.pageExecute(alipayRequest).getBody(); //调用SDK生成表单 } catch (AlipayApiException e) {
e.printStackTrace();
}
return form;
}
}

4.4.2.3 控制器

| AlipayController
@Controller
@RequestMapping(“/api/payment/alipay”)public class AlipayController {

@Autowired<br />    **private **AlipayService **alipayService**;

 @RequestMapping(**"submit/{orderId}"**)<br />@ResponseBody    **public **String submitOrder(@PathVariable Long orderId){<br />        String from = **alipayService**.createaliPay(orderId);<br />        **return **from;<br />    }<br />}<br />  |

| —- |

4.4.3 前端页面

<ul class=”payType”>
<a th:href=”@{http://api.gmall.com/api/payment/alipay/submit/{orderId}(orderId=${orderInfo.id})}“ target=”_blank”><li><img src=”./img/_/pay2.jpg”></li></a>

</ul>

五、支付后回调—同步回调

15 支付宝支付 - 图9

5.1 控制器AlipayController

/
支付宝回调
@return
* /
@RequestMapping(“callback/return”)
public String callBack() {
// 同步回调给用户展示信息
return “redirect:” + AlipayConfig.return_order_url;
}

5.2 在web-all 项目中添加对应的返回控制器

/
支付成功页
@return
* /
@GetMapping(“pay/success.html”)public String success() {
return “payment/success”;
}

六、支付宝回调—异步回调

15 支付宝支付 - 图10
异步回调有两个重要的职责:
确认并记录用户已付款通知电商模块。新版本的支付接口已经取消了同步回调的支付结果传递。所以用户付款成功与否全看异步回调。

接收到回调要做的事情:
1、 验证回调信息的真伪
2、 验证用户付款的成功与否
3、 把新的支付状态写入支付信息表{paymentInfo}中。
4、 通知电商
5、 给支付宝返回回执。

6.1 控制器AlipayController

@Autowired
private PaymentService paymentService;
@PostMapping(“callback/notify”)@ResponseBodypublic String callbackNotify(@RequestParam Map paramMap){
System.out.println(“你回来了….”);
// 需要如何处理!https://opendocs.alipay.com/open/270/105902 // Map paramsMap = … //将异步通知中收到的所有参数都存放到map中 boolean signVerified = false; //调用SDK验证签名 try {
signVerified = AlipaySignature._rsaCheckV1
(paramMap, AlipayConfig.alipay_public_key, AlipayConfig.charset, AlipayConfig.sign_type);
} catch (AlipayApiException e) {
e.printStackTrace();
}
// 获取交易状态! _String tradeStatus = paramMap.get(“trade_status”);
// 验证outtrade_no 是否为 商户系统中创建的订单号; // 可以使用 out_trade_no 查询交易记录中的数据! _String outTradeNo = paramMap.get(“out_trade_no”);
PaymentInfo paymentInfoQuery = paymentService.getPaymentInfo(outTradeNo, PaymentType.**_ALIPAY
.name());
if (paymentInfoQuery==null){
return “failure”;
}
// total_amount , seller_id , app_id
// String outTradeNo = paramMap.get(“total_amount”); paymentInfoQuery.getTotalAmount();
// String sellerId = paramMap.get(“seller_id”);
// String appId = paramMap.get(“app_id”); naocs 的appId 比较!
_
if(signVerified){
//
_TODO
验签成功后,按照支付结果异步通知中的描述,对支付结果中的业务内容进行二次校验,校验成功后在response中返回success并继续商户自身业务处理,校验失败返回failure
**// 只有交易通知状态为 TRADESUCCESS 或 TRADE_FINISHED 时,支付宝才会认定为买家付款成功 if (“TRADE_SUCCESS”.equals(tradeStatus) || “TRADE_FINISHED”.equals(tradeStatus)){
// 支付成功之后,需要更新交易记录状态! // 更新payment_status , trade_no ,callback_time ,callback_content
paymentService.paySuccess(outTradeNo,PaymentType.ALIPAY.name(),paramMap);
return “success”;
}
return “failure”;
}else{
// TODO 验签失败则记录异常日志,并在response中返回failure.
_
_return “failure”
;
}
}

6.2 接口PaymentService

/
获取paymentInfo 对象
@param **
outTradeNo
__
*@param name
__
@return
**
**
/_PaymentInfo getPaymentInfo(String outTradeNo, String name);
/
支付成功更新交易记录方法
@param _outTradeNo
__
_* @param _name
__
_* @param _paramMap
__
*/void paySuccess(String outTradeNo, String name, Map paramMap);
_/

根据outTradeNo 支付方式name 更新数据 @param **outTradeNo
_
_* @param _name
__
_* @param _paymentInfo
__
*/void **updatePaymentInfo(String outTradeNo, String name, PaymentInfo paymentInfo);

6.3 实现类

| @Overridepublic PaymentInfo getPaymentInfo(String outTradeNo, String name) {
// select from payment_info where out_trade_no = ? and payment_type = ?
_QueryWrapper paymentInfoQueryWrapper = new QueryWrapper<>();
paymentInfoQueryWrapper.eq(“out_trade_no”,outTradeNo);
paymentInfoQueryWrapper.eq(“payment_type”,name);
return paymentInfoMapper.selectOne(paymentInfoQueryWrapper);
}
@Override*public void
paySuccess(String outTradeNo, String name, Map paramMap) {
// 通过outTradeNo,name 查询到paymentInfo 对象! PaymentInfo paymentInfoQuery = getPaymentInfo(outTradeNo, name);
// 判断 if(“PAID”.equals(paymentInfoQuery.getPaymentStatus()) || “CLOSED”.equals(paymentInfoQuery.getPaymentStatus())){
return;
}
// 调用mapper
PaymentInfo paymentInfo = new PaymentInfo();
paymentInfo.setCallbackTime(new Date());
paymentInfo.setTradeNo(paramMap.get(“trade_no”));
paymentInfo.setPaymentStatus(PaymentStatus.**_PAID
**.name());
paymentInfo.setCallbackContent(paramMap.toString());
_// 第一个参数设置更新的内容,第二个参数表示更新的条件! // QueryWrapper paymentInfoQueryWrapper = new QueryWrapper<>();
// paymentInfoQueryWrapper.eq(“out_trade_no”,outTradeNo);
// paymentInfoQueryWrapper.eq(“payment_type”,name);
// paymentInfoMapper.update(paymentInfo,paymentInfoQueryWrapper);

_**this**.updatePaymentInfo(outTradeNo,name,paymentInfo);

}_// 更新交易状态记录!public void updatePaymentInfo(String outTradeNo, String name, PaymentInfo paymentInfo) {
QueryWrapper paymentInfoQueryWrapper = new QueryWrapper<>();
paymentInfoQueryWrapper.eq(“out_trade_no”,outTradeNo);
paymentInfoQueryWrapper.eq(“payment_type”,name);
paymentInfoMapper.update(paymentInfo,paymentInfoQueryWrapper);
}
| | —- |

七、退款

直接在浏览器发起请求即可!
商户与客户协商一致的情况下,才可以退款!

7.1 接口

boolean refund(Long orderId);

7.2 实现类

@Override
public boolean refund(Long orderId) {
_AlipayTradeRefundRequest request = new AlipayTradeRefundRequest();

OrderInfo orderInfo = orderFeignClient.getOrderInfo(orderId);
HashMap map = new HashMap<>();
map.put(“out_trade_no”, orderInfo.getOutTradeNo());
map.put(“refund_amount”, orderInfo.getTotalAmount());
map.put(“refund_reason”, “颜色浅了点”);
// outrequest_no

_request.setBizContent(JSON._toJSONString
(map));
AlipayTradeRefundResponse response = null;
try {
response = alipayClient.execute(request);
} catch (AlipayApiException e) {
e.printStackTrace();
}
if (response.isSuccess()) {
// 更新交易记录 : 关闭
_PaymentInfo paymentInfo = new PaymentInfo();
paymentInfo.setPaymentStatus(PaymentStatus.**_ClOSED
.name());
paymentService.updatePaymentInfo(orderInfo.getOutTradeNo(), paymentInfo);
return true;
}
else {
return false**;
}
}

7.3 控制器

// 发起退款!http://localhost:8205/api/payment/alipay/refund/20 @RequestMapping(“refund/{orderId}”)@ResponseBody
public Result refund(@PathVariable(value = “orderId”)Long orderId) {
// 调用退款接口
boolean flag = alipayService.refund(orderId);

return Result.ok(flag);
}