前言:个人上篇文章介绍了Android微信支付接入,这里将支付宝支付也介绍下,希望对大家能有帮助。
正文:无论是微信支付还是支付宝支付,其总体流程基本都是相同的。结构图如下:
订单信息:其中包含订单价格,订单号,密钥等订单信息。在支付宝支付中,该订单信息是一段由key&value拼接的字符串。如果在项目中,该订单字符串由服务器构造完成,那么在APP客户端只需使用该订单信息串即可,无需自己构造,支付工作在Android端也会简单很多。如果服务端没有对该串进行构造,而只是返回了关键数据给APP端,那么APP端就要完成拼接工作。(支付宝也是极力推荐订单信息在服务端构造的,当然后文还是会详细说明订单信息的详细字段以及构造方法)
支付请求:在APP端调用支付SDK进行支付(详情见后文)
支付结果反馈:支付宝SDK调用支付方法会直接将结果返回,由于该操作有网络延迟,故要在子线程中调用, 并采用Handler通知的形式将支付结果返回到UI层。
1,接入与配置
a,官方SDK&demo下载地址,下载后将alipaySdk-20161129.jar导入项目中,这里本文介绍的2016版的这个jar包,也就是2.0版的。
b,AndroidManifest配置

    name="android.permission.INTERNET" />    name="android.permission.ACCESS_NETWORK_STATE" />    name="android.permission.ACCESS_WIFI_STATE" />    name="android.permission.READ_PHONE_STATE" />    name="android.permission.WRITE_EXTERNAL_STORAGE" />name="com.alipay.sdk.app.H5PayActivity"                                      android:configChanges="orientation|keyboardHidden|navigation"android:exported="false"android:screenOrientation="behind" />

2,代码调用支付接口
这里注意了,一般来讲下面代码中orderInfo是完全由服务端构造生成的,APP端只需获取后在此使用即可。如果服务端没有构造该参数,那么参考后文中提到的参数构造。

        Runnable payRunnable = new Runnable() {            @Override            public void run() {                PayTask alipay = new PayTask(activity);                Map<String, String> result = alipay.payV2(orderInfo, true);                Log.i("msp", result.toString());                Message msg = new Message();                msg.what = SDK_PAY_FLAG;                msg.obj = result;                mHandler.sendMessage(msg);            }        };        Thread payThread = new Thread(payRunnable);        payThread.start();

3,实现回调
从上述代码中可看到支付宝已经将结果在payV2函数将其返回,并且发送给指定handler。故在handler中对返回结果进行判断即可。

