Java 关于微信公众号支付总结附代码

很多朋友第一次做微信支付的时候都有蒙,但当你完整的做一次就会发现其实并没有那么难

业务流程和应用场景官网有详细的说明:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_4,我就不做具体介绍了,我主要介绍一下作为开发的我们需要做的事情,微信官网要完成的部分我们就先不要考虑了

开发流程:

  1.获取用户授权,有两种形式:

  一种是snsapi_base (不弹出授权页面,直接跳转,只能获取用户openid);另一种是:snsapi_userinfo (弹出授权页面,可通过openid拿到昵称、性别、所在地。并且, 即使在未关注的情况下,只要用户授权,也能获取其信息 ) 

  2.调用统一下单接口获取预支付ID;(重点)

  3.H5调起微信支付的内置JS

  4.支付完成后,微信回调的处理,这里主要就是处理支付成功后自己项目的业务逻辑

获取统一下单接口所需参数:

  所需参数去哪里找其实官方也有具体说明:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1

  appid:微信支付分配的公众账号ID(企业号corpid即为此appId) 登陆微信公众号后台==>开发==>基本配置

  mch_id : 微信支付分配的商户号 ;  登陆位置公众号支付后台就可以看到

  device_info :设备号;可以为终端设备号(门店号或收银设备ID) PC网页或公众号内支付可以传"WEB"   

  nonce_str:随机字符串,长度要求在32位以内;自己随机生成就行

    小科普一下:nonce=Number once 表示在密码学中Nonce是一个只被使用一次的任意或非重复的随机数值

  sign:签名,官方详细描述:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=4_3;

    这是小白们经常栽跟头的地方,不过不用担心,下面我会附上前面算法

  sign_type:签名类型,默认为MD5,支持HMAC-SHA256和MD5。

  body:商品简单描述信息,自定义即可

  out_trade_no:商户系统内部订单号,要求32个字符内,只能是数字、大小写字母_-|* 且在同一个商户号下唯一;

  total_fee:订单总金额,注意单位为分 

  spbill_create_ip:终端IP;APP和网页支付提交用户端ip

  notify_url:回调函数的通知地址,异步接收微信支付结果通知的回调地址,通知url必须为外网可访问的url,不能携带参数,就是自己处理支付完业务的接口路径

  trade_type:交易类型,JSAPI 公众号支付;NATIVE 扫码支付;APP APP支付

  key  :密钥,签名时用,key设置路径:微信商户平台(pay.weixin.qq.com)-->账户设置-->API安全-->密钥设置

  最方面的就是自己想了一个字符串,然后用MD5(其他加密方式也可以)加密成32位的字符串,重新设置。

调用统一下单接口:

  所需要的参数签名已经都有具体介绍了,这里我就直接上代码了

SortedMap<Object,Object> packageParams = new TreeMap<Object,Object>();
        packageParams.put("appid", appid);
        packageParams.put("mch_id", mch_id);
        packageParams.put("device_info", "WEB");
        packageParams.put("body", "自定义描述信息");
        packageParams.put("out_trade_no","商户系统内部订单号");
        packageParams.put("total_fee","支付金额");
        packageParams.put("spbill_create_ip", "终端IP");
        packageParams.put("notify_url","回调地址url");
        packageParams.put("trade_type","JSAPI");
        packageParams.put("nonce_str", "随机字符串");
        packageParams.put("sign","签名结果");

  统一下单接口地址:https://api.mch.weixin.qq.com/pay/unifiedorder

  准备好数据后,封装成XML,调起统一下单接口,会返回一个XML格式数据 return_code为SUCCESS时,请求成功,获取我们需要的预支付id(prepay_id)这时就已经成功了一大半,可以稍作休息,犒劳一下自己了。

附上加密算法:传入上面准备的map数据

		StringBuffer sb = new StringBuffer();
		Set es = parameters.entrySet();
		Iterator it = es.iterator();
		while(it.hasNext()) {
			Map.Entry entry = (Map.Entry)it.next();
			String k = (String)entry.getKey();
			Object v = entry.getValue();
			if(null != v && !"".equals(v) 
					&& !"sign".equals(k) && !"key".equals(k)) {
				sb.append(k + "=" + v + "&");
			}
		}
		sb.append("key=" +密钥key);
		String sign = MD5Util.MD5Encode(sb.toString(), characterEncoding).toUpperCase();

H5调起微信支付的内置JS 准备数据:

  这次的字段名和上次准备数据有所不一样

  appId:只是I大写,值还是一样的

     nonceStr:随机字符串,长度要求在32位以内;自己随机生成就行
        package:统一下单接口返回的prepay_id参数值,提交格式如:prepay_id=***
     timeStamp:时间戳

  signType:签名方式,默认MD5

  paySign:签名

        SortedMap<Object,Object> packageParams = new TreeMap<Object,Object>();
        packageParams.put("appId","商户号");
        packageParams.put("timeStamp","时间戳");
        packageParams.put("nonceStr","随机字符串");
        packageParams.put("prepayid","前面得到的预支付id");
        packageParams.put("package","prepay_id=前面得到的预支付id");
        packageParams.put("signType","MD5");
        packageParams.put("paySign","签名");

  如果所有配置都OK,就会弹出请输入密码的界面,反之提示目录未授权,这时候你就应该想时候配置授权目录了

  微信公众平台后台 ==> 微信支付 ==> 开发者配置,S API网页支付,然后添加系统后台的授权目录

  其他微信支付配置参考:https://jingyan.baidu.com/article/77b8dc7fef360d6174eab690.html

回调函数处理:

    1)解析传过来的流信息,通过重新签名的方式验证流中包含的信息的正确性。就是判断这个信息到底是不是微信发的

    2)return_code和result_code都是SUCCESS的话,处理商户自己的业务逻辑。就是订单的支付状态啊等一些信息。

    3)告诉微信,我收到你的返回值了。不用在发了。

猜你喜欢

转载自www.cnblogs.com/bert227/p/9570623.html