导入证书
微信退款是需要证书的 https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=4_3
以在windows为例,解压之后的文件
双击.p12结尾的文件,导入证书,会要求输入密码,密码就是商户ID,注意一定要是在自己的商户平台上下载的证书,不然会提示密码错误。
导入成功
Java代码
封装了一个RefundVo对象,字段设定根据官方文档
public class RefundVo { private String appid; private String mchId; private String deviceInfo; private String nonceStr; private String sign; private String signType; private String transactionId; private String outTradeNo; private String outRefundNo; private int totalFee; private int refundFee; private String refundFeeType; private String opUserId; private String refundAccount; //省略get set方法 }
设置各个参数
String key = "xxxxxxx"; RefundVo vo = new RefundVo(); vo.setOutTradeNo("2016006092770333");//商户订单号(微信订单号二选一即可) vo.setAppid("appid"); vo.setMchId("mchid"); vo.setOutRefundNo("2016006092770333");//某付款的定单号 vo.setTotalFee(1);//订单金额 vo.setRefundFee(1);退款金额 vo.setOpUserId("1410417402");//默认商户号 String certificatePath = "E:/工作/cert/apiclient_cert.p12";//证书的绝对路径 refund(key ,vo,certificatePath );
封装退款的结果
public class RefundResult { private String returnCode; private String returnMsg; private String resultCode; private String errCode; private String errCodeDes; private String appid; private String mchId; private String deviceInfo; private String nonceStr; private String sign; private String transactionId; private String outTradeNo; private String outRefundNo; private String refundId; /** * ORIGINAL—原路退款 * BALANCE—退回到余额 */ private String refundChannel; /** * 申请退款金额 */ private int refundFee; /** * 退款金额 */ private int settlementRefundFee; private int totalFee; private int settlementTotalFee; private String feeType; private int cashFee; private int cashRefundFee; //省略set get方法 }
退款的方法
public RefundResult refund(String key,RefundVo vo,String certificatePath){ RefundResult refundResult = new RefundResult(); vo.setNonceStr(RandomUtil.wechatRandomString());//设置随机字符串 vo.setSign(new RefundBuilder().build(vo));//设置签名 check(vo);//检查参数 //将参数放入Map中 Map<String,String> params=new RefundBuilder().getParams(vo); //转成Xml形式的String String xml=XmlParseUtils.assembleXml(params); /** <xml> <appid>wx2421b1c4370ec43b</appid> <mch_id>10000100</mch_id> <nonce_str>6cefdb308e1e2e8aabd48cf79e546a02</nonce_str> <op_user_id>10000100</op_user_id> <out_refund_no>1415701182</out_refund_no> <out_trade_no>1415757673</out_trade_no> <refund_fee>1</refund_fee> <total_fee>1</total_fee> <transaction_id></transaction_id> <sign>FE56DD4AA85C0EECA82C35595A69E153</sign> </xml> **/ //调用微信接口 String result = HttpClientUtils.executeBySslPost(refundURL,xml,vo.getCertificatePath(),vo.getRefundVo().getMchId());//发送http请求 //接收xml解析的结果 Map<String, String> map = new HashMap<String,String>(); //返回结果为xml形式,转成map然后封装成refundResult即可 map = XmlParseUtils.parseXml(result); refundResult = new RefundResultBuilder().build(map); }
参数检查

private void check(RefundVo vo){ if (VerifyUtils.isEmpty(vo.getAppid())) { throw new PayException("申请退款参数为空——appid"); } if (VerifyUtils.isEmpty(vo.getMchId())) { throw new PayException("申请退款参数为空——mch_id"); } if (VerifyUtils.isEmpty(vo.getNonceStr())) { throw new PayException("申请退款参数为空——nonce_str"); } if (VerifyUtils.isEmpty(vo.getSign())) { throw new PayException("申请退款参数为空——sign"); } if (VerifyUtils.isEmpty(vo.getTransactionId()) && VerifyUtils.isEmpty(vo.getOutTradeNo())) { throw new PayException("申请退款参数为空——transaction_id或者out_trade_no"); } if (VerifyUtils.isEmpty(vo.getOutRefundNo())) { throw new PayException("申请退款参数为空——out_refund_no"); } if (VerifyUtils.isEmpty(vo.getTotalFee())) { throw new PayException("申请退款参数为空——total_fee"); } if (VerifyUtils.isEmpty(vo.getRefundFee())) { throw new PayException("申请退款参数为空——refund_fee"); } if (VerifyUtils.isEmpty(vo.getOpUserId())) { throw new PayException("申请退款参数为空——op_user_id"); } }
Map构建
public class RefundBuilder extends SignBuilder { @Override public Map<String, String> getParams(Refund vo) { Map<String,String> params = new HashMap<String, String>(); if(VerifyUtils.isNotEmpty(vo.getAppid())){ params.put("appid",vo.getAppid()); } if (VerifyUtils.isNotEmpty(vo.getMchId())) { params.put("mch_id", vo.getMchId()); } if (VerifyUtils.isNotEmpty(vo.getDeviceInfo())) { params.put("device_info", vo.getDeviceInfo()); } if(VerifyUtils.isNotEmpty(vo.getNonceStr())){ params.put("nonce_str",vo.getNonceStr()); } if (VerifyUtils.isNotEmpty(vo.getSign())) { params.put("sign", vo.getSign()); } if (VerifyUtils.isNotEmpty(vo.getSignType())) { params.put("sign_type", vo.getSignType()); } if (VerifyUtils.isNotEmpty(vo.getTransactionId())) { params.put("transaction_id", vo.getTransactionId()); } if (VerifyUtils.isNotEmpty(vo.getOutTradeNo())) { params.put("out_trade_no", vo.getOutTradeNo()); } if (VerifyUtils.isNotEmpty(vo.getOutRefundNo())) { params.put("out_refund_no", vo.getOutRefundNo()); } if (VerifyUtils.isNotEmpty(vo.getTotalFee())) { params.put("total_fee",Integer.toString(vo.getTotalFee())); } if (VerifyUtils.isNotEmpty(vo.getRefundFee())) { params.put("refund_fee", Integer.toString(vo.getRefundFee())); } if (VerifyUtils.isNotEmpty(vo.getRefundFeeType())) { params.put("refund_fee_type",vo.getRefundFeeType()); } if (VerifyUtils.isNotEmpty(vo.getOpUserId())) { params.put("op_user_id",vo.getOpUserId()); } if (VerifyUtils.isNotEmpty(vo.getRefundAccount())) { params.put("refund_account",vo.getRefundAccount()); } return params; } }
http执行的方法

public static String executeBySslPost(String url, String body,String certificatePath,String password) throws Exception { String result = ""; //商户id //指定读取证书格式为PKCS12 KeyStore keyStore = KeyStore.getInstance("PKCS12"); //读取本机存放的PKCS12证书文件 FileInputStream instream = new FileInputStream(new File(certificatePath)); try { //指定PKCS12的密码(商户ID) keyStore.load(instream, password.toCharArray()); } finally { instream.close(); } SSLContext sslcontext = SSLContexts.custom().loadKeyMaterial(keyStore, password.toCharArray()).build(); //指定TLS版本 SSLConnectionSocketFactory sslsf = new SSLConnectionSocketFactory(sslcontext, new String[]{"TLSv1"}, null, SSLConnectionSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER); //设置httpclient的SSLSocketFactory CloseableHttpClient httpclient = HttpClients.custom().setSSLSocketFactory(sslsf).build(); try { HttpPost httppost = new HttpPost(url); StringEntity reqEntity = new StringEntity(body, "UTF-8"); httppost.setEntity(reqEntity); System.out.println("Executing request: " + httppost.getRequestLine()); CloseableHttpResponse response = null; try { response = httpclient.execute(httppost); result = EntityUtils.toString(response.getEntity(),"UTF-8"); } catch (Exception e) { e.printStackTrace(); log.error("请求失败", e); throw new RuntimeException(e); } finally { try { response.close(); } catch (IOException e) { e.printStackTrace(); } } } catch (Exception e) { e.printStackTrace(); log.error("请求失败", e); throw new RuntimeException(e); } finally { try { httpclient.close(); } catch (IOException e) { e.printStackTrace(); } } return result; }
最终的返回结果
System.out.println(JSON.toJSONString(result,true));