微信公众号给微信用户转账的Java实现

微信公众号给微信用户转账的Java实现,第1张

没有直接转账功能,可以通过红包的方式进行奖励。

微信公众号有红包业务和接口。

1、普通提现的银行(1-3天内到账)有:工行、农行、招行、建行、中行、交行、广发银行、光大银行、兴业银行、平安银行、浦发银行、民生银行、中信银行、上海银行、华夏银行、北京银行、宁波银行。

2、快速提现(在支持时间段里2-4小时内到账)支持的银行:工行、农行、建行、招行、中行、交行、浦发。

支持时间段:工行、招行、交行、浦发:07:00 — 22:00(节假日:9:00-22:00);建行、中行:07:00 —18:00(节假日:9:00-18:00);农行:7:00-20:00(节假日:9:00-20:00)。

微信支付系列文章

微信支付-java后端实现

微信支付-vue 前端实现

java demo: 下载地址文章底部

技术栈

Spring boot

java

XML (微信在http协议中数据传输方案)

MD5 签名

微信支付术语

openid (OpenID是公众号一对一对应用户身份的标识)

app_id (公众号id,登录微信公众号–开发–基本配置中获得;)

key (收款商户后台进行配置,登录微信商户平台–账户中心–API安全-设置秘钥,设置32位key值;)

mch_id (收款商家商户号;)

certPath (API证书, 登录微信商户平台–账户中心-API安全-下载证书)

后端流程

服务端需要的核心操作, 总共分为以下几步:

统一下单

前端调起微信支付必要参数 (需加密)

订单结果主动通知 (回调接口)

查询订单结果

结束订单支付接口(关闭订单,支付订单关闭)

代码

微信总共支持多种语言的sdk, 在官网可以下载例子, java程序也可以引入微信支付的sdk包, 但是github上的sdk已经很久没有更新了, 最好的选择, 也是我的选择, 在官网上下载sdk项目, 将其中所有java类copy到自己的项目中.

官网sdk下载目录

链接: 商户平台首页

#### 根据微信sdk生成配置类 WXPayConfig

创建IWxPayConfig.class, 继承sdk WXPayConfig.class, 实现sdk中部分抽象方法, 读取本地证书, 加载到配置类中.

package core.com.chidori.wxpay

import core.com.wxpay.IWXPayDomain

import core.com.wxpay.WXPayConfig

import core.com.wxpay.WXPayConstants

import org.springframework.beans.factory.annotation.Value

import org.springframework.stereotype.Service

import java.io.ByteArrayInputStream

import java.io.File

import java.io.FileInputStream

import java.io.InputStream

@Service

public class IWxPayConfig extends WXPayConfig { // 继承sdk WXPayConfig 实现sdk中部分抽象方法

private byte[] certData

@Value("${vendor.wx.config.app_id}")

private String app_id

@Value("${vendor.wx.pay.key}")

private String wx_pay_key

@Value("${vendor.wx.pay.mch_id}")

private String wx_pay_mch_id

public IWxPayConfig() throws Exception { // 构造方法读取证书, 通过getCertStream 可以使sdk获取到证书

String certPath = "/data/config/chidori/apiclient_cert.p12"

File file = new File(certPath)

InputStream certStream = new FileInputStream(file)

this.certData = new byte[(int) file.length()]

certStream.read(this.certData)

certStream.close()

}

@Override

public String getAppID() {

return app_id

}

@Override

public String getMchID() {

return wx_pay_mch_id

}

@Override

public String getKey() {

return wx_pay_key

}

@Override

public InputStream getCertStream() {

return new ByteArrayInputStream(this.certData)

}

@Override

public IWXPayDomain getWXPayDomain() { // 这个方法需要这样实现, 否则无法正常初始化WXPay

IWXPayDomain iwxPayDomain = new IWXPayDomain() {

@Override

public void report(String domain, long elapsedTimeMillis, Exception ex) {

}

@Override

public DomainInfo getDomain(WXPayConfig config) {

return new IWXPayDomain.DomainInfo(WXPayConstants.DOMAIN_API, true)

}

}

return iwxPayDomain

}

}

