亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

如何在Android App中接入微信支付

 更新時(shí)間:2021年03月18日 09:57:29   作者:可愛(ài)的黑精靈  
這篇文章主要介紹了如何在Android App中接入微信支付,幫助大家更好的理解和學(xué)習(xí)使用Android開(kāi)發(fā),感興趣的朋友可以了解下

本篇簡(jiǎn)單介紹Android App中接入微信支付,包括App內(nèi)支付和掃碼支付。分享+支付 pofei

微信支付

wechat 官方接入文檔

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

查詢(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防止按鈕重復(fù)點(diǎn)擊示例代碼

    Android防止按鈕重復(fù)點(diǎn)擊示例代碼

    本文介紹封裝的一個(gè)小的工具類(lèi)庫(kù),按鈕點(diǎn)擊事件類(lèi)庫(kù),該類(lèi)庫(kù)可以防止按鈕重復(fù)點(diǎn)擊,可以判斷網(wǎng)絡(luò)狀態(tài),可以判斷用戶登錄態(tài),以及自定義驗(yàn)證條件等等,有興趣的可以了解一下
    2018-05-05
  • Android性能優(yōu)化之利用強(qiáng)大的LeakCanary檢測(cè)內(nèi)存泄漏及解決辦法

    Android性能優(yōu)化之利用強(qiáng)大的LeakCanary檢測(cè)內(nèi)存泄漏及解決辦法

    本篇文章主要介紹了Android性能優(yōu)化之利用LeakCanary檢測(cè)內(nèi)存泄漏及解決辦法,有興趣的同學(xué)可以了解一下。
    2016-11-11
  • BroadcastReceiver動(dòng)態(tài)注冊(cè)案例詳解

    BroadcastReceiver動(dòng)態(tài)注冊(cè)案例詳解

    這篇文章主要為大家詳細(xì)介紹了BroadcastReceiver動(dòng)態(tài)注冊(cè)案例,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-08-08
  • Android使用AlertDialog創(chuàng)建對(duì)話框

    Android使用AlertDialog創(chuàng)建對(duì)話框

    這篇文章主要為大家詳細(xì)介紹了Android使用AlertDialog創(chuàng)建對(duì)話框的方法料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • Android 個(gè)人理財(cái)工具三:添加賬單頁(yè)面 上

    Android 個(gè)人理財(cái)工具三:添加賬單頁(yè)面 上

    本文主要介紹Android 個(gè)人理財(cái)工具添加賬單頁(yè)面的功能實(shí)現(xiàn),這里提供實(shí)例代碼和實(shí)現(xiàn)效果圖,有興趣的小伙伴可以參考下
    2016-08-08
  • Android判斷用戶2G/3G/4G移動(dòng)數(shù)據(jù)網(wǎng)絡(luò)

    Android判斷用戶2G/3G/4G移動(dòng)數(shù)據(jù)網(wǎng)絡(luò)

    這篇文章主要介紹了Android判斷用戶2G/3G/4G移動(dòng)數(shù)據(jù)網(wǎng)絡(luò)的方法,感興趣的小伙伴們可以參考一下
    2015-12-12
  • Android開(kāi)發(fā)改變字體顏色方法

    Android開(kāi)發(fā)改變字體顏色方法

    用以下方法基本上可以解決大多數(shù)字體顏色設(shè)置問(wèn)題,先發(fā)一篇,后續(xù)繼續(xù)發(fā)一篇高級(jí)的,通過(guò)用戶選擇的方式,改變字體顏色。
    2015-05-05
  • Android實(shí)現(xiàn)倒計(jì)時(shí)結(jié)束后跳轉(zhuǎn)頁(yè)面功能

    Android實(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-11
  • android2.3.5 CDMA/EVDO撥號(hào)APN解決方案

    android2.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-01
  • Android圓形控件實(shí)現(xiàn)畫(huà)圓效果

    Android圓形控件實(shí)現(xiàn)畫(huà)圓效果

    這篇文章主要為大家詳細(xì)介紹了Android圓形控件實(shí)現(xiàn)畫(huà)圓效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-05-05

最新評(píng)論