Android 轻松完成支付宝支付教程

4,088 阅读6分钟

参考安卓Dialog源码,他的builder设计模式实现方式是,使用内部类来实现功能,外部类的作用是通过build()函数,来对内部类进行参数设置,例如setter方法。

Buidler设计模式它一般用于用户不知道内部构建细节的情况下更加精细地控制对象构建流程,例如安卓系统的AlertDialog类,它包括了icon、title、msg、button1、button2等,这些装配的部分是比较多的,代码实现过程也较为漫长。使用Builder模式,可以使得这些调用变得简单,在构建过程各个部件可以自由扩展,以及重新设置参数。

支付宝支付

首先集成支付宝SDK

下载地址:doc.open.alipay.com/doc2/detail…

这里写图片描述

下载完成后,把压缩包解压,在下图的路径下找到这个jar文件,并把它引入到项目中。

这里写图片描述

然后把demo中的三个java文件复制到项目中,它们是:Base64.java、PayResult.java、SignUtils.java。

这里写图片描述

开始写AliPay这个类

笔者参考AlerDialog的类源码,对支付宝支付的过程进行一个Builder设计模式封装。你可以支付复制拿去用,立刻完成支付功能。其AliPay.java代码如下:

public class AliPay {

    public static class Builder {
        private Activity mActivity;

        public Builder(Activity activity) {
            this.mActivity = activity;
        }
        /**
         * 设置商户PID
         */
        private String PARTNER = "";
        // 商户收款账号
        private String SELLER = "";
        // 商户私钥,pkcs8格式
        private String RSA_PRIVATE = "";
        // 支付宝公钥
        private String RSA_PUBLIC = "";
        private int SDK_PAY_FLAG = 6406;

        private String orderTitle = "";
        private String subTitle = "";
        private String price = "";
        private String notifyURL = "";

        /**
         * 设置商户PID
         */
        public AliPay.Builder setPARTNER(String PARTNER) {
            this.PARTNER = PARTNER;
            return this;
        }
        /**
         * 设置商户收款账号
         */
        public AliPay.Builder setSELLER(String SELLER) {
            this.SELLER = SELLER;
            return this;
        }
        /**
         * 设置商户私钥,pkcs8格式
         */
        public AliPay.Builder setRSA_PRIVATE(String RSA_PRIVATE) {
            this.RSA_PRIVATE = RSA_PRIVATE;
            return this;
        }

        /**
         * 设置支付宝公钥
         */
        public AliPay.Builder setRSA_PUBLIC(String RSA_PUBLIC) {
            this.RSA_PUBLIC = RSA_PUBLIC;
            return this;
        }

        /**
         * 设置商品名称
         */
        public AliPay.Builder setOrderTitle(String orderTitle) {
            this.orderTitle = orderTitle;
            return this;
        }
        /**
         * 设置商品详情
         */
        public AliPay.Builder setSubTitle(String subTitle) {
            this.subTitle = subTitle;
            return this;
        }
        /**
         * 设置商品价格
         */
        public AliPay.Builder setPrice(String price) {
            this.price = price;
            return this;
        }
        /**
         * 设置支付宝支付成功后通知的地址,可以填写你公司的地址
         */
        public AliPay.Builder setNotifyURL(String notifyURL) {
            this.notifyURL = notifyURL;
            return this;
        }

        @SuppressLint("HandlerLeak")
        private Handler mHandler = new Handler() {
            @SuppressWarnings("unused")
            public void handleMessage(Message msg) {
                switch (msg.what) {
                    case 6406: {
                        PayResult payResult = new PayResult((String) msg.obj);
                        /**
                         * 同步返回的结果必须放置到服务端进行验证(验证的规则请看https://doc.open.alipay.com/doc2/
                         * detail.htm?spm=0.0.0.0.xdvAU6&treeId=59&articleId=103665&
                         * docType=1) 建议商户依赖异步通知
                         */
                        String resultInfo = payResult.getResult();// 同步返回需要验证的信息

                        String resultStatus = payResult.getResultStatus();
                        // 判断resultStatus 为“9000”则代表支付成功,具体状态码代表含义可参考接口文档
                        if (TextUtils.equals(resultStatus, "9000")) {
                            if (mPayCallBackListener != null) {
                                mPayCallBackListener.onPayCallBack(9000, "9000", "支付成功");
                            }
                        } else {
                            // 判断resultStatus 为非"9000"则代表可能支付失败
                            // "8000"代表支付结果因为支付渠道原因或者系统原因还在等待支付结果确认,最终交易是否成功以服务端异步通知为准(小概率状态)
                            if (TextUtils.equals(resultStatus, "8000")) {
                                if (mPayCallBackListener != null) {
                                    mPayCallBackListener.onPayCallBack(8000, "8000", "支付结果确认中");
                                }

                            } else {
                                // 其他值就可以判断为支付失败,包括用户主动取消支付,或者系统返回的错误
                                if (mPayCallBackListener != null) {
                                    mPayCallBackListener.onPayCallBack(0, "0", "支付失败");
                                }
                            }
                        }
                        break;
                    }
                    default:
                        break;
                }
            }
        };


