天天看點

Android 支付寶API使用流程

一、準備工作

〉1、下載下傳開發包

https://b.alipay.com/order/productDetail.htm?productId=2014110308141993&tabId=4#ps-tabinfo-hash

Android 支付寶API使用流程

壓縮包下的“支付寶錢包支付接口開發包”中即有Andoid使用支付寶的JAR和Demo

〉2、建立支付寶應用

在支付寶開放平台申請建立應用

https://open.alipay.com/index.htm

支付寶平台的應用建立僅限于公司實名認證使用者,個人帳号是不能建立應用的,應用申請我沒試過,暫且不做說明

〉3、AndroidManifest.xml 修改 (權限、界面、服務等申明)

<activity
android:name="com.alipay.sdk.app.H5PayActivity"
android:configChanges="orientation|keyboardHidden|navigation"
android:exported="false"
android:screenOrientation="behind" >
</activity>
<activity
android:name="com.alipay.sdk.auth.AuthActivity"
android:configChanges="orientation|keyboardHidden|navigation"
android:exported="false"
android:screenOrientation="behind" >
</activity>
           
<uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
           

>4、擷取開發所需要的參數資料

//商戶PID
	public static final String PARTNER = "";
	//商戶收款賬号
	public static final String SELLER = "";
	//商戶私鑰,pkcs8格式
	public static final String RSA_PRIVATE = "";
	//支付寶公鑰
	public static final String RSA_PUBLIC = "";
           

商戶PID和商戶收款賬号 都是在支付寶申請上都有

商戶私鑰和支付寶公鑰都是需要通過支付寶開發包中的openssl檔案夾下的bin目錄下的OpenSSL程式生成

指令如下

RSA密鑰生成指令
生成RSA私鑰
openssl>genrsa -out rsa_private_key.pem 1024
生成RSA公鑰
openssl>rsa -in rsa_private_key.pem -pubout -out rsa_public_key.pem
将RSA私鑰轉換成PKCS8格式
openssl>pkcs8 -topk8 -nocrypt -inform PEM -in rsa_private_key.pem -outform PEM outform
           

生成RSA公鑰

Android 支付寶API使用流程

注:開發者的私鑰

1、必須保證隻有一行文字,即,沒有回車、換行、空格等

2、去掉“—–BEGIN RSA PRIVATE KEY—–”、“—–END RSA PRIVATE KEY—–”,隻儲存這兩條文字之中的部分

參數設定完成

〉5、加載Jar檔案

将開發包中的alipay-sdk-common檔案夾下面的jar檔案複制到項目的libs目錄下并加載

〉6、添加混淆規則

-libraryjars libs/alipaysdk.jar
-libraryjars libs/alipaysecsdk.jar
-libraryjars libs/alipayutdid.jar
-keep class com.alipay.android.app.IAlixPay{*;}
-keep class com.alipay.android.app.IAlixPay$Stub{*;}
-keep class com.alipay.android.app.IRemoteServiceCallback{*;}
-keep class com.alipay.android.app.IRemoteServiceCallback$Stub{*;}
-keep class com.alipay.sdk.app.PayTask{ public *;}
-keep class com.alipay.sdk.app.AuthTask{ public *;}
-keep class com.alipay.mobilesecuritysdk.*
-keep class com.ut.*
           

二、開發

1、建立訂單資訊

/**
	 * create the order info. 建立訂單資訊
	 * 
	 */
	public String getOrderInfo(String subject, String body, String price) {
		// 簽約合作者身份ID
		String orderInfo = "partner=" + "\"" + PARTNER + "\"";

		// 簽約賣家支付寶賬号
		orderInfo += "&seller_id=" + "\"" + SELLER + "\"";

		// 商戶網站唯一訂單号
		orderInfo += "&out_trade_no=" + "\"" + getOutTradeNo() + "\"";

		// 商品名稱
		orderInfo += "&subject=" + "\"" + subject + "\"";

		// 商品詳情
		orderInfo += "&body=" + "\"" + body + "\"";

		// 商品金額
		orderInfo += "&total_fee=" + "\"" + price + "\"";

		// 伺服器異步通知頁面路徑
		orderInfo += "&notify_url=" + "\"" + "http://notify.msp.hk/notify.htm"
				+ "\"";

		// 服務接口名稱, 固定值
		orderInfo += "&service=\"mobile.securitypay.pay\"";

		// 支付類型, 固定值
		orderInfo += "&payment_type=\"1\"";

		// 參數編碼, 固定值
		orderInfo += "&_input_charset=\"utf-8\"";

		// 設定未付款交易的逾時時間
		// 預設30分鐘,一旦逾時,該筆交易就會自動被關閉。
		// 取值範圍:1m~15d。
		// m-分鐘,h-小時,d-天,1c-當天(無論交易何時建立,都在0點關閉)。
		// 該參數數值不接受小數點,如1.5h,可轉換為90m。
		orderInfo += "&it_b_pay=\"30m\"";

		// extern_token為經過快登授權擷取到的alipay_open_id,帶上此參數使用者将使用授權的賬戶進行支付
		// orderInfo += "&extern_token=" + "\"" + extern_token + "\"";

		// 支付寶處理完請求後,目前頁面跳轉到商戶指定頁面的路徑,可空
		orderInfo += "&return_url=\"m.alipay.com\"";

		// 調用銀行卡支付,需配置此參數,參與簽名, 固定值 (需要簽約《無線銀行卡快捷支付》才能使用)
		// orderInfo += "&paymethod=\"expressGateway\"";

		return orderInfo;
	}
           