发起统一下单 AND 前端调起微信支付必要参数

// 发起微信支付

WXPay wxpay = null

Map result = new HashMap>()

try {

// ******************************************

//

// 统一下单

//

// ******************************************

wxpay = new WXPay(iWxPayConfig)// *** 注入自己实现的微信配置类, 创建WXPay核心类, WXPay 包括统一下单接口

Map data = new HashMap ()

data.put("body", "订单详情")

data.put("out_trade_no", transOrder.getGlobalOrderId())// 订单唯一编号, 不允许重复

data.put("total_fee", String.valueOf(transOrder.getOrderAmount().multiply(new BigDecimal(100)).intValue()))// 订单金额, 单位分

data.put("spbill_create_ip", "192.168.31.166")// 下单ip

data.put("openid", openId)// 微信公众号统一标示openid

data.put("notify_url", "http://wxlj.oopmind.com/payCallback")// 订单结果通知, 微信主动回调此接口

data.put("trade_type", "JSAPI")// 固定填写

logger.info("发起微信支付下单接口, request={}", data)

Map response = wxpay.unifiedOrder(data)// 微信sdk集成方法, 统一下单接口unifiedOrder, 此处请求 MD5加密 加密方式

logger.info("微信支付下单成功, 返回值 response={}", response)

String returnCode = response.get("return_code")

if (!SUCCESS.equals(returnCode)) {

return null

}

String resultCode = response.get("result_code")

if (!SUCCESS.equals(resultCode)) {

return null

}

String prepay_id = response.get("prepay_id")

if (prepay_id == null) {

return null

}

// ******************************************

//

// 前端调起微信支付必要参数

//

// ******************************************

String packages = "prepay_id=" + prepay_id

Map wxPayMap = new HashMap ()

wxPayMap.put("appId", iWxPayConfig.getAppID())

wxPayMap.put("timeStamp", String.valueOf(Utility.getCurrentTimeStamp()))

wxPayMap.put("nonceStr", Utility.generateUUID())

wxPayMap.put("package", packages)

wxPayMap.put("signType", "MD5")

// 加密串中包括 appId timeStamp nonceStr package signType 5个参数, 通过sdk WXPayUtil类加密, 注意, 此处使用 MD5加密 方式

String sign = WXPayUtil.generateSignature(wxPayMap, iWxPayConfig.getKey())

// ******************************************

//

// 返回给前端调起微信支付的必要参数

//

// ******************************************

result.put("prepay_id", prepay_id)

result.put("sign", sign)

result.putAll(wxPayMap)

return result

} catch (Exception e) {

}

回调结果处理

核心是支付订单回调时, 需校验加密签名是否匹配, 防止出现模拟成功通知

@RequestMapping(value = "/payCallback", method = RequestMethod.POST)

public String payCallback(HttpServletRequest request, HttpServletResponse response) {

logger.info("进入微信支付异步通知")

String resXml=""

try{

//

InputStream is = request.getInputStream()

//将InputStream转换成String

BufferedReader reader = new BufferedReader(new InputStreamReader(is))

StringBuilder sb = new StringBuilder()

String line = null

try {

while ((line = reader.readLine()) != null) {

sb.append(line + " ")

}

} catch (IOException e) {

e.printStackTrace()

} finally {

try {

is.close()

} catch (IOException e) {

e.printStackTrace()

}

}

resXml=sb.toString()

logger.info("微信支付异步通知请求包: {}", resXml)

return wxTicketService.payBack(resXml)

}catch (Exception e){

logger.error("微信支付回调通知失败",e)

String result = " "

return result

}

}

@Override

