1. 程式人生 > >支付寶API使用流程

支付寶API使用流程

一、準備工作

〉1、下載開發包

image

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

〉2、建立支付寶應用

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

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

〉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私鑰轉換成PKCS8格式 

生成RSA公鑰

image

開發者的私鑰

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

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

◆開發者的公鑰

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

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

3、儲存到一個臨時的記事本中。

引數設定完成

〉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;
			}
		};
	};
4、其它功能程式碼
/**
	 * 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();
	}
具體可參照支付寶Demo 及《接入與使用規則》