Map result = alipay.payV2(orderInfo, true);private Handler mHandler = new Handler() {        public void handleMessage(Message msg) {            switch (msg.what) {                case 1: {                    @SuppressWarnings("unchecked")                    PayResult payResult = new PayResult((Map) msg.obj);                    /**                     对于支付结果,请商户依赖服务端的异步通知结果。同步通知结果,仅作为支付结束的通知。                     */                    String resultInfo = payResult.getResult();// 同步返回需要验证的信息                    String resultStatus = payResult.getResultStatus();                    // 判断resultStatus 为9000则代表支付成功                    if (TextUtils.equals(resultStatus, "9000")) {                        // 该笔订单是否真实支付成功,需要依赖服务端的异步通知。                        Toast.makeText(PayActivity.this, "支付成功", Toast.LENGTH_SHORT).show();                    } else {                        // 该笔订单真实的支付结果,需要依赖服务端的异步通知。                        Toast.makeText(PayActivity.this, "支付失败", Toast.LENGTH_SHORT).show();                    }                    break;                }                default:                    break;            }        }        ;    };

4,订单信息构造
在本文第2点:代码调用支付接口中需得到一个String类型的订单信息参数,在服务端没有给定该完整参数的情况下,我们需要自己构造拼接该参数。
下面代码举例已知APPID,out_trade_no(订单号),subject(订单关键字等),body(交易的具体描述信息),total_fee(),5个参数的拼接代码。buildOrderParamMap()函数中的参数可自定义。具体参数说明

Map<String, String> params = OrderInfoUtil2_0.buildOrderParamMap(APPID,out_trade_no,subject,body,total_fee);        String orderParam = OrderInfoUtil2_0.buildOrderParam(params);        String sign = OrderInfoUtil2_0.getSign(params, RSA_PRIVATE);//RSA_PRIVATE商户私钥,pkcs8格式        final String orderInfo = orderParam + "&" + sign;

下面是所用到的工具类

public class OrderInfoUtil2_0 {    /**     * 构造支付订单参数列表     * @param app_id     * @return     */    public static Map buildOrderParamMap(String app_id,String out_trade_no,String subject,String body,String total_fee) {        Map keyValues = new HashMap();        keyValues.put("app_id", app_id);        keyValues.put("biz_content", "{\"timeout_express\":\"30m\",\"product_code\":\"QUICK_MSECURITY_PAY\",\"total_amount\":\""+total_fee+"\",\"subject\":\""+subject+"\",\"body\":\""+body+"\",\"out_trade_no\":\"" + out_trade_no +  "\"}");//这里可以拼接自己需要填写的参数        keyValues.put("charset", "utf-8");        keyValues.put("method", "alipay.trade.app.pay");        keyValues.put("sign_type", "RSA");        keyValues.put("timestamp", "2016-07-29 16:55:53");        keyValues.put("version", "1.0");        keyValues.put("notify_url", "http://XXX");//服务端支付回调接口        return keyValues;    }    /**     * 构造支付订单参数信息     *      * @param map     * 支付订单参数     * @return     */    public static String buildOrderParam(Map map) {        List keys = new ArrayList(map.keySet());        StringBuilder sb = new StringBuilder();        for (int i = 0; i < keys.size() - 1; i++) {            String key = keys.get(i);            String value = map.get(key);            sb.append(buildKeyValue(key, value, true));            sb.append("&");        }        String tailKey = keys.get(keys.size() - 1);        String tailValue = map.get(tailKey);        sb.append(buildKeyValue(tailKey, tailValue, true));        return sb.toString();    }    /**     * 拼接键值对     *      * @param key     * @param value     * @param isEncode     * @return     */    private static String buildKeyValue(String key, String value, boolean isEncode) {        StringBuilder sb = new StringBuilder();        sb.append(key);        sb.append("=");        if (isEncode) {            try {                sb.append(URLEncoder.encode(value, "UTF-8"));            } catch (UnsupportedEncodingException e) {                sb.append(value);            }        } else {            sb.append(value);        }        return sb.toString();    }    /**     * 对支付参数信息进行签名     *      * @param map     *            待签名授权信息     *      * @return     */    public static String getSign(Map map, String rsaKey) {        List keys = new ArrayList(map.keySet());        // key排序        Collections.sort(keys);        StringBuilder authInfo = new StringBuilder();        for (int i = 0; i < keys.size() - 1; i++) {            String key = keys.get(i);            String value = map.get(key);            authInfo.append(buildKeyValue(key, value, false));            authInfo.append("&");        }        String tailKey = keys.get(keys.size() - 1);        String tailValue = map.get(tailKey);        authInfo.append(buildKeyValue(tailKey, tailValue, false));        String oriSign = SignUtils.sign(authInfo.toString(), rsaKey);        String encodedSign = "";        try {            encodedSign = URLEncoder.encode(oriSign, "UTF-8");        } catch (UnsupportedEncodingException e) {            e.printStackTrace();        }        return "sign=" + encodedSign;    }}

更多相关文章

  1. PayPal发布新版Android(安卓)SDK 简化移动平台“应用内支付”的
  2. android studio引入最新版银联支付功能
  3. 我的Android知识架构
  4. 准备你的应用(Android免安装应用)
  5. (转)支付宝 Android(安卓)版使用的开源组件
  6. Android客户端和Java服务器端集成支付宝
  7. 【Android开源项目解析】仿支付宝付款成功及"天女散花"效果实现
  8. 安卓国内开发者陷盈利困局 版本过多支付不畅
  9. Android(安卓)编程技巧之 ----- 自定义 View 踩坑总结

随机推荐

  1. 浅谈android系统java调用C++函数传参过程
  2. Android(安卓)Bundle类
  3. 分析android的gadge
  4. android的各种*.img 文件
  5. GitHub 优秀的 Android(安卓)开源项目
  6. 解析Android消息处理机制:Handler/Thread/
  7. 可折叠的列表ExpandableListView及其适配
  8. 覆盖Android开发各个领域的近百个源码项
  9. 刚开始安卓,记录一个刚做的图片缩放程序
  10. Android常用命令