一、准備工作
〉1、下載開發包
https://b.alipay.com/order/productDetail.htm?productId=2014110308141993&tabId=4#ps-tabinfo-hash
壓縮包下的“支付寶錢包支付接口開發包”中即有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私鑰轉換成PKCS8格式
生成RSA公鑰
開發者的私鑰
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 += "¬ify_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 及《接入與使用規則》