微信刷卡支付java,微信刷卡支付一天限额多少

本文目录一览:

用Java怎么实现微信支付

具体方法步骤:

一、准备阶段:已认证微信号,且通过微信支付认证,这个可以看微信文档,很详细,这里就不再重复。

二、配置授权目录,官方推荐使用https类型的url,不知道http能不能行,个人也推荐使用https的保证不会错。

配置授权域名

三、微信支付二次开发所需要的参数:

APP_ID,APP_KEY,PARTNER,PARTNER_KEY(AppSecret)

APP_ID和PARTNER_KEY(AppSecret)

PARTNER

APP_KEY(自行设置32位字符)

四、具体编程

1、通过页面跳转到确认支付页面,其中的redirect_uri必须是配置授权目录下的。

2、获取到openid,再经服务器向微信请求获取prepay_id,封装字段并进行签名后通过jsapi调起微信支付

3、测试结果

         

微信支付Java如何判断回调

微信支付Java判断回调方法:

微信支付完成支付调用的时候,在传入的参数中有一个是执行支付完成之后结果回调的参数,这个回调函数就是微信调用这个接口来将支付成功的结果。

微信支付后端篇

微信支付系列文章

微信支付-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”, “”); // 订单结果通知, 微信主动回调此接口

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);

}

结束语

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

怎么用java调用微信支付接口

java调用微信支付接口方法:

RequestHandler requestHandler = new RequestHandler(super.getRequest(),super.getResponse());

//获取token //两小时内有效,两小时后重新获取

Token = requestHandler.GetToken();

//更新token 到应用中

requestHandler.getTokenReal();

System.out.println(“微信支付获取token=======================:” +Token);

//requestHandler 初始化

requestHandler.init();

requestHandler.init(appid,appsecret, appkey,partnerkey, key);

// ——————————–本地系统生成订单————————————-

// 设置package订单参数

SortedMapString, String packageParams = new TreeMapString, String();

packageParams.put(“bank_type”, “WX”); // 支付类型

packageParams.put(“body”, “xxxx”); // 商品描述

packageParams.put(“fee_type”, “1”); // 银行币种

packageParams.put(“input_charset”, “UTF-8”); // 字符集

packageParams.put(“notify_url”, “”); // 通知地址 这里的通知地址使用外网地址测试,注意80端口是否打开。

packageParams.put(“out_trade_no”, no); // 商户订单号

packageParams.put(“partner”, partenerid); // 设置商户号

packageParams.put(“spbill_create_ip”, super.getRequest().getRemoteHost()); // 订单生成的机器IP,指用户浏览器端IP

packageParams.put(“total_fee”, String.valueOf(rstotal)); // 商品总金额,以分为单位

// 设置支付参数

SortedMapString, String signParams = new TreeMapString, String();

signParams.put(“appid”, appid);

signParams.put(“noncestr”, noncestr);

signParams.put(“traceid”, PropertiesUtils.getOrderNO());

signParams.put(“timestamp”, timestamp);

signParams.put(“package”, packageValue);

signParams.put(“appkey”, this.appkey);

// 生成支付签名,要采用URLENCODER的原始值进行SHA1算法!

String sign =””;

try {

sign = Sha1Util.createSHA1Sign(signParams);

} catch (Exception e) {

e.printStackTrace();

}

// 增加非参与签名的额外参数

signParams.put(“sign_method”, “sha1”);

signParams.put(“app_signature”, sign);

// api支付拼包结束————————————

//获取prepayid

String prepayid = requestHandler.sendPrepay(signParams);

System.out.println(“prepayid :” + prepayid);

// ——————————–生成完成———————————————

//生成预付快订单完成,返回给android,ios 掉起微信所需要的参数。

SortedMapString, String payParams = new TreeMapString, String();

payParams.put(“appid”, appid);

payParams.put(“noncestr”, noncestr);

payParams.put(“package”, “Sign=WXPay”);

payParams.put(“partnerid”, partenerid);

payParams.put(“prepayid”, prepayid);

payParams.put(“appkey”, this.appkey);

//这里除1000 是因为参数长度限制。

int time = (int) (System.currentTimeMillis() / 1000);

payParams.put(“timestamp”,String.valueOf(time));

System.out.println(“timestamp:” + time);

//签名

String paysign =””;

try {

paysign = Sha1Util.createSHA1Sign(payParams);

} catch (Exception e) {

e.printStackTrace();

}

payParams.put(“sign”, paysign);

//拼json 数据返回给客户端

BasicDBObject backObject = new BasicDBObject();

backObject.put(“appid”, appid);

backObject.put(“noncestr”, payParams.get(“noncestr”));

backObject.put(“package”, “Sign=WXPay”);

backObject.put(“partnerid”, payParams.get(“partnerid”));

backObject.put(“prepayid”, payParams.get(“prepayid”));

backObject.put(“appkey”, this.appkey);

backObject.put(“timestamp”,payParams.get(“timestamp”));

