android端微信支付V3版本地簽名統(tǒng)一下單詳解
滿滿的都是坑,因為服務(wù)器偷懶讓客服端寫統(tǒng)一下單,服務(wù)器只給了通知的url。微信的支付demo并沒有統(tǒng)一下單的代碼。
讀此文前先閱讀: https://pay.weixin.qq.com/wiki/doc/api/app/app.PHP?chapter=9_1
一步步的來 先根據(jù)統(tǒng)一下單的參數(shù)介紹工具:
1. 獲取到當(dāng)前的ip:
<span style="font-size:14px;">public String getLocalIpAddress() { try { for (Enumeration<NetworkInterface> en = NetworkInterface.getNetworkInterfaces(); en.hasMoreElements(); ) { NetworkInterface intf = en.nextElement(); for (Enumeration<InetAddress> enumIpAddr = intf.getInetAddresses(); enumIpAddr.hasMoreElements(); ) { InetAddress inetAddress = enumIpAddr.nextElement(); if (!inetAddress.isLoopbackAddress()) { return inetAddress.getHostAddress().toString(); } } } } catch (SocketException ex) { } return null; } private String getWifiIp() { //獲取wifi服務(wù) WifiManager wifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE); //判斷wifi是否開啟 if (!wifiManager.isWifiEnabled()) { wifiManager.setWifiEnabled(true); } WifiInfo wifiInfo = wifiManager.getConnectionInfo(); int ipAddress = wifiInfo.getIpAddress(); String ip = intToIp(ipAddress); return ip; } private String intToIp(int i) { return (i & 0xFF) + "." + ((i >> 8) & 0xFF) + "." + ((i >> 16) & 0xFF) + "." + (i >> 24 & 0xFF); } </span>
2.隨機(jī)訂單號生成 test 你們可根據(jù)自己生成隨機(jī)數(shù):
<span style="font-size:14px;">private String genOutTradNo() { Random random = new Random(); return MD5.getMessageDigest(String.valueOf(random.nextInt(10000)).getBytes()); }</span>
3.簽名工具:
<span style="font-size:14px;">private String genAppSign(List<NameValuePair> params) { StringBuilder sb = new StringBuilder(); for (int i = 0; i < params.size(); i++) { sb.append(params.get(i).getName()); sb.append('='); sb.append(params.get(i).getValue()); sb.append('&'); } sb.append("key="); sb.append(Constants.API_KEY); this.sb.append("sign str\n"+sb.toString()+"\n\n"); String appSign = MD5.getMessageDigest(sb.toString().getBytes()); Log.e("orion",appSign); return appSign; }</span>
差不多了 現(xiàn)在我們需要生成傳遞的參數(shù) 參數(shù)要求是xml 格式的:
<span style="font-size:14px;">private String genProductArgs() { StringBuffer xml = new StringBuffer(); String ip = getWifiIp(); if (ip == "" && ip == "") { ip = getLocalIpAddress(); } try { String nonceStr = genNonceStr(); xml.append("</xml>"); List<NameValuePair> packageParams = new LinkedList<NameValuePair>(); packageParams.add(new BasicNameValuePair("appid", Constants.APP_ID)); packageParams.add(new BasicNameValuePair("body", "APP pay test")); packageParams.add(new BasicNameValuePair("mch_id", Constants.MCH_ID)); packageParams.add(new BasicNameValuePair("nonce_str", nonceStr)); packageParams.add(new BasicNameValuePair("notify_url",ConfigUtil.NOTIFY_URL)); packageParams.add(new BasicNameValuePair("out_trade_no",genOutTradNo())); packageParams.add(new BasicNameValuePair("spbill_create_ip",ip)); packageParams.add(new BasicNameValuePair("total_fee", "1")); packageParams.add(new BasicNameValuePair("trade_type", "APP")); String sign = genPackageSign(packageParams); packageParams.add(new BasicNameValuePair("sign", sign)); String xmlstring =toXml(packageParams); return xmlstring; } catch (Exception e) { Log.e("TAG", "fail, ex = " + e.getMessage()); return null; } }</span>
其中 toxml:
<span style="font-size:14px;">private String toXml(List<NameValuePair> params) { StringBuilder sb = new StringBuilder(); sb.append("<xml>"); for (int i = 0; i < params.size(); i++) { sb.append("<"+params.get(i).getName()+">"); sb.append(params.get(i).getValue()); sb.append("</"+params.get(i).getName()+">"); } sb.append("</xml>"); Log.e("orion",sb.toString()); return sb.toString(); }</span>
得到傳遞的參數(shù),根據(jù)文檔指示,我們需要用post去吊連接URL地址:https://api.mch.weixin.qq.com/pay/unifiedorder:
<span style="font-size:14px;"> private class GetPrepayIdTask extends AsyncTask<Void, Void, Map<String,String>> { private ProgressDialog dialog; @Override protected void onPreExecute() { dialog = ProgressDialog.show(PayActivity.this, getString(R.string.app_tip), getString(R.string.getting_prepayid)); } @Override protected void onPostExecute(Map<String,String> result) { if (dialog != null) { dialog.dismiss(); } sb.append("prepay_id\n"+result.get("prepay_id")+"\n\n"); resultunifiedorder=result; } @Override protected void onCancelled() { super.onCancelled(); } @Override protected Map<String,String> doInBackground(Void... params) { String url = String.format("https://api.mch.weixin.qq.com/pay/unifiedorder"); String entity = genProductArgs(); Log.e("orion",entity); byte[] buf = Util.httpPost(url, entity); String content = new String(buf); Log.e("orion", content); Map<String,String> xml=decodeXml(content); return xml; } }</span>
其中decodexml就是:
<span style="font-size:14px;">public Map<String,String> decodeXml(String content) { try { Map<String, String> xml = new HashMap<String, String>(); XmlPullParser parser = Xml.newPullParser(); parser.setInput(new StringReader(content)); int event = parser.getEventType(); while (event != XmlPullParser.END_DOCUMENT) { String nodeName=parser.getName(); switch (event) { case XmlPullParser.START_DOCUMENT: break; case XmlPullParser.START_TAG: if("xml".equals(nodeName)==false){ //實例化student對象 xml.put(nodeName,parser.nextText()); } break; case XmlPullParser.END_TAG: break; } event = parser.next(); } return xml; } catch (Exception e) { Log.e("orion",e.toString()); } return null; }</span>
下單完成,第二部就是給微信支付傳遞調(diào)起微信支付的參數(shù)(具體參數(shù)看文檔說明):
<span style="font-size:14px;">private void genPayReq() { req.appId = Constants.APP_ID; req.partnerId = Constants.MCH_ID; req.prepayId = resultunifiedorder.get("prepay_id"); req.packageValue = "prepay_id="+resultunifiedorder.get("prepay_id"); req.nonceStr = genNonceStr(); req.timeStamp = String.valueOf(genTimeStamp()); List<NameValuePair> signParams = new LinkedList<NameValuePair>(); signParams.add(new BasicNameValuePair("appid", req.appId)); signParams.add(new BasicNameValuePair("noncestr", req.nonceStr)); signParams.add(new BasicNameValuePair("package", req.packageValue)); signParams.add(new BasicNameValuePair("partnerid", req.partnerId)); signParams.add(new BasicNameValuePair("prepayid", req.prepayId)); signParams.add(new BasicNameValuePair("timestamp", req.timeStamp)); req.sign = genAppSign(signParams); sb.append("sign\n"+req.sign+"\n\n"); Log.e("orion", signParams.toString()); }</span>
第三部,調(diào)微信支付:
<span style="font-size:14px;">private void sendPayReq() { msgApi.registerApp(Constants.APP_ID); msgApi.sendReq(req); }</span>
其中:
<span style="font-size:14px;">Constants.APP_ID</span>
是appid 在開發(fā)者平臺獲取
<span style="font-size:14px;"> req.partnerId = Constants.MCH_ID;</span>
商戶id
<span style="font-size:14px;">ConfigUtil.NOTIFY_URL</span>
支付后的回調(diào)通知地址。
簽名兩次,然后用的api——key是商戶平臺api安全里面自定義的。
對了 還有個獲取時間工具
<span style="font-size:14px;">private long genTimeStamp() { return System.currentTimeMillis() / 1000; }</span>
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android實現(xiàn)計步進(jìn)度的環(huán)形Progress
這篇文章主要為大家詳細(xì)介紹了Android實現(xiàn)計步進(jìn)度的環(huán)形Progress,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-02-02Android實戰(zhàn)打飛機(jī)游戲之怪物(敵機(jī))類的實現(xiàn)(4)
這篇文章主要為大家詳細(xì)介紹了Android實戰(zhàn)打飛機(jī)游戲之怪物(敵機(jī))類的實現(xiàn),具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-07-07Android 5.0中CoordinatorLayout的使用技巧
這篇文章主要為大家詳細(xì)介紹了Android 5.0中CoordinatorLayout的使用技巧,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-09-09Android自定義View展示W(wǎng)ifi信號強(qiáng)弱指示方法示例
這篇文章主要給大家介紹了關(guān)于Android自定義View展示W(wǎng)ifi信號強(qiáng)弱指示的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),文末給出了完整的實例供大家參考學(xué)習(xí),需要的朋友可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-08-08Android中關(guān)于CoordinatorLayout的一些實用布局技巧
大家都知道CoordinatorLayout是一個“加強(qiáng)版”的 FrameLayout,那么下面這篇文章主要給大家分享了Android中關(guān)于CoordinatorLayout的一些布局技巧,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來一起看看吧。2017-06-06