本次講述的是關於如何在本身app
中集成支付寶的支付功能的解說性文章,因爲此類文章過多,因此我想一想仍是講一些看完這些文章以後,本身的想法跟套路,這也更貼近跟我同樣第一次看別人教程的人的思惟,讓咱們開始吧。java
->1 下載開發包
https://b.alipay.com/order/productDetail.htm?productId=2014110308141993&tabId=4#ps-tabinfo-hashexpress
下載壓縮包,並將裏面的Android使用支付寶的JAR
和Demo
解壓出來。->2 簽約移動支付
在支付寶開放平臺申請簽約移動支付(這步驟須要公司的執照證件,因此須要運營配合註冊)
https://open.alipay.com/index.htm
支付寶平臺的移動支付簽約僅限於公司實名認證用戶
該步驟的目的是爲了獲取商戶PID
和商戶收款帳號
,商戶公鑰的寫入配置
,以及查看支付寶公鑰
,這四個細節。->3 AndriodMainfest.xml增長(所需權限,界面註冊)
服務器
切記配置,否則有些莫名其妙的問題就會找上門,不要錯的渾然不知,留意配置文件,細心!微信
->4 獲取開發所須要的四個參數
app
商戶PID
和商戶收款帳號
都是在支付寶申請上都有商戶私鑰
和商戶公鑰
都是須要通ssl
文件夾下的bin
目錄下的OpenSSL
程序生成
(特別注意MAC
自帶openssl
),支付寶公鑰
是全部應用都同樣,在支付寶申請裏面點擊查看支付寶公鑰裏面就有,也沒有什麼卵用,商戶公鑰
是要在支付寶申請裏面放入,來呼應app
中的私鑰,從而驗證一致有效性。
命令以下:異步
生成私鑰圖ide
將RSA私鑰
轉換成PKCS8
格式(特別注意,在java開發環境
它要的商戶私鑰
是PKCS8
格式的,不要直接用原始
的私鑰)測試
生成RSA
公鑰(就是商戶公鑰
,要在支付寶申請裏面配置進去,這樣私鑰跟公鑰才能一塊兒驗證有效性)網站
開發者的私鑰
一、必須保證只有一行文字,即,沒有回車、換行、空格等
二、去掉「-----BEGIN RSA PRIVATE KEY-----」
、「-----END RSA PRIVATE KEY-----」
,只保存這兩條文字之中的部分ui
◆開發者的公鑰
一、必須保證只有一行文字,即,沒有回車、換行、空格等
二、去掉「-----BEGIN PUBLIC KEY-----」
、「-----END PUBLIC KEY-----」
,只保存這兩條文字之中的部分
三、保存到一個臨時的記事本中。
參數設置完成->5 加載Jar文件
將開發包中的alipay-sdk-common
文件夾下面的jar
文件複製到項目的libs
目錄下並加載
->六、添加混淆規則
一、建立訂單信息
/** * 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; }
二、調用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(); }
三、支付結果獲取和處理
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(); }
關於支付寶的配置其實就是這些,只要細心不出錯,功能就能實現,那麼究竟它是如何結合咱們本身的業務來完成對應的功能呢,待我細細道來,支付寶的支付功能的本質是調用咱們手機用戶的支付寶去往商戶的支付寶付款,而不是直接往有些應用中的帳號充值。之因此咱們能經過支付寶付款的方式,來對本身應用的帳戶充值,或者買到對應的商品,這真正的玄機就在
這個建立訂單中服務器異步通知頁面,它會在支付寶付款成功或失敗的時候發送訂單信息以及是否支付成功信息一塊兒給服務器(注意這個服務器就是本身的業務服務器,這裏根據支付寶的反饋過來的支付是否成功信息和訂單信息,來後臺完成業務的實現)這也就完成了。
由於不少步驟性的操做無法避免,因此我這邊文章也借鑑了一些他人的文章,彙總了一些看完文章後本身疑惑的地方,而後研究,理解,從而完善起來,應該會對你們有些許的幫助(PS:官方的教程其實比較全,可是看起來也是有點累,因此寫這篇實戰性的文章,減小你們的閱讀量,快速理解每一步的用意,以及一些易犯錯的細節提早預防)。
最後奉上個人我的技術訂閱號,鏘哥的覺悟,微信號:DY_suixincq
個人博客最新文章會在這裏同步推出,歡迎關注交流