如何在Android App中接入微信支付
本篇簡(jiǎn)單介紹Android App中接入微信支付,包括App內(nèi)支付和掃碼支付。分享+支付 pofei
微信支付
App內(nèi)支付
主要流程:
1.微信支付平臺(tái)注冊(cè)賬號(hào)
注:注冊(cè)并申請(qǐng)成功以后,需要在API安全中設(shè)置你的API密鑰 32個(gè)字符。建議使用 MD5加密 ,并且需要妥善的保存。因?yàn)闊o(wú)法查看。
2.生成預(yù)支付訂單
3.生成簽名參數(shù)
4.調(diào)起微信,完成支付
掃碼支付
掃碼支付使用的是微信統(tǒng)一下單API ,使用的是模式二,模式一 一直說(shuō)URL參數(shù)錯(cuò)誤,完全按照官方文檔來(lái)的 令人費(fèi)解。
統(tǒng)一下單API
統(tǒng)一下單API
在上面的基礎(chǔ)上,修改
private String getProductArgs() { // TODO Auto-generated method stub StringBuffer xml=new StringBuffer(); try { String nonceStr=getNonceStr(); currentOrderId = getOutTrade(); xml.append("<xml>"); List<NameValuePair> packageParams=new LinkedList<NameValuePair>(); packageParams.add(new BasicNameValuePair("appid", WXConstants.APP_ID)); packageParams.add(new BasicNameValuePair("body", "APP pay test")); packageParams.add(new BasicNameValuePair("mch_id", WXConstants.MCH_ID)); packageParams.add(new BasicNameValuePair("nonce_str", nonceStr)); // 回調(diào) URL 地址,這里是第三方 packageParams.add(new BasicNameValuePair("notify_url", "http://www.weixunyunduan.com/yunduanwx/wxpay/getpackage")); // 商戶系統(tǒng)內(nèi)部訂單號(hào),要求32個(gè)字符 且同個(gè)商戶下唯一 packageParams.add(new BasicNameValuePair("out_trade_no", getNonceStr())); // APP和網(wǎng)頁(yè)支付提交用戶端,Native支付填調(diào)用微信支付API的機(jī)器IP packageParams.add(new BasicNameValuePair("spbill_create_ip", "192.168.0.1")); packageParams.add(new BasicNameValuePair("total_fee", "1")); // Native支付 packageParams.add(new BasicNameValuePair("trade_type", "NATIVE")); String sign=getPackageSign(packageParams); packageParams.add(new BasicNameValuePair("sign", sign)); String xmlString=toXml(packageParams); return xmlString; } catch (Exception e) { // TODO: handle exception return null; } } private String getOutTrade(){ return UUID.randomUUID().toString().replace("-", ""); }
NATIVE請(qǐng)求返回值如下:
<xml> <return_code><![CDATA[SUCCESS]]></return_code> <return_msg><![CDATA[OK]]></return_msg> <appid><![CDATA[]]></appid> <mch_id><![CDATA[]]></mch_id> <nonce_str><![CDATA[]]></nonce_str> <sign><![CDATA[]]></sign> <result_code><![CDATA[SUCCESS]]></result_code> <prepay_id><![CDATA[]]></prepay_id> <trade_type><![CDATA[NATIVE]]></trade_type> <code_url><![CDATA[weixin://wxpay/bizpayurl?pr=]></code_url> </xml>
獲取code_url,并使用第三方二維碼生成庫(kù) 如ZXing 生成二維碼。
ZXingUtils
import android.content.Context; import android.graphics.Bitmap; import android.graphics.Bitmap.Config; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.PointF; import android.view.Gravity; import android.view.View.MeasureSpec; import android.widget.LinearLayout; import android.widget.LinearLayout.LayoutParams; import android.widget.TextView; import com.google.zxing.BarcodeFormat; import com.google.zxing.EncodeHintType; import com.google.zxing.MultiFormatWriter; import com.google.zxing.WriterException; import com.google.zxing.common.BitMatrix; import com.google.zxing.qrcode.QRCodeWriter; import java.util.Hashtable; /** * * 生成條形碼和二維碼的工具 */ public class ZXingUtils { /** * 生成二維碼 要轉(zhuǎn)換的地址或字符串,可以是中文 * * @param url * @param width * @param height * @return */ public static Bitmap createQRImage(String url, final int width, final int height) { try { // 判斷URL合法性 if (url == null || "".equals(url) || url.length() < 1) { return null; } Hashtable<EncodeHintType, String> hints = new Hashtable<EncodeHintType, String>(); hints.put(EncodeHintType.CHARACTER_SET, "utf-8"); // 圖像數(shù)據(jù)轉(zhuǎn)換,使用了矩陣轉(zhuǎn)換 BitMatrix bitMatrix = new QRCodeWriter().encode(url, BarcodeFormat.QR_CODE, width, height, hints); int[] pixels = new int[width * height]; // 下面這里按照二維碼的算法,逐個(gè)生成二維碼的圖片, // 兩個(gè)for循環(huán)是圖片橫列掃描的結(jié)果 for (int y = 0; y < height; y++) { for (int x = 0; x < width; x++) { if (bitMatrix.get(x, y)) { pixels[y * width + x] = 0xff000000; } else { pixels[y * width + x] = 0xffffffff; } } } // 生成二維碼圖片的格式,使用ARGB_8888 Bitmap bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); bitmap.setPixels(pixels, 0, width, 0, 0, width, height); return bitmap; } catch (WriterException e) { e.printStackTrace(); } return null; } /** * 生成條形碼 * * @param context * @param contents * 需要生成的內(nèi)容 * @param desiredWidth * 生成條形碼的寬帶 * @param desiredHeight * 生成條形碼的高度 * @param displayCode * 是否在條形碼下方顯示內(nèi)容 * @return */ public static Bitmap creatBarcode(Context context, String contents, int desiredWidth, int desiredHeight, boolean displayCode) { Bitmap ruseltBitmap = null; /** * 圖片兩端所保留的空白的寬度 */ int marginW = 20; /** * 條形碼的編碼類(lèi)型 */ BarcodeFormat barcodeFormat = BarcodeFormat.CODE_128; if (displayCode) { Bitmap barcodeBitmap = encodeAsBitmap(contents, barcodeFormat, desiredWidth, desiredHeight); Bitmap codeBitmap = creatCodeBitmap(contents, desiredWidth + 2 * marginW, desiredHeight, context); ruseltBitmap = mixtureBitmap(barcodeBitmap, codeBitmap, new PointF( 0, desiredHeight)); } else { ruseltBitmap = encodeAsBitmap(contents, barcodeFormat, desiredWidth, desiredHeight); } return ruseltBitmap; } /** * 生成條形碼的Bitmap * * @param contents * 需要生成的內(nèi)容 * @param format * 編碼格式 * @param desiredWidth * @param desiredHeight * @return * @throws WriterException */ protected static Bitmap encodeAsBitmap(String contents, BarcodeFormat format, int desiredWidth, int desiredHeight) { final int WHITE = 0xFFFFFFFF; final int BLACK = 0xFF000000; MultiFormatWriter writer = new MultiFormatWriter(); BitMatrix result = null; try { result = writer.encode(contents, format, desiredWidth, desiredHeight, null); } catch (WriterException e) { // TODO Auto-generated catch block e.printStackTrace(); } int width = result.getWidth(); int height = result.getHeight(); int[] pixels = new int[width * height]; // All are 0, or black, by default for (int y = 0; y < height; y++) { int offset = y * width; for (int x = 0; x < width; x++) { pixels[offset + x] = result.get(x, y) ? BLACK : WHITE; } } Bitmap bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888); bitmap.setPixels(pixels, 0, width, 0, 0, width, height); return bitmap; } /** * 生成顯示編碼的Bitmap * * @param contents * @param width * @param height * @param context * @return */ protected static Bitmap creatCodeBitmap(String contents, int width, int height, Context context) { TextView tv = new TextView(context); LinearLayout.LayoutParams layoutParams = new LinearLayout.LayoutParams( LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT); tv.setLayoutParams(layoutParams); tv.setText(contents); tv.setHeight(height); tv.setGravity(Gravity.CENTER_HORIZONTAL); tv.setWidth(width); tv.setDrawingCacheEnabled(true); tv.setTextColor(Color.BLACK); tv.measure(MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED), MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED)); tv.layout(0, 0, tv.getMeasuredWidth(), tv.getMeasuredHeight()); tv.buildDrawingCache(); Bitmap bitmapCode = tv.getDrawingCache(); return bitmapCode; } /** * 將兩個(gè)Bitmap合并成一個(gè) * * @param first * @param second * @param fromPoint * 第二個(gè)Bitmap開(kāi)始繪制的起始位置(相對(duì)于第一個(gè)Bitmap) * @return */ protected static Bitmap mixtureBitmap(Bitmap first, Bitmap second, PointF fromPoint) { if (first == null || second == null || fromPoint == null) { return null; } int marginW = 20; Bitmap newBitmap = Bitmap.createBitmap( first.getWidth() + second.getWidth() + marginW, first.getHeight() + second.getHeight(), Config.ARGB_4444); Canvas cv = new Canvas(newBitmap); cv.drawBitmap(first, marginW, 0, null); cv.drawBitmap(second, fromPoint.x, fromPoint.y, null); cv.save(Canvas.ALL_SAVE_FLAG); cv.restore(); return newBitmap; } }
Bitmap bitmap = ZXingUtils.createQRImage(wxUrl,200,200);
code_url為微信可以識(shí)別的短鏈。
用戶掃描便可在手機(jī)上支付。
查詢(xún)訂單API
獲取支付回調(diào),使用查詢(xún)訂單API
String urlString="https://api.mch.weixin.qq.com/pay/orderquery"; CheckAsyncTask checkAsyncTask = new CheckAsyncTask(); checkAsyncTask.execute(urlString); private class CheckAsyncTask extends AsyncTask<String,Void, Map<String, String>> { private ProgressDialog dialog; @Override protected void onPreExecute() { // TODO Auto-generated method stub super.onPreExecute(); dialog = ProgressDialog.show(PayActivity.this, "提示", "正在查看訂單狀態(tài)!"); } @Override protected Map<String, String> doInBackground(String... params) { // TODO Auto-generated method stub String url=String.format(params[0]); String entity=getProductCheckArgs(); byte[] buf= wxUtils.httpPost(url, entity); String content = new String(buf); Map<String,String> xml=decodeXml(content); // 可以通過(guò) xml.get("trade_state"); 獲取訂單的狀態(tài) return xml; } @Override protected void onPostExecute(Map<String, String> result) { // TODO Auto-generated method stub super.onPostExecute(result); if (dialog != null) { dialog.dismiss(); } } }
以上就是如何在Android App中接入微信支付的詳細(xì)內(nèi)容,更多關(guān)于在Android App中接入微信支付的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android性能優(yōu)化之利用強(qiáng)大的LeakCanary檢測(cè)內(nèi)存泄漏及解決辦法
本篇文章主要介紹了Android性能優(yōu)化之利用LeakCanary檢測(cè)內(nèi)存泄漏及解決辦法,有興趣的同學(xué)可以了解一下。2016-11-11BroadcastReceiver動(dòng)態(tài)注冊(cè)案例詳解
這篇文章主要為大家詳細(xì)介紹了BroadcastReceiver動(dòng)態(tài)注冊(cè)案例,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08Android使用AlertDialog創(chuàng)建對(duì)話框
這篇文章主要為大家詳細(xì)介紹了Android使用AlertDialog創(chuàng)建對(duì)話框的方法料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12Android 個(gè)人理財(cái)工具三:添加賬單頁(yè)面 上
本文主要介紹Android 個(gè)人理財(cái)工具添加賬單頁(yè)面的功能實(shí)現(xiàn),這里提供實(shí)例代碼和實(shí)現(xiàn)效果圖,有興趣的小伙伴可以參考下2016-08-08Android判斷用戶2G/3G/4G移動(dòng)數(shù)據(jù)網(wǎng)絡(luò)
這篇文章主要介紹了Android判斷用戶2G/3G/4G移動(dòng)數(shù)據(jù)網(wǎng)絡(luò)的方法,感興趣的小伙伴們可以參考一下2015-12-12Android實(shí)現(xiàn)倒計(jì)時(shí)結(jié)束后跳轉(zhuǎn)頁(yè)面功能
最近在工作中遇到一個(gè)需求,需要在倒計(jì)時(shí)一段時(shí)間后進(jìn)行跳轉(zhuǎn)頁(yè)面,通過(guò)查找相關(guān)資料發(fā)現(xiàn)其中涉及的知識(shí)還不少,所以分享出來(lái),下面這篇文章主要給大家介紹了關(guān)于Android實(shí)現(xiàn)倒計(jì)時(shí)結(jié)束后跳轉(zhuǎn)頁(yè)面功能的相關(guān)資料,需要的朋友可以參考下。2017-11-11android2.3.5 CDMA/EVDO撥號(hào)APN解決方案
google提供的android2.3里面,只能在GSM/WCDMA情況下才能從“設(shè)置”->“無(wú)線和網(wǎng)絡(luò)”->“移動(dòng)網(wǎng)絡(luò)”->“接入點(diǎn)名稱(chēng)”中選擇不同的apn帳號(hào)進(jìn)行撥號(hào)連接,而CDMA/EVDO則沒(méi)有這個(gè)功能,接下來(lái)本文介紹一些方法實(shí)現(xiàn)這個(gè)功能,感興趣的朋友可以了解下2013-01-01Android圓形控件實(shí)現(xiàn)畫(huà)圓效果
這篇文章主要為大家詳細(xì)介紹了Android圓形控件實(shí)現(xiàn)畫(huà)圓效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-05-05