当前位置: 移动技术网 > IT编程>开发语言>.net > 微信公众平台支付开发详解

微信公众平台支付开发详解

2017年12月12日  | 移动技术网IT编程  | 我要评论

苏志刚,赛尔号科雷傲,北京家电维修公司

公众号支付就是在微信里面的h5页面唤起微信支付,不用扫码即可付款的功能。做这个功能首先要明确的就是,只有和商户号mch_id匹配的appid才能成功支付。商户号在注册成功的时候就会将相关信息发送到邮箱里面。而唤起支付的一个关键是靠openid拿到统一下单。而openid是和appid一一对应的。也就是说如果你登录使用的appid不是公众号的appid,得到的openid就无法唤起公众号内的支付(会出现appid和商户号不匹配的错误)。曾经就在这个地方绕了个弯,因为微信的开放平台可以创建网站应用,也有一个appid和appsecreat,也可以在微信里面一键登录。

业务流程

下面是微信的官方流程,看似有点复杂,重点就是要拿到统一下单接口返回的json串,其他按照官方demo基本就能正确,下面说一下几个细节。

创建订单

在调用微信公众号支付之前,首先我们自己要把订单创建好。比如一个充值的订单。主要是先确定下金额再进行下一步。

public jsonresult createrecharegorder(decimal money)
 {
 if (money < (decimal)0.01) return json(new paymentresult("充值金额非法!"));
 var user = _workcontext.currentuser;
 var order = _paymentservice.createrechargeorder(user.id, money);
 return json(new paymentresult(true) {orderid = order.ordernumber});
 }

调用统一下单

订单创建成功之后,页面跳转到支付页面,这个时候就是按照官方的流程去拿prepay_id和paysign,微信的demo中提供了一个jsapipay的对象。但这个对象需要一个page对象初始化。

[loginvalid]
 public actionresult h5pay(string ordernumber)
 {
 var user = _workcontext.currentuser;
 var order = _paymentservice.getorderbyordernumber(ordernumber);
 //判断订单是否存在
 //订单是否已经支付了
 var openid = user.openid;
 var jsapipay = new jsapipaymvc(this.controllercontext.httpcontext);
 jsapipay.openid = openid;
 jsapipay.total_fee = (int)order.amount * 100;
 wxpaydata unifiedorderresult = jsapipay.getunifiedorderresult();
 viewbag.wxjsapiparam = jsapipay.getjsapiparameters();//获取h5调起js api参数 
 viewbag.unifiedorder = unifiedorderresult.toprintstr();
 viewbag.ordernumber = order.ordernumber;
 return view();
 }

在mvc中我们简单改一下就可以了。也就是把page对象换成httpcontext即可。然后里面的方法就可以直接用了。

jsapipaymvc:

