最近在做支付寶支付模塊琼牧,集成過程中走了一些彎路。現(xiàn)將集成過程總結(jié)一些哀卫。
這里我直接寫了一個(gè)工具類巨坊,需要的時(shí)候直接調(diào)用對象的支付方法即可,很方便了此改。
AlipayUtil aliPay=new AlipayUtil(activity,price,callBackUrl);
aliPay.Pay();
完整的代碼如下:
/**
* 可直接將代碼當(dāng)做自己的工具類
* 使用方法:
* AlipayUtil aliPay=new AlipayUtil(activity,price,callBackUrl);
* aliPay.Pay();
*/
public class AlipayUtil {
private float price;
private Activity mActivity;
private String notify_url;
//一下四個(gè)參數(shù)趾撵,需要可開發(fā)者自己配置成自己的阿里上后臺數(shù)據(jù)
// 需要注意的是: 1 商戶的私鑰是開發(fā)者在本地生成
//? ? ? ? ? ? ? ? 2 將秘鑰上傳至阿里后臺的“合作伙伴秘鑰管理”中的 RSA秘鑰。
//? ? ? ? ? ? ? ? 3 生成的公鑰在客戶端沒有用處
// 商戶PID
public static final String PARTNER = "2088221700022324";
// 商戶收款賬號
public static final String SELLER = "3372459162@qq.com";
// 商戶私鑰共啃,pkcs8格式
public static final String RSA_PRIVATE = "你的私鑰";
private static final int SDK_PAY_FLAG = 1;
private static final int SDK_CHECK_FLAG = 2;
public AlipayUtil(Activity mActivity, float price,String url) {
this.mActivity = mActivity;
this.price = price;
this.notify_url=url;
}
public void Pay() {
String orderInfo = getOrderInfo("測試", "test",
String.valueOf(price));
// 對訂單做RSA 簽名
String sign = sign(orderInfo);
try {
// 僅需對sign 做URL編碼
sign = URLEncoder.encode(sign, "UTF-8");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
// 完整的符合支付寶參數(shù)規(guī)范的訂單信息
final String payInfo = orderInfo + "&sign=\"" + sign + "\"&"
+ getSignType();
Runnable payRunnable = new Runnable() {
@Override
public void run() {
// 構(gòu)造PayTask 對象
PayTask alipay = new PayTask(mActivity);
// 調(diào)用支付接口鼓寺,獲取支付結(jié)果
String result = alipay.pay(payInfo, true);
Message msg = new Message();
msg.what = SDK_PAY_FLAG;
msg.obj = result;
mHandler.sendMessage(msg);
}
};
// 必須異步調(diào)用
Thread payThread = new Thread(payRunnable);
payThread.start();
}
/**
* sign the order info. 對訂單信息進(jìn)行簽名
* @param content
* 待簽名訂單信息
*/
public String sign(String content) {
return SignUtils.sign(content, RSA_PRIVATE);
}
/**
* get the sign type we use. 獲取簽名方式
*
*/
public String getSignType() {
return "sign_type=\"RSA\"";
}
/**
* create the order info. 創(chuàng)建訂單信息
*
*/
public String getOrderInfo(String subject, String body, String price) {
// 簽約合作者身份ID
String orderInfo = "partner=" + "\"" + PARTNER + "\"";
// 簽約賣家支付寶賬號
orderInfo += "&seller_id=" + "\"" + SELLER + "\"";
// 商戶網(wǎng)站唯一訂單號
orderInfo += "&out_trade_no=" + "\"" + getOutTradeNo() + "\"";
// 商品名稱
orderInfo += "&subject=" + "\"" + subject + "\"";
// 商品詳情
orderInfo += "&body=" + "\"" + body + "\"";
// 商品金額
orderInfo += "&total_fee=" + "\"" + price + "\"";
// 服務(wù)器異步通知頁面路徑
orderInfo += "?ify_url=" + "\"" +notify_url
+ "\"";
// 服務(wù)接口名稱, 固定值
orderInfo += "&service=\"mobile.securitypay.pay\"";
// 支付類型勋磕, 固定值
orderInfo += "&payment_type=\"1\"";
// 參數(shù)編碼妈候, 固定值
orderInfo += "&_input_charset=\"utf-8\"";
// 設(shè)置未付款交易的超時(shí)時(shí)間
// 默認(rèn)30分鐘,一旦超時(shí)挂滓,該筆交易就會自動被關(guān)閉苦银。
// 取值范圍:1m~15d。
// m-分鐘,h-小時(shí)幔虏,d-天纺念,1c-當(dāng)天(無論交易何時(shí)創(chuàng)建,都在0點(diǎn)關(guān)閉)想括。
// 該參數(shù)數(shù)值不接受小數(shù)點(diǎn)陷谱,如1.5h,可轉(zhuǎn)換為90m瑟蜈。
orderInfo += "&it_b_pay=\"30m\"";
// extern_token為經(jīng)過快登授權(quán)獲取到的alipay_open_id,帶上此參數(shù)用戶將使用授權(quán)的賬戶進(jìn)行支付
// orderInfo += "&extern_token=" + "\"" + extern_token + "\"";
// 支付寶處理完請求后烟逊,當(dāng)前頁面跳轉(zhuǎn)到商戶指定頁面的路徑,可空
orderInfo += "&return_url=\"m.alipay.com\"";
// 調(diào)用銀行卡支付铺根,需配置此參數(shù)宪躯,參與簽名, 固定值 (需要簽約《無線銀行卡快捷支付》才能使用)
// orderInfo += "&paymethod=\"expressGateway\"";
return orderInfo;
}
/**
* get the out_trade_no for an order. 生成商戶訂單號位迂,該值在商戶端應(yīng)保持唯一(可自定義格式規(guī)范)
*
*/
public String getOutTradeNo() {
SimpleDateFormat format = new SimpleDateFormat("MMddHHmmss",
Locale.getDefault());
Date date = new Date();
String key = format.format(date);
Random r = new Random();
key = key + r.nextInt();
key = key.substring(0, 15);
return key;
}
private Handler mHandler = new Handler() {
public void handleMessage(Message msg) {
switch (msg.what) {
case SDK_PAY_FLAG: {
PayResult payResult = new PayResult((String) msg.obj);
// 支付寶返回此次支付結(jié)果及加簽访雪,建議對支付寶簽名信息拿簽約時(shí)支付寶提供的公鑰做驗(yàn)簽
String resultInfo = payResult.getResult();
String resultStatus = payResult.getResultStatus();
// 判斷resultStatus 為“9000”則代表支付成功,具體狀態(tài)碼代表含義可參考接口文檔
if (TextUtils.equals(resultStatus, "9000")) {
//支付成功 發(fā)送通知操作掂林,開發(fā)者自行定義臣缀。
} else {
// 判斷resultStatus 為非“9000”則代表可能支付失敗
// “8000”代表支付結(jié)果因?yàn)橹Ц肚涝蚧蛘呦到y(tǒng)原因還在等待支付結(jié)果確認(rèn),最終交易是否成功以服務(wù)端異步通知為準(zhǔn)(小概率狀態(tài))
if (TextUtils.equals(resultStatus, "8000")) {
Toast.makeText(mActivity, "支付結(jié)果確認(rèn)中", Toast.LENGTH_SHORT)
.show();
} else {
// 其他值就可以判斷為支付失敗泻帮,包括用戶主動取消支付肝陪,或者系統(tǒng)返回的錯(cuò)誤
Toast.makeText(mActivity, "支付失敗", Toast.LENGTH_SHORT)
.show();
}
}
break;
}
case SDK_CHECK_FLAG: {
Toast.makeText(mActivity, "檢查結(jié)果為:" + msg.obj, Toast.LENGTH_SHORT)
.show();
break;
}
default:
break;
}
};
};
}