backObject.put(“sign”,payParams.get(“sign”));

String backstr = dataObject.toString();

System.out.println(“backstr:” + backstr);

return backstr;

====================到此为止,预付款订单已生成,并且已返回客户端====================

//坐等微信服务器通知,通知的地址就是生成预付款订单的notify_url

ResponseHandler resHandler = new ResponseHandler(request, response);

resHandler.setKey(partnerkey);

//创建请求对象

//RequestHandler queryReq = new RequestHandler(request, response);

//queryReq.init();

if (resHandler.isTenpaySign() == true) {

//商户订单号

String out_trade_no = resHandler.getParameter(“out_trade_no”);

System.out.println(“out_trade_no:” + out_trade_no);

//财付通订单号

String transaction_id = resHandler.getParameter(“transaction_id”);

System.out.println(“transaction_id:” + transaction_id);

//金额,以分为单位

String total_fee = resHandler.getParameter(“total_fee”);

//如果有使用折扣券,discount有值,total_fee+discount=原请求的total_fee

String discount = resHandler.getParameter(“discount”);

//支付结果

String trade_state = resHandler.getParameter(“trade_state”);

//判断签名及结果

if (“0”.equals(trade_state)) {

//——————————

//即时到账处理业务开始

//——————————

System.out.println(“—————-业务逻辑执行—————–“);

//——请根据您的业务逻辑来编写程序(以上代码仅作参考)——

System.out.println(“—————-业务逻辑执行完毕—————–“);

System.out.println(“success”); // 请不要修改或删除

System.out.println(“即时到账支付成功”);

//给财付通系统发送成功信息,财付通系统收到此结果后不再进行后续通知

resHandler.sendToCFT(“success”);

//给微信服务器返回success 否则30分钟通知8次

return “success”;

}else{

System.out.println(“通知签名验证失败”);

resHandler.sendToCFT(“fail”);

response.setCharacterEncoding(“utf-8”);

}

}else {

System.out.println(“fail -Md5 failed”);

原创文章,作者:小蓝,如若转载,请注明出处:https://www.506064.com/n/246433.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
小蓝小蓝
上一篇 2024-12-12 13:15
下一篇 2024-12-12 13:15

相关推荐

  • Java JsonPath 效率优化指南

    本篇文章将深入探讨Java JsonPath的效率问题,并提供一些优化方案。 一、JsonPath 简介 JsonPath是一个可用于从JSON数据中获取信息的库。它提供了一种DS…

    编程 2025-04-29
  • java client.getacsresponse 编译报错解决方法

    java client.getacsresponse 编译报错是Java编程过程中常见的错误,常见的原因是代码的语法错误、类库依赖问题和编译环境的配置问题。下面将从多个方面进行分析…

    编程 2025-04-29
  • Java Bean加载过程

    Java Bean加载过程涉及到类加载器、反射机制和Java虚拟机的执行过程。在本文中,将从这三个方面详细阐述Java Bean加载的过程。 一、类加载器 类加载器是Java虚拟机…

    编程 2025-04-29
  • Java腾讯云音视频对接

    本文旨在从多个方面详细阐述Java腾讯云音视频对接,提供完整的代码示例。 一、腾讯云音视频介绍 腾讯云音视频服务(Cloud Tencent Real-Time Communica…

    编程 2025-04-29
  • Java Milvus SearchParam withoutFields用法介绍

    本文将详细介绍Java Milvus SearchParam withoutFields的相关知识和用法。 一、什么是Java Milvus SearchParam without…

    编程 2025-04-29
  • Java 8中某一周的周一

    Java 8是Java语言中的一个版本,于2014年3月18日发布。本文将从多个方面对Java 8中某一周的周一进行详细的阐述。 一、数组处理 Java 8新特性之一是Stream…

    编程 2025-04-29
  • Java判断字符串是否存在多个

    本文将从以下几个方面详细阐述如何使用Java判断一个字符串中是否存在多个指定字符: 一、字符串遍历 字符串是Java编程中非常重要的一种数据类型。要判断字符串中是否存在多个指定字符…

    编程 2025-04-29
  • VSCode为什么无法运行Java

    解答:VSCode无法运行Java是因为默认情况下,VSCode并没有集成Java运行环境,需要手动添加Java运行环境或安装相关插件才能实现Java代码的编写、调试和运行。 一、…

    编程 2025-04-29
  • Java任务下发回滚系统的设计与实现

    本文将介绍一个Java任务下发回滚系统的设计与实现。该系统可以用于执行复杂的任务,包括可回滚的任务,及时恢复任务失败前的状态。系统使用Java语言进行开发,可以支持多种类型的任务。…

    编程 2025-04-29
  • Java 8 Group By 会影响排序吗?

    是的,Java 8中的Group By会对排序产生影响。本文将从多个方面探讨Group By对排序的影响。 一、Group By的概述 Group By是SQL中的一种常见操作,它…

    编程 2025-04-29

发表回复

登录后才能评论