public String payBack(String notifyData) {

logger.info("payBack() start, notifyData={}", notifyData)

String xmlBack=""

Map notifyMap = null

try {

WXPay wxpay = new WXPay(iWxPayConfig)

notifyMap = WXPayUtil.xmlToMap(notifyData)// 转换成map

if (wxpay.isPayResultNotifySignatureValid(notifyMap)) {

// 签名正确

// 进行处理。

// 注意特殊情况:订单已经退款,但收到了支付结果成功的通知,不应把商户侧订单状态从退款改成支付成功

String return_code = notifyMap.get("return_code")//状态

String out_trade_no = notifyMap.get("out_trade_no")//订单号

if (out_trade_no == null) {

logger.info("微信支付回调失败订单号: {}", notifyMap)

xmlBack = " "

return xmlBack

}

// 业务逻辑处理 ****************************

logger.info("微信支付回调成功订单号: {}", notifyMap)

xmlBack = " "

return xmlBack

} else {

logger.error("微信支付回调通知签名错误")

xmlBack = " "

return xmlBack

}

} catch (Exception e) {

logger.error("微信支付回调通知失败",e)

xmlBack = " "

}

return xmlBack

}

统一下单的签名和后续前端拉取微信支付的签名需要统一, 也就是都采用MD5加密, 如果2者不同, 会导致前端拉取微信支付fail, 这是一个巨大的坑, 因为这个原因调试了好久, 微信在文档里没有明确标出统一下单的签名校验方式 需要和前端拉取微信支付的签名校验保持一致.

微信sdk里的源码需要针对这个问题调整一下, 调整如下:

WXPay类需要修改下加密判断,在WXPay构造方法中,调整如下

public WXPay(final WXPayConfig config, final String notifyUrl, final boolean autoReport, final boolean useSandbox) throws Exception {

this.config = config

this.notifyUrl = notifyUrl

this.autoReport = autoReport

this.useSandbox = useSandbox

if (useSandbox) {

this.signType = SignType.MD5// 沙箱环境

}

else {

this.signType = SignType.MD5 // 将这里的加密方式修改为SignType.MD5, 保持跟前端吊起微信加密方式保持一致

}

this.wxPayRequest = new WXPayRequest(config)

}

结束语

做完以后, 微信支付的后端逻辑还是很清晰的, 但是在开发过程中很煎熬, 不清楚每个专业术语在微信哪里配置, 加密方式乱的很

1、找到微信支付公众号兑换免费提现券

这个是微信认可的最正规的一个提现步骤,大家需要来到微信支付公众号,点击右下角的更多,进入微信支付有优惠之后,就可以获取自己使用微信支付获取的所有积分,使用十个积分,就可以兑免费提现额度券,这样每天可以领一次。

2、开通微信商家使用积分免费提现

如果各位朋友们拥有自己的一个小店,那么开通了微信商家收款码之后,只要关注微信收账助手公众号,点击下面的收款有礼,就可以将自己通过微信经营收款获得的积分来兑换成免费提现额度,兑换比例是1:1。

3、和其他用户进行交易

有一些朋友可能急需微信转账一定的数目,而各位朋友如果可以接受现金的话,就可以通过微信将想要提现的数目以转账的形式给自己的朋友,每次转账的数目都是有限制的。

这样朋友直接将现金交给你,就可以实现无手续费的提现,只不过要去银行存一下,基本上就相当于帮助朋友跑腿去银行存钱,省下了手续费。

4、将钱放到零钱通进行提现

零钱通相当于微信的一款理财类产品,但是完全没有任何风险,相应的可能存进去报酬也是比较少的,大家可以将微信里通过交易获得的零钱转入零钱通里面,需要放个七天到十天左右。

进入零钱通之后直接点击转出,这里的转出纸的,直接就是转出到银行卡,转出的数目会随着时间的增加而增加,所以对于大额数目的用户来说,放进零钱通还是比较不错的。

5、等待各大节日微信支付活动

就比如说今年端午节的时候,就出现了一个类似于雷霆战机的吃粽子小游戏,玩家们在游戏中吃到的粽子越多,就可以获取更大额度的微信免费提现券,今年的中秋节也出了一个吃月饼的活动。

可以及时在这些节日中留意一下微信支付是否有相应的活动,这些微信支付券看起来每笔为用户上的钱并不是很多,但是长期加起来就是一个大数目了。


欢迎分享,转载请注明来源:夏雨云

原文地址:https://www.xiayuyun.com/zonghe/560845.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-07-01
下一篇2023-07-01

发表评论

登录后才能评论

评论列表(0条)

    保存