美文网首页微信公众平台技术专题
微信支付(退款为例)

微信支付(退款为例)

作者: 大都小酒馆 | 来源:发表于2016-12-23 15:46 被阅读3712次
    微信支付退款的官方文档

    https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_4

    导入证书

    微信退款是需要证书的
    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));
    

    欢迎大家讨论~我的博客地址 http://blog.doublez.cc

    相关文章

      网友评论

      • simplife木易:问一下作者:new RefundBuilder().build(vo) 这个地方的RefundBuilder类在哪里??
        f42822589d15:能不能也发给我啊1458713877@qq.com
      • 21ea7cf2d4a9:非全额退款怎么处理呀!!!
      • 022803d2a0a5:老哥 上面RefundBuilder extends SignBuilder, 这个SignBuilder是个什么类啊,能发一下代码吗?
        xqfeng@sogaa.net,我的邮箱,谢谢啦
        f42822589d15:能不能也发给我啊1458713877@qq.com
      • ef21d44066f0:退款之后不需要进行签名校验吗?
        大都小酒馆:需要的
      • 空乱木:非常感谢,参考代码,把退款调通了,,,:grin:

      本文标题:微信支付(退款为例)

      本文链接:https://www.haomeiwen.com/subject/fglkvttx.html