1.准备资料
(1)服务商id
(2)服务商Appid
(3)程序商户的appid
(4)小程序商户号ID
(5)服务商的APIKey
(5)证书
(6)证书密码
以上资料准备完善 先去看看微信支付的官方的API 根据你的需要找到具体的
https://pay.weixin.qq.com/wiki/doc/api/index.html
我用到的是JSAPI支付并且是服务商版下面我的实例均是服务商版本
从官方的文档里面我们找到一些必要的信息
官方提供的API列表
1.统一下单 https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1
2.查询订单 https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_2
3.关闭订单 https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_3
4.申请退款 https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_4
5.查询退款 https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_5
以上为一些在正式开始前所需要了解的一些基本内容 下面进入正题
2.代码部分
引入依赖
<dependency> <groupId>com.github.wxpay</groupId> <artifactId>wxpay-sdk</artifactId> <version>0.0.3</version> </dependency>
在application.yml中配置一些基本的信息
pay:
#用一下单API地址
mchPayUrl: https://api.mch.weixin.qq.com/pay/unifiedorder
#服务商APPID
appid: --
#服务商ID
mch_id: --
#交易类型
trade_type: JSAPI
#小程序APPID
sub_appid: --
#小程序商户号
sub_mch_id: --
#异步回调通知地址
notify_url: http://192.168.1.34:8080/searchNotifyPay
#证书的密钥
serverPayKey: --
#证书的路径
certificate: D://DevelopmentTool//apiclient_cert.p12
#证书的密码
certificatePassword: --
#用一下单API地址
seachOrderURL: https://api.mch.weixin.qq.com/pay/orderquery
创建微信支付参数的compoent主要从application.yml中把参数绑定到实体类中
package com.landimc.compoent.pay; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; @Component public class WxPayCompoent { /*** * 服务商统一下单 url **/ @Value("${pay.mchPayUrl}") public String mchPayUrl = ""; /*** * 服务商id **/ @Value("${pay.mch_id}") public String mch_id = ""; /*** *服务商Appid **/ @Value("${pay.appid}") public String appid = ""; /** * 交易类型 */ @Value("${pay.trade_type}") public String trade_type = "JSAPI"; /** * 小程序商户的appid */ @Value("${pay.sub_appid}") public String sub_appid = ""; /** * 小程序商户号ID */ @Value("${pay.sub_mch_id}") public String sub_mch_id = ""; /** * 异步回调通知地址 */ @Value("${pay.notify_url}") public String notify_url = ""; /** * 服务商的APIKey */ @Value("${pay.serverPayKey}") public String serverPayKey = ""; /** * 证书存放路径 */ @Value("${pay.certificate}") public String certificate = ""; /** * 证书密码 (默认商户号) */ @Value("${pay.certificatePassword}") public String certificatePassword = ""; /** * 查询订单的地址 */ @Value("${pay.seachOrderURL}") public String seachOrderURL = ""; }
下面是一些用到的辅助工具类
1.生成签名
package com.landimc.wxpay.utils; import com.github.wxpay.sdk.WXPayConstants; import com.github.wxpay.sdk.WXPayConstants.SignType; import com.landimc.compoent.pay.WxPayCompoent; import com.landimc.tools.MD5; import com.landimc.tools.SysConfig; import com.landimc.wxpay.model.WxPayConfig; import org.apache.commons.codec.digest.DigestUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.w3c.dom.Node; import org.w3c.dom.NodeList; import javax.annotation.Resource; import javax.crypto.Mac; import javax.crypto.spec.SecretKeySpec; import javax.xml.parsers.DocumentBuilder; import javax.xml.transform.OutputKeys; import javax.xml.transform.Transformer; import javax.xml.transform.TransformerFactory; import javax.xml.transform.dom.DOMSource; import javax.xml.transform.stream.StreamResult; import java.io.ByteArrayInputStream; import java.io.InputStream; import java.io.StringWriter; import java.io.UnsupportedEncodingException; import java.security.MessageDigest; import java.security.SecureRandom; import java.util.*; public class WXPayUtil { private static final String SYMBOLS = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; private static final Random RANDOM = new SecureRandom(); @Resource private WxPayCompoent wxPayCompoent; /** * XML格式字符串转换为Map * * @param strXML XML字符串 * @return XML数据转换后的Map * @throws Exception */ public static Map<String, String> xmlToMap(String strXML) throws Exception { try { Map<String, String> data = new HashMap<String, String>(); DocumentBuilder documentBuilder = WXPayXmlUtil.newDocumentBuilder(); InputStream stream = new ByteArrayInputStream(strXML.getBytes("UTF-8")); org.w3c.dom.Document doc = documentBuilder.parse(stream); doc.getDocumentElement().normalize(); NodeList nodeList = doc.getDocumentElement().getChildNodes(); for (int idx = 0; idx < nodeList.getLength(); ++idx) { Node node = nodeList.item(idx); if (node.getNodeType() == Node.ELEMENT_NODE) { org.w3c.dom.Element element = (org.w3c.dom.Element) node; data.put(element.getNodeName(), element.getTextContent()); } } try { stream.close(); } catch (Exception ex) { // do nothing } return data; } catch (Exception ex) { WXPayUtil.getLogger().warn("Invalid XML, can not convert to map. Error message: {}. XML content: {}", ex.getMessage(), strXML); throw ex; } } /** * 将Map转换为XML格式的字符串 * * @param data Map类型数据 * @return XML格式的字符串 * @throws Exception */ public static String mapToXml(Map<String, String> data) throws Exception { org.w3c.dom.Document document = WXPayXmlUtil.newDocument(); org.w3c.dom.Element root = document.createElement("xml"); document.appendChild(root); for (String key : data.keySet()) { String value = data.get(key); if (value == null) { value = ""; } value = value.trim(); org.w3c.dom.Element filed = document.createElement(key); filed.appendChild(document.createTextNode(value)); root.appendChild(filed); } TransformerFactory tf = TransformerFactory.newInstance(); Transformer transformer = tf.newTransformer(); DOMSource source = new DOMSource(document); transformer.setOutputProperty(OutputKeys.ENCODING, "UTF-8"); transformer.setOutputProperty(OutputKeys.INDENT, "yes"); StringWriter writer = new StringWriter(); StreamResult result = new StreamResult(writer); transformer.transform(source, result); String output = writer.getBuffer().toString(); //.replaceAll("\n|\r", ""); try { writer.close(); } catch (Exception ex) { } return output; } /** * 生成带有 sign 的 XML 格式字符串 * * @param data Map类型数据 * @param key API密钥 * @return 含有sign字段的XML */ public static String generateSignedXml(final Map<String, String> data, String key) throws Exception { return generateSignedXml(data, key, SignType.MD5); } /** * 生成带有 sign 的 XML 格式字符串 * * @param data Map类型数据 * @param key API密钥 * @param signType 签名类型 * @return 含有sign字段的XML */ public static String generateSignedXml(final Map<String, String> data, String key, SignType signType) throws Exception { String sign = generateSignature(data, key, signType); data.put(WXPayConstants.FIELD_SIGN, sign); return mapToXml(data); } /** * 判断签名是否正确 * * @param xmlStr XML格式数据 * @param key API密钥 * @return 签名是否正确 * @throws Exception */ public static boolean isSignatureValid(String xmlStr, String key) throws Exception { Map<String, String> data = xmlToMap(xmlStr); if (!data.containsKey(WXPayConstants.FIELD_SIGN)) { return false; } String sign = data.get(WXPayConstants.FIELD_SIGN); return generateSignature(data, key).equals(sign); } /** * 判断签名是否正确,必须包含sign字段,否则返回false。使用MD5签名。 * * @param data Map类型数据 * @param key API密钥 * @return 签名是否正确 * @throws Exception */ public static boolean isSignatureValid(Map<String, String> data, String key) throws Exception { return isSignatureValid(data, key, SignType.MD5); } /** * 判断签名是否正确,必须包含sign字段,否则返回false。 * * @param data Map类型数据 * @param key API密钥 * @param signType 签名方式 * @return 签名是否正确 * @throws Exception */ public static boolean isSignatureValid(Map<String, String> data, String key, SignType signType) throws Exception { if (!data.containsKey(WXPayConstants.FIELD_SIGN)) { return false; } String sign = data.get(WXPayConstants.FIELD_SIGN); return generateSignature(data, key, signType).equals(sign); } /** * 生成签名 * * @param data 待签名数据 * @param key API密钥 * @return 签名 */ public static String generateSignature(final Map<String, String> data, String key) throws Exception { return generateSignature(data, key, SignType.MD5); } /** * 生成签名. 注意,若含有sign_type字段,必须和signType参数保持一致。 * * @param data 待签名数据 * @param key API密钥 * @param signType 签名方式 * @return 签名 */ public static String generateSignature(final Map<String, String> data, String key, SignType signType) throws Exception { Set<String> keySet = data.keySet(); String[] keyArray = keySet.toArray(new String[keySet.size()]); Arrays.sort(keyArray); StringBuilder sb = new StringBuilder(); for (String k : keyArray) { if (k.equals(WXPayConstants.FIELD_SIGN)) { continue; } if (data.get(k).trim().length() > 0) // 参数值为空,则不参与签名 sb.append(k).append("=").append(data.get(k).trim()).append("&"); } sb.append("key=").append(key); if (SignType.MD5.equals(signType)) { return MD5(sb.toString()).toUpperCase(); } else if (SignType.HMACSHA256.equals(signType)) { return HMACSHA256(sb.toString(), key); } else { throw new Exception(String.format("Invalid sign_type: %s", signType)); } } /** * 获取随机字符串 Nonce Str * * @return String 随机字符串 */ public static String generateNonceStr() { char[] nonceChars = new char[32]; for (int index = 0; index < nonceChars.length; ++index) { nonceChars[index] = SYMBOLS.charAt(RANDOM.nextInt(SYMBOLS.length())); } return new String(nonceChars); } /** * 生成 MD5 * * @param data 待处理数据 * @return MD5结果 */ public static String MD5(String data) throws Exception { MessageDigest md = MessageDigest.getInstance("MD5"); byte[] array = md.digest(data.getBytes("UTF-8")); StringBuilder sb = new StringBuilder(); for (byte item : array) { sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3)); } return sb.toString().toUpperCase(); } /** * 生成 HMACSHA256 * * @param data 待处理数据 * @param key 密钥 * @return 加密结果 * @throws Exception */ public static String HMACSHA256(String data, String key) throws Exception { Mac sha256_HMAC = Mac.getInstance("HmacSHA256"); SecretKeySpec secret_key = new SecretKeySpec(key.getBytes("UTF-8"), "HmacSHA256"); sha256_HMAC.init(secret_key); byte[] array = sha256_HMAC.doFinal(data.getBytes("UTF-8")); StringBuilder sb = new StringBuilder(); for (byte item : array) { sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3)); } return sb.toString().toUpperCase(); } /** * 日志 * * @return */ public static Logger getLogger() { Logger logger = LoggerFactory.getLogger("wxpay java sdk"); return logger; } /** * 获取当前时间戳,单位秒 * * @return */ public static long getCurrentTimestamp() { return System.currentTimeMillis() / 1000; } /** * 获取当前时间戳,单位毫秒 * * @return */ public static long getCurrentTimestampMs() { return System.currentTimeMillis(); } //生成签名 public static String createSign(SortedMap<String, Object> parameters) { StringBuilder sb = new StringBuilder(); Set es = parameters.entrySet(); for (Object e : es) { Map.Entry entry = (Map.Entry) e; String k = (String) entry.getKey(); Object v = entry.getValue(); if (null != v && !"".equals(v) && !"sign".equals(k) && !"key".equals(k)) { sb.append(k).append("=").append(v).append("&"); } } sb.append("key=").append("1234abcd1234abcd1234abcd1234abcd"); return MD5.MD5Encode(sb.toString(), "UTF-8").toUpperCase(); } /** * 第二次签名专用 * MD5 加密,转为指定类型 * * @param text * @param key * @param input_charset * @return */ public static String sign(String text, String key, String input_charset) { text = text + key; return DigestUtils.md5Hex(getContentBytes(text, input_charset)); } public static byte[] getContentBytes(String content, String charset) { if (charset == null || "".equals(charset)) { return content.getBytes(); } try { return content.getBytes(charset); } catch (UnsupportedEncodingException e) { throw new RuntimeException("MD5签名过程中出现错误,指定的编码集不对,您目前指定的编码集是:" + charset); } } }
2.统一下单
package com.landimc.wxpay; import com.landimc.tools.ClassToMap; import com.landimc.wxpay.model.WxPayModel; import com.landimc.wxpay.utils.HttpClientUtils; import com.landimc.wxpay.utils.WXPayUtil; import com.landimc.wxpay.utils.XMLUtils; import org.apache.commons.codec.digest.DigestUtils; import org.jdom.JDOMException; import java.io.IOException; import java.util.*; /** * 统一下单 * * @author Yang */ public class UnifiedOrderWxPay { /** * 发起支付业务 * * @param * @return * @throws Exception */ public static Map<String, String> toMchPay(String url,SortedMap<String, Object> parameters, String certificate, String pass) throws Exception { String s = HttpClientUtils.httpsRequest(url, "POST", XMLUtils.getRequestXml(parameters), certificate, pass); Map<String, String> returnMap = new HashMap<>(); try { //将返回的XML结果再次转换为Map集合 returnMap = XMLUtils.xmlToMap(s); System.out.println(returnMap); } catch (JDOMException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } return returnMap; } /** * 生成签名 * * @param map * @return */ public static String getSign(Map<String, String> map) { String result = ""; try { List<Map.Entry<String, String>> infoIds = new ArrayList<Map.Entry<String, String>>(map.entrySet()); // 对所有传入参数按照字段名的 ASCII 码从小到大排序(字典序) Collections.sort(infoIds, new Comparator<Map.Entry<String, String>>() { @Override public int compare(Map.Entry<String, String> o1, Map.Entry<String, String> o2) { return (o1.getKey()).toString().compareTo(o2.getKey()); } }); // 构造签名键值对的格式 StringBuilder sb = new StringBuilder(); for (Map.Entry<String, String> item : infoIds) { if (item.getKey() != null || item.getKey() != "") { String key = item.getKey(); String val = item.getValue(); if (!(val == "" || val == null)) { sb.append(key + ":" + val + ":"); } } } result = sb.toString(); //进行MD5加密 result = DigestUtils.md5Hex(result).toUpperCase(); } catch (Exception e) { return null; } return result; } }
2.生成签名 给二次签名使用 这里你们可以自行优化把两个签名方法整合为一个 因为签名的方法是一样的
package com.landimc.wxpay.utils; import com.github.wxpay.sdk.WXPayConstants; import com.github.wxpay.sdk.WXPayConstants.SignType; import com.landimc.compoent.pay.WxPayCompoent; import com.landimc.tools.MD5; import com.landimc.tools.SysConfig; import com.landimc.wxpay.model.WxPayConfig; import org.apache.commons.codec.digest.DigestUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.w3c.dom.Node; import org.w3c.dom.NodeList; import javax.annotation.Resource; import javax.crypto.Mac; import javax.crypto.spec.SecretKeySpec; import javax.xml.parsers.DocumentBuilder; import javax.xml.transform.OutputKeys; import javax.xml.transform.Transformer; import javax.xml.transform.TransformerFactory; import javax.xml.transform.dom.DOMSource; import javax.xml.transform.stream.StreamResult; import java.io.ByteArrayInputStream; import java.io.InputStream; import java.io.StringWriter; import java.io.UnsupportedEncodingException; import java.security.MessageDigest; import java.security.SecureRandom; import java.util.*; public class WXPayUtil { private static final String SYMBOLS = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; private static final Random RANDOM = new SecureRandom(); @Resource private WxPayCompoent wxPayCompoent; /** * XML格式字符串转换为Map * * @param strXML XML字符串 * @return XML数据转换后的Map * @throws Exception */ public static Map<String, String> xmlToMap(String strXML) throws Exception { try { Map<String, String> data = new HashMap<String, String>(); DocumentBuilder documentBuilder = WXPayXmlUtil.newDocumentBuilder(); InputStream stream = new ByteArrayInputStream(strXML.getBytes("UTF-8")); org.w3c.dom.Document doc = documentBuilder.parse(stream); doc.getDocumentElement().normalize(); NodeList nodeList = doc.getDocumentElement().getChildNodes(); for (int idx = 0; idx < nodeList.getLength(); ++idx) { Node node = nodeList.item(idx); if (node.getNodeType() == Node.ELEMENT_NODE) { org.w3c.dom.Element element = (org.w3c.dom.Element) node; data.put(element.getNodeName(), element.getTextContent()); } } try { stream.close(); } catch (Exception ex) { // do nothing } return data; } catch (Exception ex) { WXPayUtil.getLogger().warn("Invalid XML, can not convert to map. Error message: {}. XML content: {}", ex.getMessage(), strXML); throw ex; } } /** * 将Map转换为XML格式的字符串 * * @param data Map类型数据 * @return XML格式的字符串 * @throws Exception */ public static String mapToXml(Map<String, String> data) throws Exception { org.w3c.dom.Document document = WXPayXmlUtil.newDocument(); org.w3c.dom.Element root = document.createElement("xml"); document.appendChild(root); for (String key : data.keySet()) { String value = data.get(key); if (value == null) { value = ""; } value = value.trim(); org.w3c.dom.Element filed = document.createElement(key); filed.appendChild(document.createTextNode(value)); root.appendChild(filed); } TransformerFactory tf = TransformerFactory.newInstance(); Transformer transformer = tf.newTransformer(); DOMSource source = new DOMSource(document); transformer.setOutputProperty(OutputKeys.ENCODING, "UTF-8"); transformer.setOutputProperty(OutputKeys.INDENT, "yes"); StringWriter writer = new StringWriter(); StreamResult result = new StreamResult(writer); transformer.transform(source, result); String output = writer.getBuffer().toString(); //.replaceAll("\n|\r", ""); try { writer.close(); } catch (Exception ex) { } return output; } /** * 生成带有 sign 的 XML 格式字符串 * * @param data Map类型数据 * @param key API密钥 * @return 含有sign字段的XML */ public static String generateSignedXml(final Map<String, String> data, String key) throws Exception { return generateSignedXml(data, key, SignType.MD5); } /** * 生成带有 sign 的 XML 格式字符串 * * @param data Map类型数据 * @param key API密钥 * @param signType 签名类型 * @return 含有sign字段的XML */ public static String generateSignedXml(final Map<String, String> data, String key, SignType signType) throws Exception { String sign = generateSignature(data, key, signType); data.put(WXPayConstants.FIELD_SIGN, sign); return mapToXml(data); } /** * 判断签名是否正确 * * @param xmlStr XML格式数据 * @param key API密钥 * @return 签名是否正确 * @throws Exception */ public static boolean isSignatureValid(String xmlStr, String key) throws Exception { Map<String, String> data = xmlToMap(xmlStr); if (!data.containsKey(WXPayConstants.FIELD_SIGN)) { return false; } String sign = data.get(WXPayConstants.FIELD_SIGN); return generateSignature(data, key).equals(sign); } /** * 判断签名是否正确,必须包含sign字段,否则返回false。使用MD5签名。 * * @param data Map类型数据 * @param key API密钥 * @return 签名是否正确 * @throws Exception */ public static boolean isSignatureValid(Map<String, String> data, String key) throws Exception { return isSignatureValid(data, key, SignType.MD5); } /** * 判断签名是否正确,必须包含sign字段,否则返回false。 * * @param data Map类型数据 * @param key API密钥 * @param signType 签名方式 * @return 签名是否正确 * @throws Exception */ public static boolean isSignatureValid(Map<String, String> data, String key, SignType signType) throws Exception { if (!data.containsKey(WXPayConstants.FIELD_SIGN)) { return false; } String sign = data.get(WXPayConstants.FIELD_SIGN); return generateSignature(data, key, signType).equals(sign); } /** * 生成签名 * * @param data 待签名数据 * @param key API密钥 * @return 签名 */ public static String generateSignature(final Map<String, String> data, String key) throws Exception { return generateSignature(data, key, SignType.MD5); } /** * 生成签名. 注意,若含有sign_type字段,必须和signType参数保持一致。 * * @param data 待签名数据 * @param key API密钥 * @param signType 签名方式 * @return 签名 */ public static String generateSignature(final Map<String, String> data, String key, SignType signType) throws Exception { Set<String> keySet = data.keySet(); String[] keyArray = keySet.toArray(new String[keySet.size()]); Arrays.sort(keyArray); StringBuilder sb = new StringBuilder(); for (String k : keyArray) { if (k.equals(WXPayConstants.FIELD_SIGN)) { continue; } if (data.get(k).trim().length() > 0) // 参数值为空,则不参与签名 sb.append(k).append("=").append(data.get(k).trim()).append("&"); } sb.append("key=").append(key); if (SignType.MD5.equals(signType)) { return MD5(sb.toString()).toUpperCase(); } else if (SignType.HMACSHA256.equals(signType)) { return HMACSHA256(sb.toString(), key); } else { throw new Exception(String.format("Invalid sign_type: %s", signType)); } } /** * 获取随机字符串 Nonce Str * * @return String 随机字符串 */ public static String generateNonceStr() { char[] nonceChars = new char[32]; for (int index = 0; index < nonceChars.length; ++index) { nonceChars[index] = SYMBOLS.charAt(RANDOM.nextInt(SYMBOLS.length())); } return new String(nonceChars); } /** * 生成 MD5 * * @param data 待处理数据 * @return MD5结果 */ public static String MD5(String data) throws Exception { MessageDigest md = MessageDigest.getInstance("MD5"); byte[] array = md.digest(data.getBytes("UTF-8")); StringBuilder sb = new StringBuilder(); for (byte item : array) { sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3)); } return sb.toString().toUpperCase(); } /** * 生成 HMACSHA256 * * @param data 待处理数据 * @param key 密钥 * @return 加密结果 * @throws Exception */ public static String HMACSHA256(String data, String key) throws Exception { Mac sha256_HMAC = Mac.getInstance("HmacSHA256"); SecretKeySpec secret_key = new SecretKeySpec(key.getBytes("UTF-8"), "HmacSHA256"); sha256_HMAC.init(secret_key); byte[] array = sha256_HMAC.doFinal(data.getBytes("UTF-8")); StringBuilder sb = new StringBuilder(); for (byte item : array) { sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3)); } return sb.toString().toUpperCase(); } /** * 日志 * * @return */ public static Logger getLogger() { Logger logger = LoggerFactory.getLogger("wxpay java sdk"); return logger; } /** * 获取当前时间戳,单位秒 * * @return */ public static long getCurrentTimestamp() { return System.currentTimeMillis() / 1000; } /** * 获取当前时间戳,单位毫秒 * * @return */ public static long getCurrentTimestampMs() { return System.currentTimeMillis(); } //生成签名 public static String createSign(SortedMap<String, Object> parameters) { StringBuilder sb = new StringBuilder(); Set es = parameters.entrySet(); for (Object e : es) { Map.Entry entry = (Map.Entry) e; String k = (String) entry.getKey(); Object v = entry.getValue(); if (null != v && !"".equals(v) && !"sign".equals(k) && !"key".equals(k)) { sb.append(k).append("=").append(v).append("&"); } } sb.append("key=").append("1234abcd1234abcd1234abcd1234abcd"); return MD5.MD5Encode(sb.toString(), "UTF-8").toUpperCase(); } /** * 第二次签名专用 * MD5 加密,转为指定类型 * * @param text * @param key * @param input_charset * @return */ public static String sign(String text, String key, String input_charset) { text = text + key; return DigestUtils.md5Hex(getContentBytes(text, input_charset)); } public static byte[] getContentBytes(String content, String charset) { if (charset == null || "".equals(charset)) { return content.getBytes(); } try { return content.getBytes(charset); } catch (UnsupportedEncodingException e) { throw new RuntimeException("MD5签名过程中出现错误,指定的编码集不对,您目前指定的编码集是:" + charset); } } }
下面是controller层 这边返回的数据正是前端那边需要的参数 注意下面有个TODO的地方改成你的appid
package com.landimc.controller; import com.alibaba.fastjson.JSON; import com.landimc.compoent.pay.WxPayCompoent; import com.landimc.pojo.TWxjyjl; import com.landimc.service.WeChartTransactionService; import com.landimc.tools.PageUtils; import com.landimc.wxpay.UnifiedOrderWxPay; import com.landimc.wxpay.model.WxPayModel; import com.landimc.wxpay.utils.WXPayUtil; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestParam; import org.springframework.web.bind.annotation.RestController; import javax.annotation.Resource; import java.util.*; /** * @author Yang */ @RestController() public class TestController { @Autowired private WxPayCompoent wxPayCompoent; @Resource private WeChartTransactionService weChartTransactionService; @RequestMapping("/searchPayList") public String searchPayList(TWxjyjl tWxjyjl, PageUtils pageUtils) { return JSON.toJSONString(weChartTransactionService.getWechatTransactionsAll(tWxjyjl, pageUtils)); } @RequestMapping("/pay") public String test() throws Exception { String orderNo = System.currentTimeMillis() + ""; SortedMap<String, Object> paramsMap = new TreeMap<>(); paramsMap.put("appid", wxPayCompoent.appid); paramsMap.put("mch_id", wxPayCompoent.mch_id); paramsMap.put("sub_appid", wxPayCompoent.sub_appid); paramsMap.put("sub_mch_id", wxPayCompoent.sub_mch_id); paramsMap.put("nonce_str", UUID.randomUUID().toString().replace("-", "")); paramsMap.put("body", "demo-01"); paramsMap.put("out_trade_no", orderNo); paramsMap.put("total_fee", "1"); paramsMap.put("trade_type", "JSAPI"); paramsMap.put("sub_openid", "oi0pP5UhaAENr2b66xzb5Rx0a1Do"); paramsMap.put("spbill_create_ip", "127.0.0.1"); paramsMap.put("notify_url", "https://www.baidu.com"); paramsMap.put("sign", WXPayUtil.createSign(paramsMap)); System.out.println(WXPayUtil.createSign(paramsMap)); Map<String, String> map = UnifiedOrderWxPay.toMchPay(wxPayCompoent.mchPayUrl, paramsMap, wxPayCompoent.certificate, wxPayCompoent.certificatePassword); String timeStamp = String.valueOf(System.currentTimeMillis() / 1000); // 支付密钥 String key = "&key=" + wxPayCompoent.serverPayKey; String prepay_id = ""; prepay_id = map.get("prepay_id"); String packages = "prepay_id=" + prepay_id; //生成32位以下随机编码 String nonceStr1 = UUID.randomUUID().toString().replace("-", ""); // 开始第二次签名 //TODO 改成你的APPID String mapStr1 = "appId=wx52311f98asdasd&nonceStr=" + nonceStr1 + "&package=prepay_id=" + prepay_id + "&signType=MD5&timeStamp=" + timeStamp; String paySign = WXPayUtil.sign(mapStr1, key, "utf-8").toUpperCase(); Map<String, String> resultMap = new HashMap<>(); resultMap.put("timeStamp", timeStamp); resultMap.put("nonceStr", nonceStr1); resultMap.put("package", packages); resultMap.put("paySign", paySign); resultMap.put("payno", orderNo); return JSON.toJSONString(resultMap); } @RequestMapping("/search") public String search(String tradeNo) throws Exception { SortedMap<String, Object> paramsMap = new TreeMap<>(); paramsMap.put("appid", wxPayCompoent.appid); paramsMap.put("mch_id", wxPayCompoent.mch_id); paramsMap.put("sub_mch_id", wxPayCompoent.sub_mch_id); paramsMap.put("out_trade_no", tradeNo); paramsMap.put("nonce_str", UUID.randomUUID().toString().replace("-", "")); paramsMap.put("sign_type", "MD5"); paramsMap.put("sign", WXPayUtil.createSign(paramsMap)); Map<String, String> map = UnifiedOrderWxPay.toMchPay(wxPayCompoent.seachOrderURL, paramsMap, wxPayCompoent.certificate, wxPayCompoent.certificatePassword); return JSON.toJSONString(map); } }
下面提供前端的代码 这边是uniapp 可以看一下大概的参数 换成自己的前端
uni.requestPayment({
provider: ‘wxpay‘,
timeStamp: res.data.timeStamp,
nonceStr: res.data.nonceStr,
package: res.data.package,
signType: ‘MD5‘,
paySign: res.data.paySign,
success: function(res) {
//TODO 执行自己的业务代码 进行查单
},
fail: function(err) {
console.log(‘fail:‘ + JSON.stringify(err));
}
})