using system;
using system.collections.generic;
using system.web;
using system.web.ui;
using system.web.ui.webcontrols;
using system.runtime.serialization;
using system.io;
using system.text;
using system.net;
using system.web.security;
using litjson;
namespace wxpayapi
{
 public class jsapipaymvc
 {
 /// <summary>
 /// 保存页面对象,因为要在类的方法中使用page的request对象
 /// </summary>
 public httpcontextbase context { get; set; }
 /// <summary>
 /// openid用于调用统一下单接口
 /// </summary>
 public string openid { get; set; }
 /// <summary>
 /// access_token用于获取收货地址js函数入口参数
 /// </summary>
 public string access_token { get; set; }
 /// <summary>
 /// 商品金额,用于统一下单
 /// </summary>
 public int total_fee { get; set; }
 /// <summary>
 /// 统一下单接口返回结果
 /// </summary>
 public wxpaydata unifiedorderresult { get; set; }
 public jsapipaymvc(httpcontextbase _context)
 {
 context = _context;
 }
 /**
 * 
 * 网页授权获取用户基本信息的全部过程
 * 详情请参看网页授权获取用户基本信息:http://mp.weixin.qq.com/wiki/17/c0f37d5704f0b64713d5d2c37b468d75.html
 * 第一步:利用url跳转获取code
 * 第二步:利用code去获取openid和access_token
 * 
 */
 public void getopenidandaccesstoken(string code)
 {
 if (!string.isnullorempty(code))
 {
 //获取code码,以获取openid和access_token
 log.debug(this.gettype().tostring(), "get code : " + code);
 getopenidandaccesstokenfromcode(code);
 }
 else
 {
 //构造网页授权获取code的url
 string host = context.request.url.host;
 string path = context.request.path;
 string redirect_uri = httputility.urlencode("http://" + host + path);
 wxpaydata data = new wxpaydata();
 data.setvalue("appid", wxpayconfig.appid);
 data.setvalue("redirect_uri", redirect_uri);
 data.setvalue("response_type", "code");
 data.setvalue("scope", "snsapi_base");
 data.setvalue("state", "state" + "#wechat_redirect");
 string url = "https://open.weixin.qq.com/connect/oauth2/authorize?" + data.tourl();
 log.debug(this.gettype().tostring(), "will redirect to url : " + url);
 try
 {
 //触发微信返回code码 
 context.response.redirect(url);//redirect函数会抛出threadabortexception异常,不用处理这个异常
 }
 catch(system.threading.threadabortexception ex)
 {
 }
 }
 }
 /**
 * 
 * 通过code换取网页授权access_token和openid的返回数据,正确时返回的json数据包如下:
 * {
 * "access_token":"access_token",
 * "expires_in":7200,
 * "refresh_token":"refresh_token",
 * "openid":"openid",
 * "scope":"scope",
 * "unionid": "o6_bmasdasdsad6_2sgvt7hmzopfl"
 * }
 * 其中access_token可用于获取共享收货地址
 * openid是微信支付jsapi支付接口统一下单时必须的参数
 * 更详细的说明请参考网页授权获取用户基本信息:http://mp.weixin.qq.com/wiki/17/c0f37d5704f0b64713d5d2c37b468d75.html
 * @失败时抛异常wxpayexception
 */
 public void getopenidandaccesstokenfromcode(string code)
 {
 try
 {
 //构造获取openid及access_token的url
 wxpaydata data = new wxpaydata();
 data.setvalue("appid", wxpayconfig.appid);
 data.setvalue("secret", wxpayconfig.appsecret);
 data.setvalue("code", code);
 data.setvalue("grant_type", "authorization_code");
 string url = "https://api.weixin.qq.com/sns/oauth2/access_token?" + data.tourl();
 //请求url以获取数据
 string result = httpservice.get(url);
 log.debug(this.gettype().tostring(), "getopenidandaccesstokenfromcode response : " + result);
 //保存access_token,用于收货地址获取
 jsondata jd = jsonmapper.toobject(result);
 access_token = (string)jd["access_token"];
 //获取用户openid
 openid = (string)jd["openid"];
 log.debug(this.gettype().tostring(), "get openid : " + openid);
 log.debug(this.gettype().tostring(), "get access_token : " + access_token);
 }
 catch (exception ex)
 {
 log.error(this.gettype().tostring(), ex.tostring());
 throw new wxpayexception(ex.tostring());
 }
 }
 /**
 * 调用统一下单,获得下单结果
 * @return 统一下单结果
 * @失败时抛异常wxpayexception
 */
 public wxpaydata getunifiedorderresult()
 {
 //统一下单 
 wxpaydata data = new wxpaydata();
 data.setvalue("body", "test");
 data.setvalue("attach", "test");
 data.setvalue("out_trade_no", wxpayapi.generateouttradeno());
 data.setvalue("total_fee", total_fee);
 data.setvalue("time_start", datetime.now.tostring("yyyymmddhhmmss"));
 data.setvalue("time_expire", datetime.now.addminutes(10).tostring("yyyymmddhhmmss"));
 data.setvalue("goods_tag", "test");
 data.setvalue("trade_type", "jsapi");
 data.setvalue("openid", openid);
 wxpaydata result = wxpayapi.unifiedorder(data);
 if (!result.isset("appid") || !result.isset("prepay_id") || result.getvalue("prepay_id").tostring() == "")
 {
 log.error(this.gettype().tostring(), "unifiedorder response error!");
 throw new wxpayexception("unifiedorder response error!");
 }
 unifiedorderresult = result;
 return result;
 }
 /**
 * 
 * 从统一下单成功返回的数据中获取微信浏览器调起jsapi支付所需的参数,
 * 微信浏览器调起jsapi时的输入参数格式如下:
 * {
 * "appid" : "wx2421b1c4370ec43b", //公众号名称,由商户传入 
 * "timestamp":" 1395712654", //时间戳,自1970年以来的秒数 
* "noncestr" : "e61463f8efa94090b1f366cccfbbb444", //随机串 
* "package" : "prepay_id=u802345jgfjsdfgsdg888", 
 * "signtype" : "md5", //微信签名方式: 
* "paysign" : "70ea570631e4bb79628fbca90534c63ff7fadd89" //微信签名 
 * }
 * @return string 微信浏览器调起jsapi时的输入参数,json格式可以直接做参数用
 * 更详细的说明请参考网页端调起支付api:http://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_7
 * 
 */
 public string getjsapiparameters()
 {
 log.debug(this.gettype().tostring(), "jsapipay::getjsapiparam is processing...");

 wxpaydata jsapiparam = new wxpaydata();
 jsapiparam.setvalue("appid", unifiedorderresult.getvalue("appid"));
 jsapiparam.setvalue("timestamp", wxpayapi.generatetimestamp());
 jsapiparam.setvalue("noncestr", wxpayapi.generatenoncestr());
 jsapiparam.setvalue("package", "prepay_id=" + unifiedorderresult.getvalue("prepay_id"));
 jsapiparam.setvalue("signtype", "md5");
 jsapiparam.setvalue("paysign", jsapiparam.makesign());
 string parameters = jsapiparam.tojson();
 log.debug(this.gettype().tostring(), "get jsapiparam : " + parameters);
 return parameters;
 }
 /**
 * 
 * 获取收货地址js函数入口参数,详情请参考收货地址共享接口:http://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_9
 * @return string 共享收货地址js函数需要的参数,json格式可以直接做参数使用
 */
 public string geteditaddressparameters()
 {
 string parameter = "";
 try
 {
 string host = context.request.url.host;
 string path = context.request.path;
 string querystring = context.request.url.query;
 //这个地方要注意,参与签名的是网页授权获取用户信息时微信后台回传的完整url
 string url = "http://" + host + path + querystring;
 //构造需要用sha1算法加密的数据
 wxpaydata signdata = new wxpaydata();
 signdata.setvalue("appid",wxpayconfig.appid);
 signdata.setvalue("url", url);
 signdata.setvalue("timestamp",wxpayapi.generatetimestamp());
 signdata.setvalue("noncestr",wxpayapi.generatenoncestr());
 signdata.setvalue("accesstoken",access_token);
 string param = signdata.tourl();
 log.debug(this.gettype().tostring(), "sha1 encrypt param : " + param);
 //sha1加密
 string addrsign = formsauthentication.hashpasswordforstoringinconfigfile(param, "sha1");
 log.debug(this.gettype().tostring(), "sha1 encrypt result : " + addrsign);
 //获取收货地址js函数入口参数
 wxpaydata afterdata = new wxpaydata();
 afterdata.setvalue("appid",wxpayconfig.appid);
 afterdata.setvalue("scope","jsapi_address");
 afterdata.setvalue("signtype","sha1");
 afterdata.setvalue("addrsign",addrsign);
 afterdata.setvalue("timestamp",signdata.getvalue("timestamp"));
 afterdata.setvalue("noncestr",signdata.getvalue("noncestr"));
 //转为json格式
 parameter = afterdata.tojson();
 log.debug(this.gettype().tostring(), "get editaddressparam : " + parameter);
 }
 catch (exception ex)
 {
 log.error(this.gettype().tostring(), ex.tostring());
 throw new wxpayexception(ex.tostring());
 }
 return parameter;
 }
 }
}

