在網上搜索了以下,有很多這方面的教程,但大部分教程過於陳舊,而且描述的過於簡單。而且支付寶提供的接口一直在更新,可能支付寶那邊是爲了讓接口更容易被調用吧,以前有些老的教程稍微跟現在接口有些不能“對號入座”,於是,我決定抽空寫一篇關於調用支付寶接口的文章,跟大家分享,讓大家以最快的速度掌握如何調用支付寶接口的方法。如果寫的不好,請大家多多指教哦。
不多說了,開寫。
1,到支付寶官網,下載支付寶集成開發包,看懂裏面的關鍵代碼
由於android設備一般用的都是無線支付,所有我們申請的就是支付寶無線快捷支付接口。下面是申請的地址,下面還有下載接口開發包的地方
https://b.alipay.com/order/productDetail.htm?productId=2013080604609654
如果鏈接失效,你可以到支付寶官網商家服務模塊中找到 快捷支付(無線)這個服務。
下載集成開發包,解壓發現裏面有客戶端的demo即說明文檔,在客戶端的demo中找到Android_SDK,這個就是你要用到的支付寶接口及demo。
把demo(alipay_sdk_demo)和(alipay_lib)導入到你的eclipse裏面,然後你可以試着運行一遍demo(alipay_sdk_demo),只要把這個demo搞懂了,你就會調用這個支付寶接口了。至於到時候如何集成到你的項目裏面,文檔上說明很詳細,按着文檔上一步一步來就行了。我的建議是先把這個demo弄懂再設計你的項目,看看調用接口時需要哪些數據,這樣也有利於你一開始設計數據。
下面來簡單的介紹下接口demo裏面的結構。
你打開項目會發現裏面有5個類。
其中,Base64.java、Result.java、Rsa.java這3個類不需要動,需要用到的,到時候直接放到你項目裏面就行了,如果你想了解下它們你也可以看看。重點就是ExternalPartner.java和Keys.java這兩個類。先來說說Keys.java類吧。打開你會發現這個類很簡單,如下所示:
- package com.alipay.android.msp.demo;
- //
- // 請參考 Android平臺安全支付服務(msp)應用開發接口(4.2 RSA算法簽名)部分,並使用壓縮包中的openssl RSA密鑰生成工具,生成一套RSA公私鑰。
- // 這裏簽名時,只需要使用生成的RSA私鑰。
- // Note: 爲安全起見,使用RSA私鑰進行簽名的操作過程,應該儘量放到商家服務器端去進行。
- public final class Keys {
- //合作身份者id,以2088開頭的16位純數字,這個你申請支付寶簽約成功後就會看見
- public static final String DEFAULT_PARTNER = "";
- //這裏填寫收款支付寶賬號,即你付款後到賬的支付寶賬號
- public static final String DEFAULT_SELLER = "";
- //商戶私鑰,自助生成,即rsa_private_key.pem中去掉首行,最後一行,空格和換行最後拼成一行的字符串,rsa_private_key.pem這個文件等你申請支付寶簽約成功後,按照文檔說明你會生成的.........................如果android版本太高,這裏要用PKCS8格式用戶私鑰,不然調用不會成功的,那個格式你到時候會生成的,表急。
- public static final String PRIVATE = "";
- //支付寶(RSA)公鑰 ,demo自帶不用改,或者用簽約支付寶賬號登錄ms.alipay.com後,在密鑰管理頁面獲取;或者文檔上也有。
- public static final String PUBLIC = "MIGfMA0GCSqGSIb3...................";
- }
然後是ExternalPartner.java這個類。這個類其實也很簡單,你運行這個demo會發現就是一個listview裏面加載的很多商品信息,點擊一款商品就調用會支付寶接口(因爲Key.java未配置,所以你現在估計調用不成功)。我們找到listview的點擊事件的方法就找到調用支付寶的入口了,這個類的代碼如下:
- package com.alipay.android.msp.demo;
- import java.io.IOException;
- ........
- ........
- import com.alipay.android.app.sdk.AliPay;
- public class ExternalPartner extends Activity implements OnItemClickListener,
- OnClickListener {
- public static final String TAG = "alipay-sdk";
- private static final int RQF_PAY = 1;
- private static final int RQF_LOGIN = 2;
- private EditText mUserId;
- private Button mLogon;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.external_partner);
- initProducts();
- initListView();
- }
- /*
- * (non-Javadoc)
- *
- * @see android.app.Activity#onCreateOptionsMenu(android.view.Menu)
- */
- @Override
- public boolean onCreateOptionsMenu(Menu menu) {
- menu.add(Menu.NONE, Menu.FIRST, 1, "快速登錄");
- return true;
- }
- /*
- * (non-Javadoc)
- *
- * @see android.app.Activity#onOptionsItemSelected(android.view.MenuItem)
- */
- @Override
- public boolean onOptionsItemSelected(MenuItem item) {
- switch (item.getItemId()) {
- case Menu.FIRST:
- setContentView(R.layout.trustlogin);
- mUserId = (EditText) findViewById(R.id.user_id);
- mLogon = (Button) findViewById(R.id.get_token);
- mLogon.setOnClickListener(this);
- break;
- }
- return false;
- }
- private void initProducts() {
- if (sProducts != null)
- return;
- XmlResourceParser parser = getResources().getXml(R.xml.products);
- ArrayList<Product> products = new ArrayList<Product>();
- Product product = null;
- try {
- int eventType = parser.getEventType();
- while (eventType != XmlPullParser.END_DOCUMENT) {
- if (eventType == XmlPullParser.START_TAG
- && parser.getName().equalsIgnoreCase("product")) {
- product = new Product();
- product.subject = parser.getAttributeValue(0);
- product.body = parser.getAttributeValue(1);
- product.price = parser.getAttributeValue(2);
- products.add(product);
- }
- eventType = parser.next();
- }
- sProducts = new Product[products.size()];
- products.toArray(sProducts);
- } catch (XmlPullParserException e) {
- e.printStackTrace();
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- //listview點擊事件,裏面調用的支付寶接口
- @Override
- public void onItemClick(AdapterView<?> arg0, View arg1, int position,
- long arg3) {
- try {
- Log.i("ExternalPartner", "onItemClick");
- String info = getNewOrderInfo(position);//這個是訂單信息
- String sign = Rsa.sign(info, Keys.PRIVATE);//簽名加密訂單信息什麼的
- sign = URLEncoder.encode(sign);
- info += "&sign=\"" + sign + "\"&" + getSignType();
- Log.i("ExternalPartner", "start pay");
- // start the pay.
- Log.i(TAG, "info = " + info);
- final String orderInfo = info;
- new Thread() {
- public void run() {
- AliPay alipay = new AliPay(ExternalPartner.this, mHandler);//這個應該就是支付寶接口了,哈哈,支付寶現在把很多功能都封裝了,所以省了很多代碼
- //設置爲沙箱模式,不設置默認爲線上環境
- //alipay.setSandBox(true);
- String result = alipay.pay(orderInfo);//這個是返回的結果,你到時候可以根據這個結果加以操作你想操作的,然後基本就完了,其他的你想附加的功能你看着寫吧,現在Key.java配置好就能調用快捷支付了
- //後面的這些代碼可以改成你自己的,也可以在它們的基礎上改
- Log.i(TAG, "result = " + result);
- Message msg = new Message();
- msg.what = RQF_PAY;
- msg.obj = result;
- mHandler.sendMessage(msg);
- }
- }.start();
- } catch (Exception ex) {
- ex.printStackTrace();
- Toast.makeText(ExternalPartner.this, R.string.remote_call_failed,
- Toast.LENGTH_SHORT).show();
- }
- }
- //獲得訂單信息的方法
- private String getNewOrderInfo(int position) {
- StringBuilder sb = new StringBuilder();
- sb.append("partner=\"");
- sb.append(Keys.DEFAULT_PARTNER);// 合作身份者id
- sb.append("\"&out_trade_no=\"");
- sb.append(getOutTradeNo());//這個是訂單編號
- sb.append("\"&subject=\"");
- sb.append(sProducts[position].subject);//這個應該是商品名稱
- sb.append("\"&body=\"");
- sb.append(sProducts[position].body);//這個應該是商品的描述,具體你可以參考demo
- sb.append("\"&total_fee=\"");
- sb.append(sProducts[position].price.replace("一口價:", ""));//這個是要付款的金額,到時候你調用的時候改下就行了
- sb.append("\"¬ify_url=\"");
- // 網址需要做URL編碼
- sb.append(URLEncoder.encode("http://notify.java.jpxx.org/index.jsp"));//服務器異步通知頁面,完成交易後通知商家服務器的頁面,以post的形式將商品訂單信息發送到指定頁面,手機客戶端不需要可以先放在這不管
- sb.append("\"&service=\"mobile.securitypay.pay");
- sb.append("\"&_input_charset=\"UTF-8");
- sb.append("\"&return_url=\"");
- sb.append(URLEncoder.encode("http://m.alipay.com"));
- sb.append("\"&payment_type=\"1");
- sb.append("\"&seller_id=\"");
- sb.append(Keys.DEFAULT_SELLER);
- // 如果show_url值爲空,可不傳
- // sb.append("\"&show_url=\"");
- sb.append("\"&it_b_pay=\"1m");
- sb.append("\"");
- return new String(sb);
- }
- private String getOutTradeNo() {
- SimpleDateFormat format = new SimpleDateFormat("MMddHHmmss");
- Date date = new Date();
- String key = format.format(date);
- java.util.Random r = new java.util.Random();
- key += r.nextInt();
- key = key.substring(0, 15);
- Log.d(TAG, "outTradeNo: " + key);
- return key;
- }
- private String getSignType() {
- return "sign_type=\"RSA\"";
- }
- private void initListView() {
- ListView lv = (ListView) findViewById(R.id.list_view);
- lv.setAdapter(new ExternalPartnerAdapter());
- lv.setOnItemClickListener(this);
- }
- private void doLogin() {
- final String orderInfo = getUserInfo();
- new Thread() {
- public void run() {
- String result = new AliPay(ExternalPartner.this, mHandler)
- .pay(orderInfo);
- Log.i(TAG, "result = " + result);
- Message msg = new Message();
- msg.what = RQF_LOGIN;
- msg.obj = result;
- mHandler.sendMessage(msg);
- }
- }.start();
- }
- private String getUserInfo() {
- String userId = mUserId.getText().toString();
- return trustLogin(Keys.DEFAULT_PARTNER, userId);
- }
- private String trustLogin(String partnerId, String appUserId) {
- StringBuilder sb = new StringBuilder();
- sb.append("app_name=\"mc\"&biz_type=\"trust_login\"&partner=\"");
- sb.append(partnerId);
- Log.d("TAG", "UserID = " + appUserId);
- if (!TextUtils.isEmpty(appUserId)) {
- appUserId = appUserId.replace("\"", "");
- sb.append("\"&app_id=\"");
- sb.append(appUserId);
- }
- sb.append("\"");
- String info = sb.toString();
- // 請求信息簽名
- String sign = Rsa.sign(info, Keys.PRIVATE);
- try {
- sign = URLEncoder.encode(sign, "UTF-8");
- } catch (UnsupportedEncodingException e) {
- e.printStackTrace();
- }
- info += "&sign=\"" + sign + "\"&" + getSignType();
- return info;
- }
- @Override
- public void onClick(View v) {
- if (v instanceof Button) {
- switch (v.getId()) {
- case R.id.get_token:
- doLogin();
- break;
- }
- }
- }
- private class ExternalPartnerAdapter extends BaseAdapter {
- @Override
- public int getCount() {
- return sProducts.length;
- }
- @Override
- public Object getItem(int arg0) {
- return sProducts[arg0];
- }
- @Override
- public long getItemId(int position) {
- return position;
- }
- @Override
- public View getView(int position, View convertView, ViewGroup parent) {
- if (convertView == null) {
- LayoutInflater factory = LayoutInflater
- .from(ExternalPartner.this);
- convertView = factory.inflate(R.layout.product_item, null);
- }
- Product product = (Product) getItem(position);
- TextView tv = (TextView) convertView.findViewById(R.id.subject);
- tv.setText(product.subject);
- tv = (TextView) convertView.findViewById(R.id.body);
- tv.setText(product.body);
- tv = (TextView) convertView.findViewById(R.id.price);
- tv.setText(product.price);
- return convertView;
- }
- }
- Handler mHandler = new Handler() {
- public void handleMessage(android.os.Message msg) {
- Result result = new Result((String) msg.obj);
- switch (msg.what) {
- case RQF_PAY:
- case RQF_LOGIN: {
- Toast.makeText(ExternalPartner.this, result.getResult(),
- Toast.LENGTH_SHORT).show();
- }
- break;
- default:
- break;
- }
- };
- };
- public static class Product {
- public String subject;
- public String body;
- public String price;
- }
- public static Product[] sProducts;
- }
現在你要想的是,需要哪些數據,提供給onItemClick()方法裏面的 String info。demo裏面的數據是getNewOrderInfo(int position)這個方法提供的,你可以自己提供或者在上面修改下。
現在,你已經知道代碼是如何調用支付寶接口了。接下來,是如何把這些集成到你的項目中去。
2,寫好你的項目
3,如何將支付寶接口集成到你的項目
1,將alipay.jar這個包複製至商戶應用工程的libs 目錄下,通過 Java Build Path導入進工程,文檔說明上有
2,將 alipay_lib 拷貝到 Eclipse workspace,通過 Eclipse import 該工程,並在此工程的 Properties->Android 中選中爲 library 工程,文檔上有圖有真相
3,修改Manifest ,在商戶應用工程的 AndroidManifest.xml 文件裏面添加 Activity 聲明:
- <activity
- android:name="com.alipay.android.app.sdk.WapPayActivity"
- android:screenOrientation="portrait">
- </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" />
- 至此,SDK 開發資源導入完成。
如果需要對代碼進行混淆防止別人反編譯的話可以配置,不需要的話這步可省略,文檔上有配置方法。
5,然後在你的項目裏面新建一個包,名稱可取com.xxx.alipay,直接把demo(alipay_sdk_demo)中的com.alipay.android.msp.demo包裏面的類文件和demo裏面的資源文件都直接複製過來,然後把ExternalPartner.java裏面調用支付寶的核心代碼提取到你項目裏面要調用的地方,刪除ExternalPartner.java就OK了。
4,向支付寶申請簽約
進入我文章開頭的那個地址,直接點申請就行了。
目的是爲了或得PID與密鑰,到時候你好按照文檔配置你的Key.java文件。
申請要上傳你的apk或者產品說明文檔,我用的是產品說明文檔,裏面要包括產品說明(最好有產品截圖,我第一次沒截圖直接沒成功)、接口使用場景、資費說明。
申請成功後按文檔上說明,做幾個密鑰什麼的哈哈,上傳你做的公鑰。然後Key.java裏面配置。這裏要注意的是,String PRIVATE=“”;這裏要用PKCS8格式的私鑰。
配置成功即可測試啦!
好了,android集成支付寶接口我認爲比較快的過程就是這些了,祝你配置成功!