十年网站开发经验 + 多家企业客户 + 靠谱的建站团队
量身定制 + 运营维护+专业推广+无忧售后,网站问题一站解决
本篇文章为大家展示了怎么在Java中使用SSM框架实现一个微信退款功能,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。
站在用户的角度思考问题,与客户深入沟通,找到安国网站设计与安国网站推广的解决方案,凭借多年的经验,让设计与互联网技术结合,创造个性化、用户体验好的作品,建站类型包括:成都做网站、成都网站制作、企业官网、英文网站、手机端网站、网站推广、域名注册、网络空间、企业邮箱。业务覆盖安国地区。
String outTradeNo = request.getParameter("outTradeNo");// 获取商户订单号 Integer totalFee = Integer.parseInt(request.getParameter("totalFee"));// 获取支付金额 MapgetMap = new HashMap (); // 获得当前目录 String path = request.getSession().getServletContext().getRealPath("/"); Date now = new Date(); SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMddHHmmss");// 可以方便地修改日期格式 String outRefundNo = "NO" + dateFormat.format(now);
提供的参数有订单号这个是支付成功之后生成的唯一号码,然后是获取到用户支付的金额这两个参数都是由支付之后的订单上面获得的。下面那个path则是保存微信安全证书文件的位置,这里提一下要实现微信退款和微信企业转账功能是需要到微信商户平台去下载安全证书的,然后把证书放在项目的WEB-INF/目录下即可。
RefundReqData refundReqData = new RefundReqData(); refundReqData.setAppid(Configure.getAppID()); refundReqData.setMch_id(Configure.getMch_id()); refundReqData.setNonce_str(RandomStringGenerator.getRandomStringByLength(32)); refundReqData.setOut_trade_no(outTradeNo); refundReqData.setOut_refund_no(outRefundNo); refundReqData.setTotal_fee(totalFee); refundReqData.setRefund_fee(refundFee); refundReqData.setOp_user_id(Configure.getMch_id()); refundReqData.setNotify_url("https://weixin.qq.com/notify/"); String sign = Signature.getSign(refundReqData);// 生成签名 refundReqData.setSign(sign);
获取到需要的参数之后呢,我在这里使用了一个退款的实体类把这些参数保存到了我的实体类里面方便后面的签名加密。
ArrayListlist = new ArrayList (); @SuppressWarnings("rawtypes") Class cls = o.getClass(); Field[] fields = cls.getDeclaredFields(); for (Field f : fields) { f.setAccessible(true); if (f.get(o) != null && f.get(o) != "") { String name = f.getName(); XStreamAlias anno = f.getAnnotation(XStreamAlias.class); if(anno != null) name = anno.value(); list.add(name + "=" + f.get(o) + "&"); } } int size = list.size(); String [] arrayToSort = list.toArray(new String[size]); Arrays.sort(arrayToSort, String.CASE_INSENSITIVE_ORDER); StringBuilder sb = new StringBuilder(); for(int i = 0; i < size; i ++) { sb.append(arrayToSort[i]); } String result = sb.toString(); result += "key=" + Configure.getKey(); System.out.println("签名数据:"+result); result = MD5Util.MD5Encode(result,"utf-8").toUpperCase(); return result;
这个是我签名加密的方法,把数据加密之后会成为一个很长的字符串,但是官方提供的退款接口是没办法解析你这个超长字符串的数据的,所以我们要把这个字符串变成官方接口认识的数据格式也就是xml格式。
private static XStream xstream = new XStream(new XppDriver() { public HierarchicalStreamWriter createWriter(Writer out) { return new PrettyPrintWriter(out) { // 对所有xml节点的转换都增加CDATA标记 boolean cdata = true; //@SuppressWarnings("unchecked") public void startNode(String name, Class clazz) { super.startNode(name, clazz); } protected void writeText(QuickWriter writer, String text) { if (cdata) { writer.write(""); } else { writer.write(text); } } }; } });
这一段代码是我把字符串格式的数据转换成xml格式的方法。再把xml格式的数据保存在一个字符串里面,这个时候我们开始向官方接口发送数据。
public String httpsRequest(String url, String xmlObj, String path) throws Exception { // 加载证书 initCert(path); String result = null; HttpPost httpPost = new HttpPost(url); // 得指明使用UTF-8编码,否则到API服务器XML的中文不能被成功识别 StringEntity postEntity = new StringEntity(xmlObj, "UTF-8"); httpPost.addHeader("Content-Type", "text/xml"); httpPost.setEntity(postEntity); // 设置请求器的配置 httpPost.setConfig(requestConfig); try { HttpResponse response = httpClient.execute(httpPost); HttpEntity entity = response.getEntity(); result = EntityUtils.toString(entity, "UTF-8"); } catch (ConnectionPoolTimeoutException e) { e.printStackTrace(); } catch (ConnectTimeoutException e) { e.printStackTrace(); } catch (SocketTimeoutException e) { e.printStackTrace(); } catch (Exception e) { e.printStackTrace(); } finally { httpPost.abort(); } return result; }
通过Https往API post xml数据。
RefundRequest refundRequest = new RefundRequest(); String result = refundRequest.httpsRequest("https://api.mch.weixin.qq.com/secapi/pay/refund", info, path); getMap = MobiMessage.parseXml(new String(result.toString().getBytes(), "utf-8")); System.out.println(getMap + "............getMap"); json.put("return_msg", getMap.get("return_msg")); json.put("return_code", getMap.get("return_code")); json.put("outTradeNo", outTradeNo);
这一段就是给接口发送数据的代码(官方api接口,xml数据,证书的位置),然后我们接受接口返回的信息通过返回的return_msg和return_code来判断是否退款成功。
好了,微信退款就是这样完全可以照着代码把流程读出来很清晰明了也很简单,代码能力稍强的都看得懂,我主要是给大家提供一个思路。如果有同学没看懂也没关系下面是该项目的源码地址大家可以去下载退款的源代码都在里面:wechat_jb51.rar
PS:总结一下我在做微信退款的时候遇到的问题:
1.遇到了一个"Keystore password was incorrect"这个问题,原因这个退款所需要的证书不正确,这个证书是需要从微信平台去下载这个证书;
2.一定要注意在支付时的订单号码和退款时的订单号码是一致的,我碰到的这个问题是在支付时,把订单号码和微信返回的交易号码存数据库时弄反了,导致微信找不到这笔订单;
3.另外碰到的问题是退款在获取证书的时候,证书的路径不对,导致没有获取到证书,所以退款失败,所以还要检查证书是否存在,证书的路径是否正确,还要留意服务器上能否获取到证书。
上述内容就是怎么在Java中使用SSM框架实现一个微信退款功能,你们学到知识或技能了吗?如果还想学到更多技能或者丰富自己的知识储备,欢迎关注创新互联行业资讯频道。