2、調用SDK支付

/**
	 * call alipay sdk pay. 調用SDK支付
	 * 
	 */
	public void pay(View v) {
		// 訂單
		String orderInfo = getOrderInfo("測試的商品", "該測試商品的較長的描述", "0.01");

		// 對訂單做RSA 簽名
		String sign = sign(orderInfo);
		try {
			// 僅需對sign 做URL編碼
			sign = URLEncoder.encode(sign, "UTF-8");
		} catch (UnsupportedEncodingException e) {
			e.printStackTrace();
		}

		// 完整的符合支付寶參數規範的訂單資訊
		final String payInfo = orderInfo + "&sign=\"" + sign + "\"&"
				+ getSignType();

		Runnable payRunnable = new Runnable() {

			@Override
			public void run() {
				// 構造PayTask 對象
				PayTask alipay = new PayTask(PayDemoActivity.this);
				// 調用支付接口,擷取支付結果
				String result = alipay.pay(payInfo);

				Message msg = new Message();
				msg.what = SDK_PAY_FLAG;
				msg.obj = result;
				mHandler.sendMessage(msg);
			}
		};

		// 必須異步調用
		Thread payThread = new Thread(payRunnable);
		payThread.start();
	}
           

3、支付結果擷取和處理

private Handler mHandler = new Handler() {
		public void handleMessage(Message msg) {
			switch (msg.what) {
			case SDK_PAY_FLAG: {
				PayResult payResult = new PayResult((String) msg.obj);
				
				// 支付寶傳回此次支付結果及加簽,建議對支付寶簽名資訊拿簽約時支付寶提供的公鑰做驗簽
				String resultInfo = payResult.getResult();
				
				String resultStatus = payResult.getResultStatus();

				// 判斷resultStatus 為“9000”則代表支付成功,具體狀态碼代表含義可參考接口文檔
				if (TextUtils.equals(resultStatus, "9000")) {
					Toast.makeText(PayDemoActivity.this, "支付成功",
							Toast.LENGTH_SHORT).show();
				} else {
					// 判斷resultStatus 為非“9000”則代表可能支付失敗
					// “8000”代表支付結果因為支付管道原因或者系統原因還在等待支付結果确認,最終交易是否成功以服務端異步通知為準(小機率狀态)
					if (TextUtils.equals(resultStatus, "8000")) {
						Toast.makeText(PayDemoActivity.this, "支付結果确認中",
								Toast.LENGTH_SHORT).show();
					} else {
						// 其他值就可以判斷為支付失敗,包括使用者主動取消支付,或者系統傳回的錯誤
						Toast.makeText(PayDemoActivity.this, "支付失敗",
								Toast.LENGTH_SHORT).show();
					}
				}
				break;
			}
			case SDK_CHECK_FLAG: {
				Toast.makeText(PayDemoActivity.this, "檢查結果為:" + msg.obj,
						Toast.LENGTH_SHORT).show();
				break;
			}
			default:
				break;
			}
		};
	};
           
/**
	 * check whether the device has authentication alipay account.
	 * 查詢終端裝置是否存在支付寶認證賬戶
	 * 
	 */
	public void check(View v) {
		Runnable checkRunnable = new Runnable() {

			@Override
			public void run() {
				// 構造PayTask 對象
				PayTask payTask = new PayTask(PayDemoActivity.this);
				// 調用查詢接口,擷取查詢結果
				boolean isExist = payTask.checkAccountIfExist();

				Message msg = new Message();
				msg.what = SDK_CHECK_FLAG;
				msg.obj = isExist;
				mHandler.sendMessage(msg);
			}
		};

		Thread checkThread = new Thread(checkRunnable);
		checkThread.start();

	}
           
/**
	 * get the sdk version. 擷取SDK版本号
	 * 
	 */
	public void getSDKVersion() {
		PayTask payTask = new PayTask(this);
		String version = payTask.getVersion();
		Toast.makeText(this, version, Toast.LENGTH_SHORT).show();
	}
           

繼續閱讀