这个页面可以在本地调试,可以比较方便的确认参数是否ok。

唤起支付

官方页面的示例如下: 但主要的参数(mark部分)是由后台生成的,也就是上一个步骤的viewbag.wxjsapiparam

function onbridgeready(){
 weixinjsbridge.invoke(
 'getbrandwcpayrequest', {
 "appid" : "wx2421b1c4370ec43b", //公众号名称,由商户传入 
 "timestamp":" 1395712654", //时间戳,自1970年以来的秒数 
 "noncestr" : "e61463f8efa94090b1f366cccfbbb444", //随机串 
 "package" : "prepay_id=u802345jgfjsdfgsdg888", 
 "signtype" : "md5", //微信签名方式: 
 "paysign" : "70ea570631e4bb79628fbca90534c63ff7fadd89" //微信签名 
 },
 function(res){ 
 if(res.err_msg == "get_brand_wcpay_request:ok" ) {} // 使用以上方式判断前端返回,微信团队郑重提示:res.err_msg将在用户支付成功后返回 ok,但并不保证它绝对可靠。 
 }
 ); 
}

所以在mvc中要这样写:

@{
 viewbag.title = "微信支付";
 layout = "~/views/shared/_layout.cshtml";
}
<div class="page" id="wxpayment">
 <div class="content">
 <div>订单详情:@html.raw(viewbag.unifiedorder)</div>
 <button id="h5pay" onclick="callpay()">支付</button>
 </div>
 <input type="hidden" value="@viewbag.ordernumber" id="ordernum"/>
