Android實(shí)現(xiàn)微信支付功能
開(kāi)發(fā)Android APP微信支付功能,需要完成三個(gè)步驟:第一步生成預(yù)支付訂單、第二步生成微信支付參數(shù)、第三步調(diào)起微信APP支付。除了需要審核通過(guò)的APP應(yīng)用外,還需要獲得微信支付接口權(quán)限,然后獲取對(duì)應(yīng)的商戶(hù)號(hào)、API密鑰,這兩者缺一不可,并且在APP微信支付中使用
獲得商戶(hù)號(hào)、API密鑰
在微信開(kāi)放平臺(tái)中查看審核通過(guò)的APP應(yīng)用,是否申請(qǐng)支付功能,若已申請(qǐng),登錄微信支付|商戶(hù)平臺(tái):http://pay.weixin.qq.com 查看對(duì)應(yīng)的商戶(hù)號(hào)、API密鑰
》申請(qǐng)微信支付接口
》登錄商戶(hù)平臺(tái)
》查看商戶(hù)號(hào)
》獲取API密鑰
第一步:生成預(yù)支付訂單
將商品信息、商戶(hù)信息使用BasicNameValuePair存放,然后存儲(chǔ)在List列表中,并構(gòu)造成xml字符串格式,以POST方式微信提供的接口:https://api.mch.weixin.qq.com/pay/unifiedorder 發(fā)送數(shù)據(jù),接收返回信息,獲取prepay_id預(yù)支付訂單編號(hào)
》生成訂單請(qǐng)求
private String genProductArgs() { StringBuffer xml = new StringBuffer(); 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", "weixin")); packageParams .add(new BasicNameValuePair("mch_id", Constants.MCH_ID)); packageParams.add(new BasicNameValuePair("nonce_str", nonceStr)); packageParams.add(new BasicNameValuePair("notify_url", "http://121.40.35.3/test")); packageParams.add(new BasicNameValuePair("out_trade_no", genOutTradNo())); packageParams.add(new BasicNameValuePair("spbill_create_ip", "127.0.0.1")); 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, "genProductArgs fail, ex = " + e.getMessage()); return null; } }
》獲取預(yù)支付訂單編號(hào)
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);
第二步:生成微信支付參數(shù)
appId對(duì)應(yīng)APP應(yīng)用的APPID,partnerId對(duì)應(yīng)商戶(hù)號(hào),prepayId對(duì)應(yīng)第一步獲得的預(yù)支付訂單,packageValue取值Sign=WXPay,nonceStr是一串隨機(jī)數(shù),timeStamp生成的時(shí)間戳,sign對(duì)應(yīng)APP簽名
》支付參數(shù)配置
private void genPayReq() { req.appId = Constants.APP_ID; req.partnerId = Constants.MCH_ID; req.prepayId = resultunifiedorder.get("prepay_id"); req.packageValue = "Sign=WXPay"; 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"); show.setText(sb.toString()); Log.e("orion", signParams.toString()); }
》nonceStr隨機(jī)數(shù)
private String genNonceStr() { Random random = new Random(); return MD5.getMessageDigest(String.valueOf(random.nextInt(10000)) .getBytes()); }
》timeStamp時(shí)間戳
private long genTimeStamp() { return System.currentTimeMillis() / 1000; }
》APP簽名
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()) .toUpperCase(); Log.e("orion", appSign); return appSign; }
第三步:調(diào)起微信APP支付
支付之前,需要手機(jī)安裝微信客戶(hù)端,如果沒(méi)有,會(huì)提示下載安裝微信,然后再調(diào)起支付,在AndroidManifest.xml文件中添加如下配置
<activity android:name=".act.PayActivity" android:exported="true" android:launchMode="singleTop" > <intent-filter> <action android:name="android.intent.action.VIEW" /> <category android:name="android.intent.category.DEFAULT" /> <data android:scheme="wxf2f565574a968187" /> </intent-filter> </activity>
》調(diào)起微信支付
private void sendPayReq() { msgApi.registerApp(Constants.APP_ID); msgApi.sendReq(req); }
接收微信返回結(jié)果
在實(shí)現(xiàn)了IWXAPIEventHandler接口的Activity中重寫(xiě)onResp(BaseResp)方法,接收微信結(jié)果
》接收支付結(jié)果
public void onResp(BaseResp resp) { Log.d(TAG, "onPayFinish, errCode = " + resp.errCode); if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX) { AlertDialog.Builder builder = new AlertDialog.Builder(this); builder.setTitle(R.string.app_tip); builder.setMessage(getString(R.string.pay_result_callback_msg, resp.errStr + ";code=" + String.valueOf(resp.errCode))); builder.show(); } }
本文已被整理到了《Android微信支付教程匯總》,《Android微信開(kāi)發(fā)教程匯總》,歡迎大家學(xué)習(xí)閱讀。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助。
相關(guān)文章
RecyclerView設(shè)置間距和添加分割線的方法
在使用RecyclerView布局,經(jīng)常需要調(diào)整間距和添加分割線以達(dá)到更美觀效果,這篇文章主要介紹了RecyclerView設(shè)置間距和添加分割線的方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-09-09Android實(shí)現(xiàn)擴(kuò)展Menu的方法
這篇文章主要介紹了Android實(shí)現(xiàn)擴(kuò)展Menu的方法,涉及Android操作menu菜單的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10基于startActivityForResult方法處理兩個(gè)Activity之間數(shù)據(jù)傳遞問(wèn)題
這篇文章主要介紹了基于startActivityForResult方法處理兩個(gè)Activity之間數(shù)據(jù)傳遞問(wèn)題的相關(guān)資料,需要的朋友可以參考下2015-11-11ListView的Adapter使用 之 初學(xué)ArrayAdapter String
ListView是Android中經(jīng)常會(huì)使用的東西,綁定數(shù)據(jù)對(duì)于初學(xué)者來(lái)說(shuō),尤其是剛接觸編程的人來(lái)說(shuō),往往會(huì)覺(jué)得很難理解,我上大二的時(shí)候?qū)W的java,但是基本上相當(dāng)于沒(méi)有學(xué),什么都沒(méi)寫(xiě)過(guò),真正接觸編程就是開(kāi)始上手學(xué)android,把這些記錄下來(lái),自己可以回頭看下,也可以讓新手更好的理解2013-06-06Android獲取內(nèi)置sdcard跟外置sdcard路徑
這篇文章主要介紹了Android獲取內(nèi)置sdcard跟外置sdcard路徑的相關(guān)資料,希望通過(guò)本文能幫助到大家,需要的朋友可以參考下2017-09-09Android實(shí)現(xiàn)縮放動(dòng)畫(huà)
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)縮放動(dòng)畫(huà),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07