        public void pay() {
            if (TextUtils.isEmpty(PARTNER) || TextUtils.isEmpty(RSA_PRIVATE) || TextUtils.isEmpty(SELLER)) {
                new AlertDialog.Builder(mActivity).setTitle("警告").setMessage("需要配置PARTNER | RSA_PRIVATE| SELLER")
                        .setPositiveButton("确定", new DialogInterface.OnClickListener() {
                            public void onClick(DialogInterface dialoginterface, int i) {
                                //
                                mActivity.finish();
                            }
                        }).show();
                return;
            }
            String orderInfo = getOrderInfo(orderTitle, subTitle, price, notifyURL);

            /**
             * 特别注意,这里的签名逻辑需要放在服务端,切勿将私钥泄露在代码中!
             */
            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(mActivity);
                    // 调用支付接口,获取支付结果
                    String result = alipay.pay(payInfo, true);

                    Message msg = new Message();
                    msg.what = SDK_PAY_FLAG;
                    msg.obj = result;
                    mHandler.sendMessage(msg);
                }
            };

            // 必须异步调用
            Thread payThread = new Thread(payRunnable);
            payThread.start();

        }

        public void setPayCallBackListener(PayCallBackListener listener) {
            this.mPayCallBackListener = listener;
        }

        private PayCallBackListener mPayCallBackListener;

        public interface PayCallBackListener {
            void onPayCallBack(int status, String resultStatus, String progress);
        }


        /**
         * 创建订单信息
         */
        private String getOrderInfo(String subject, String body, String price, String notifyURL) {

            // 签约合作者身份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=" + "\"" + notifyURL + "\"";

            // 服务接口名称, 固定值
            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;
        }

        /**
         * 生成商户订单号,该值在商户端应保持唯一(可自定义格式规范)
         */
        private 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;
        }

        /**
         * sign the order info. 对订单信息进行签名
         *
         * @param content 待签名订单信息
         */
        private String sign(String content) {
            return SignUtils.sign(content, RSA_PRIVATE);
        }

        /**
         * 获取签名方式
         */
        private String getSignType() {
            return "sign_type=\"RSA\"";
        }
    }

    // 商户PID
    private String PARTNER="";
    // 商户收款账号
    private String SELLER="";
    // 商户私钥,pkcs8格式
    private String RSA_PRIVATE="";
    // 支付宝公钥
    private String RSA_PUBLIC="";

    private String orderTitle="";
    private String subTitle="";
    private String price="";
    private String notifyURL="";
    private Builder payData;

    public Builder build(Activity activity) {
        payData = new Builder(activity);
        payData.setPARTNER(PARTNER);
        payData.setSELLER(SELLER);
        payData.setRSA_PRIVATE(RSA_PRIVATE);
        payData.setRSA_PUBLIC(RSA_PUBLIC);
        payData.setOrderTitle(orderTitle);
        payData.setSubTitle(subTitle);
        payData.setPrice(price);
        payData.setNotifyURL(notifyURL);
        return payData;
    }

    public AliPay setPARTNER(String PARTNER) {
        this.PARTNER = PARTNER;
        return this;
    }

    public AliPay setSELLER(String SELLER) {
        this.SELLER = SELLER;
        return this;
    }

    public AliPay setRSA_PRIVATE(String RSA_PRIVATE) {
        this.RSA_PRIVATE = RSA_PRIVATE;
        return this;
    }

    public AliPay setRSA_PUBLIC(String RSA_PUBLIC) {
        this.RSA_PUBLIC = RSA_PUBLIC;
        return this;
    }

    public AliPay setOrderTitle(String orderTitle) {
        this.orderTitle = orderTitle;
        return this;
    }

    public AliPay setSubTitle(String subTitle) {
        this.subTitle = subTitle;
        return this;
    }

    public AliPay setPrice(String price) {
        this.price = price;
        return this;
    }

    public AliPay setNotifyURL(String notifyURL) {
        this.notifyURL = notifyURL;
        return this;
    }

    public void pay() {
        if (payData != null) {
            payData.pay();
        }
    }
}

调用方式

这里写图片描述

它会有一个回调状态处理,调用setPayCallBackListener即可得到。

补充点:

权限配置






AndroidManifest.xml(你可以直接复制下面的)





私钥、公钥

私钥或公钥的生成,支付宝已经在demo文件夹中提供了工具。

这里写图片描述

RSA密钥生成命令

生成RSA私钥

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 -inform PEM -in rsa_private_key.pem -outform PEM -nocrypt

然后,将生成的公钥上传到支付宝商户账户中,具体文档在下面连接中:
doc.open.alipay.com/doc2/detail…

将私钥PKCS8格式保存在代码中,

这里写图片描述

支付页面:

这里写图片描述

到这里你已经可以完成支付宝支付功能,很简单吧。笔者继续研究微信支付…