</div>
<script type="text/javascript">
 //调用微信js api 支付
 function jsapicall() {
 weixinjsbridge.invoke(
 'getbrandwcpayrequest',
 @html.raw(viewbag.wxjsapiparam),//josn串
 function (res)
 {
 weixinjsbridge.log(res.err_msg);
 //alert(res.err_code + res.err_desc + res.err_msg);
 if (res.err_msg == "get_brand_wcpay_request:ok") {
 var num = $("#ordernum").val();
 $.post("/payment/weixinpaysuccess", { ordernumber: num }, function(data) {
 if (data.issuccess === true) {
 alert("支付成功");
 location.href = document.referrer;
 } else {
 }
 });
 } 
 if (res.err_msg == 'get_brand_wcpay_request:cancel') {
 $('.button').removeattr('submitting');
 alert('取消支付');
 } 
 }
 );
 }
 function callpay()
 {
 if (typeof weixinjsbridge == "undefined")
 {
 alert("weixinjsbridge =");
 if (document.addeventlistener)
 {
 document.addeventlistener('weixinjsbridgeready', jsapicall, false);
 }
 else if (document.attachevent)
 {
 document.attachevent('weixinjsbridgeready', jsapicall);
 document.attachevent('onweixinjsbridgeready', jsapicall);
 }
 }
 else
 {
 jsapicall();
 }
 }
</script>

必须要用html.raw,不然json解析不对,无法支付。这个时候点击页面,会出现微信的加载效果,但别高兴的太早,还是会出错,出现一个“3当前的url未注册”

原因就在于,需要在公众号中。而这个支付目录是大小写敏感的,所以你得多试几次。直到弹出输入密码的窗口才是真的流程正确了。然后支付成功之后马上就可以收到js中的回调,这个时候你可以去处理你的订单和业务逻辑。

官方文档:

微信扫码支付 demo:

官方demo:

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持移动技术网!

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网