mirror of
https://gitee.com/binary/weixin-java-tools.git
synced 2026-01-23 21:32:09 +08:00
194 lines
6.6 KiB
Markdown
194 lines
6.6 KiB
Markdown
|
|
# 微信支付预约扣费功能使用说明
|
|||
|
|
|
|||
|
|
## 概述
|
|||
|
|
|
|||
|
|
微信支付预约扣费功能(连续包月功能)允许商户在用户授权的情况下,按照约定的时间和金额,自动从用户的支付账户中扣取费用。主要适用于连续包月、订阅服务等场景。
|
|||
|
|
|
|||
|
|
## 功能特性
|
|||
|
|
|
|||
|
|
- **预约扣费**:创建未来某个时间点的扣费计划
|
|||
|
|
- **查询预约**:查询已创建的扣费计划状态
|
|||
|
|
- **取消预约**:取消已创建的扣费计划
|
|||
|
|
- **立即扣费**:立即执行扣费操作
|
|||
|
|
- **扣费记录查询**:查询历史扣费记录
|
|||
|
|
|
|||
|
|
## 快速开始
|
|||
|
|
|
|||
|
|
### 1. 获取服务实例
|
|||
|
|
|
|||
|
|
```java
|
|||
|
|
// 通过 WxPayService 获取预约扣费服务
|
|||
|
|
SubscriptionBillingService subscriptionService = wxPayService.getSubscriptionBillingService();
|
|||
|
|
```
|
|||
|
|
|
|||
|
|
### 2. 创建预约扣费
|
|||
|
|
|
|||
|
|
```java
|
|||
|
|
// 创建预约扣费请求
|
|||
|
|
SubscriptionScheduleRequest request = new SubscriptionScheduleRequest();
|
|||
|
|
request.setOutTradeNo("subscription_" + System.currentTimeMillis());
|
|||
|
|
request.setOpenid("用户的openid");
|
|||
|
|
request.setDescription("腾讯视频VIP会员");
|
|||
|
|
request.setScheduleTime("2024-09-01T10:00:00+08:00");
|
|||
|
|
|
|||
|
|
// 设置扣费金额
|
|||
|
|
SubscriptionAmount amount = new SubscriptionAmount();
|
|||
|
|
amount.setTotal(3000); // 30元,单位为分
|
|||
|
|
amount.setCurrency("CNY");
|
|||
|
|
request.setAmount(amount);
|
|||
|
|
|
|||
|
|
// 设置扣费计划(可选)
|
|||
|
|
BillingPlan billingPlan = new BillingPlan();
|
|||
|
|
billingPlan.setPlanType("MONTHLY"); // 按月扣费
|
|||
|
|
billingPlan.setPeriod(1); // 每1个月
|
|||
|
|
billingPlan.setTotalCount(12); // 总共12次
|
|||
|
|
request.setBillingPlan(billingPlan);
|
|||
|
|
|
|||
|
|
// 发起预约扣费
|
|||
|
|
SubscriptionScheduleResult result = subscriptionService.scheduleSubscription(request);
|
|||
|
|
System.out.println("预约扣费ID: " + result.getSubscriptionId());
|
|||
|
|
```
|
|||
|
|
|
|||
|
|
### 3. 查询预约扣费
|
|||
|
|
|
|||
|
|
```java
|
|||
|
|
// 通过预约扣费ID查询
|
|||
|
|
String subscriptionId = "从预约扣费结果中获取的ID";
|
|||
|
|
SubscriptionQueryResult queryResult = subscriptionService.querySubscription(subscriptionId);
|
|||
|
|
System.out.println("预约状态: " + queryResult.getStatus());
|
|||
|
|
```
|
|||
|
|
|
|||
|
|
### 4. 取消预约扣费
|
|||
|
|
|
|||
|
|
```java
|
|||
|
|
// 创建取消请求
|
|||
|
|
SubscriptionCancelRequest cancelRequest = new SubscriptionCancelRequest();
|
|||
|
|
cancelRequest.setSubscriptionId(subscriptionId);
|
|||
|
|
cancelRequest.setCancelReason("用户主动取消");
|
|||
|
|
|
|||
|
|
// 取消预约扣费
|
|||
|
|
SubscriptionCancelResult cancelResult = subscriptionService.cancelSubscription(cancelRequest);
|
|||
|
|
System.out.println("取消结果: " + cancelResult.getStatus());
|
|||
|
|
```
|
|||
|
|
|
|||
|
|
### 5. 立即扣费
|
|||
|
|
|
|||
|
|
```java
|
|||
|
|
// 创建立即扣费请求
|
|||
|
|
SubscriptionInstantBillingRequest instantRequest = new SubscriptionInstantBillingRequest();
|
|||
|
|
instantRequest.setOutTradeNo("instant_" + System.currentTimeMillis());
|
|||
|
|
instantRequest.setOpenid("用户的openid");
|
|||
|
|
instantRequest.setDescription("补扣上月会员费");
|
|||
|
|
|
|||
|
|
// 设置扣费金额
|
|||
|
|
SubscriptionAmount instantAmount = new SubscriptionAmount();
|
|||
|
|
instantAmount.setTotal(3000); // 30元
|
|||
|
|
instantAmount.setCurrency("CNY");
|
|||
|
|
instantRequest.setAmount(instantAmount);
|
|||
|
|
|
|||
|
|
// 执行立即扣费
|
|||
|
|
SubscriptionInstantBillingResult instantResult = subscriptionService.instantBilling(instantRequest);
|
|||
|
|
System.out.println("扣费结果: " + instantResult.getTradeState());
|
|||
|
|
```
|
|||
|
|
|
|||
|
|
### 6. 查询扣费记录
|
|||
|
|
|
|||
|
|
```java
|
|||
|
|
// 创建查询请求
|
|||
|
|
SubscriptionTransactionQueryRequest queryRequest = new SubscriptionTransactionQueryRequest();
|
|||
|
|
queryRequest.setOpenid("用户的openid");
|
|||
|
|
queryRequest.setBeginTime("2024-08-01T00:00:00+08:00");
|
|||
|
|
queryRequest.setEndTime("2024-08-31T23:59:59+08:00");
|
|||
|
|
queryRequest.setLimit(20);
|
|||
|
|
queryRequest.setOffset(0);
|
|||
|
|
|
|||
|
|
// 查询扣费记录
|
|||
|
|
SubscriptionTransactionQueryResult transactionResult = subscriptionService.queryTransactions(queryRequest);
|
|||
|
|
System.out.println("总记录数: " + transactionResult.getTotalCount());
|
|||
|
|
for (SubscriptionTransactionQueryResult.SubscriptionTransaction transaction : transactionResult.getData()) {
|
|||
|
|
System.out.println("订单号: " + transaction.getOutTradeNo() + ", 状态: " + transaction.getTradeState());
|
|||
|
|
}
|
|||
|
|
```
|
|||
|
|
|
|||
|
|
## 扣费计划类型
|
|||
|
|
|
|||
|
|
- `MONTHLY`:按月扣费
|
|||
|
|
- `WEEKLY`:按周扣费
|
|||
|
|
- `DAILY`:按日扣费
|
|||
|
|
- `YEARLY`:按年扣费
|
|||
|
|
|
|||
|
|
## 预约状态说明
|
|||
|
|
|
|||
|
|
- `SCHEDULED`:已预约
|
|||
|
|
- `CANCELLED`:已取消
|
|||
|
|
- `EXECUTED`:已执行
|
|||
|
|
- `FAILED`:执行失败
|
|||
|
|
|
|||
|
|
## 交易状态说明
|
|||
|
|
|
|||
|
|
- `SUCCESS`:支付成功
|
|||
|
|
- `REFUND`:转入退款
|
|||
|
|
- `NOTPAY`:未支付
|
|||
|
|
- `CLOSED`:已关闭
|
|||
|
|
- `REVOKED`:已撤销(刷卡支付)
|
|||
|
|
- `USERPAYING`:用户支付中
|
|||
|
|
- `PAYERROR`:支付失败
|
|||
|
|
|
|||
|
|
## 注意事项
|
|||
|
|
|
|||
|
|
1. **用户授权**:使用预约扣费功能前,需要用户在微信内完成签约授权
|
|||
|
|
2. **商户资质**:需要具备相应的业务资质才能开通此功能
|
|||
|
|
3. **金额限制**:扣费金额需要在签约模板规定的范围内
|
|||
|
|
4. **频率限制**:API调用有频率限制,请注意控制调用频次
|
|||
|
|
5. **异常处理**:建议对所有API调用进行异常处理
|
|||
|
|
|
|||
|
|
## 相关文档
|
|||
|
|
|
|||
|
|
- [微信支付预约扣费API文档](https://pay.weixin.qq.com/doc/v3/merchant/4012161105)
|
|||
|
|
- [微信支付开发指南](https://pay.weixin.qq.com/wiki/doc/apiv3/index.shtml)
|
|||
|
|
|
|||
|
|
## 示例完整代码
|
|||
|
|
|
|||
|
|
```java
|
|||
|
|
import com.github.binarywang.wxpay.service.SubscriptionBillingService;
|
|||
|
|
import com.github.binarywang.wxpay.bean.subscriptionbilling.*;
|
|||
|
|
|
|||
|
|
public class SubscriptionBillingExample {
|
|||
|
|
|
|||
|
|
private SubscriptionBillingService subscriptionService;
|
|||
|
|
|
|||
|
|
public void example() throws Exception {
|
|||
|
|
// 1. 创建预约扣费
|
|||
|
|
SubscriptionScheduleRequest request = new SubscriptionScheduleRequest();
|
|||
|
|
request.setOutTradeNo("subscription_" + System.currentTimeMillis());
|
|||
|
|
request.setOpenid("用户openid");
|
|||
|
|
request.setDescription("VIP会员续费");
|
|||
|
|
request.setScheduleTime("2024-09-01T10:00:00+08:00");
|
|||
|
|
|
|||
|
|
SubscriptionAmount amount = new SubscriptionAmount();
|
|||
|
|
amount.setTotal(3000);
|
|||
|
|
amount.setCurrency("CNY");
|
|||
|
|
request.setAmount(amount);
|
|||
|
|
|
|||
|
|
BillingPlan plan = new BillingPlan();
|
|||
|
|
plan.setPlanType("MONTHLY");
|
|||
|
|
plan.setPeriod(1);
|
|||
|
|
plan.setTotalCount(12);
|
|||
|
|
request.setBillingPlan(plan);
|
|||
|
|
|
|||
|
|
SubscriptionScheduleResult result = subscriptionService.scheduleSubscription(request);
|
|||
|
|
|
|||
|
|
// 2. 查询预约状态
|
|||
|
|
SubscriptionQueryResult query = subscriptionService.querySubscription(result.getSubscriptionId());
|
|||
|
|
|
|||
|
|
// 3. 如需取消
|
|||
|
|
if ("SCHEDULED".equals(query.getStatus())) {
|
|||
|
|
SubscriptionCancelRequest cancelReq = new SubscriptionCancelRequest();
|
|||
|
|
cancelReq.setSubscriptionId(result.getSubscriptionId());
|
|||
|
|
cancelReq.setCancelReason("用户取消");
|
|||
|
|
|
|||
|
|
SubscriptionCancelResult cancelResult = subscriptionService.cancelSubscription(cancelReq);
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
```
|