Android接入支付宝和微信支付

参考我现在项目里面的代码,聊一下支付吧!(本文代码有些进行封装,需要的话可以留言,分享源码)

前言

很多APP都需要支付功能,国内一般就是支付宝和微信了(可能还有银联)。目前这2种接入方式对于APP端来说都已经比较方便了,因为大部分的安全校验之类的逻辑都在服务端。

APP端总结起来就是三步走:

  1. 接入支付需要的库(支付宝,微信)
  2. 请求接口,获取服务端的订单信息,发起调用支付宝和微信
  3. 接收支付宝和微信的回调(跳转对应的页面)

支付宝接入

首先是接入支付宝的aar文件

比较坑的是支付宝还需要下载aar文件导入,而不是gradle里面一行依赖就能搞定的。

我们需要去官网下载最新的DEMO和SDK,官网地址在这里

然后把下载下来的aar包,放到项目目录下面的libs目录下,通过下面的gradle依赖进来

// 支付宝 SDK AAR 包所需的配置
compile(name: 'alipaySdk-15.6.0-20190226104104-noUtdid', ext: 'aar')

我的项目用组件化,所以依赖有差异
调用支付宝SDK的方法发起支付

调用支付宝SDK发起支付,只需要一个参数,就是服务端返回的订单信息。所以这里的支付顺序是先要我们调用服务端的接口创建一个订单,然后服务端把订单信息返回给我们,我们APP拿着这个订单信息去调用支付宝支付。

//支付宝
    private void zfbZf(String orderInfo) {//orderInfo就是咱自己的服务端返回的订单信息,里面除了订单ID等,还有签名、version等安全信息

//调起支付宝的操作要在子线程中进行,所以我用了rxjava,可参考之前的文章
        Observable.just(orderInfo)
                .map(new Function<String, Map<String, String>>() {
                    @Override
                    public Map<String, String> apply(String s) throws Exception {
                      //开启一个子线程去调用支付宝的支付功能,获取到支付结果后,通知UI线程,根据支付结果去显示不同的。
                        PayTask alipay = new PayTask(PayActivity.this);
                        Map<String, String> result = alipay.payV2(orderInfo, true);
                        return result;//返回支付结果
                    }
                })
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Observer<Map<String, String>>() {
                    @Override
                    public void onSubscribe(Disposable d) {

                    }

                    @Override
                    public void onNext(Map<String, String> stringStringMap) {
                        PayResult payResult = new PayResult(stringStringMap);
                        /**
                         * 对于支付结果,请商户依赖服务端的异步通知结果。同步通知结果,仅作为支付结束的通知。
                         */
                        String resultInfo = payResult.getResult();// 同步返回需要验证的信息
                        String resultStatus = payResult.getResultStatus();
                        // 判断resultStatus 为9000则代表支付成功
                        if (TextUtils.equals(resultStatus, "9000")) {
                            // 该笔订单是否真实支付成功,需要依赖服务端的异步通知。
                            finish();
                            //跳转对应页面即可
                        } else {
                            // 该笔订单真实的支付结果,需要依赖服务端的异步通知。
                            finish();
                            //跳转对应页面即可
                        }
                    }

                    @Override
                    public void onError(Throwable e) {

                    }

                    @Override
                    public void onComplete() {

                    }
                });
    }

需要注意的是,支付结果一定要调用自己的服务端来确定,不能通过支付宝的回调结果来判断!(上面的代码是直接根据支付宝回调做的判断,不规范)

其他

实际情况里需要考虑用户手机上有没有安装过支付宝

  • 已经安装过支付宝,会直接调用支付宝支付
  • 没有安装支付宝,会调起支付宝的H5页面支付

以上就是支付宝的接入了,步骤还是比较简单的,也没有什么坑。下面的微信支付就有坑了...

微信支付接入

接入微信的SDK

官网在这

相比于支付宝,微信接入SDK就比较简单了,一行代码搞定

//微信支付SDK
compile 'com.tencent.mm.opensdk:wechat-sdk-android-with-mta:+'

这里需要注意的是,微信的SDK里面支付和社交登录等这些是集成在一起的,无法分开。所以如果项目里之前已经集成过微信登录的话就不需要重复集成了!

调用微信SDK发起支付
//微信
    private void wxZf(PayWxBean.PayResponseBean bean) {//服务器返回给我们的数据,bean里内容见下图

        //封装了一个工具类,放在下面了
        WXPayUtils.WxBuilder builder = new WXPayUtils.WxBuilder();
        builder.setAppId(WxConfig.APP_ID)//需要一个注册微信支付的APPID
                .setPartnerId(bean.getPartnerid())//商户号
                .setPrepayId(bean.getPrepayid())//预支付交易会话ID
                .setPackageValue(bean.getPackageX())//扩展字段
                .setNonceStr(bean.getNonceStr())//随机字符串
                .setTimeStamp(bean.getTimeStamp())//时间戳
                .setSign(bean.getPaySign())//签名
                .build().toWxPay(this);

    }
服务器返回用于请求微信支付

工具类在这里

public class WXPayUtils {

    private IWXAPI iwxapi;
    private WxBuilder builder;

    public WXPayUtils(WxBuilder builder) {
        this.builder = builder;
    }

    public void toWxPay(Context context) {//这个方法注意!!
        iwxapi = WXAPIFactory.createWXAPI(context, null);//
        iwxapi.registerApp(builder.getAppId());

        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                PayReq request = new PayReq();
                request.appId = builder.getAppId();
                request.partnerId = builder.getPartnerId();
                request.prepayId = builder.getPrepayId();
                request.packageValue = builder.getPackageValue();
                request.nonceStr = builder.getNonceStr();
                request.timeStamp = builder.getTimeStamp();
                request.sign = builder.getSign();
                iwxapi.sendReq(request);//这里就发起调用微信支付了
            }
        };
        Thread thread = new Thread(runnable);
        thread.start();
    }


    public static class WxBuilder {

        public String appId;
        public String partnerId;
        public String prepayId;
        public String packageValue;
        public String nonceStr;
        public String timeStamp;
        public String sign;

        public WXPayUtils build() {
            return new WXPayUtils(this);
        }

        public String getAppId() {
            return appId;
        }

        public WxBuilder setAppId(String appId) {
            this.appId = appId;
            return this;
        }

        public String getPartnerId() {
            return partnerId;
        }

        public WxBuilder setPartnerId(String partnerId) {
            this.partnerId = partnerId;
            return this;
        }

        public String getPrepayId() {
            return prepayId;
        }

        public WxBuilder setPrepayId(String prepayId) {
            this.prepayId = prepayId;
            return this;
        }

        public String getPackageValue() {
            return packageValue;
        }

        public WxBuilder setPackageValue(String packageValue) {
            this.packageValue = packageValue;
            return this;
        }

        public String getNonceStr() {
            return nonceStr;
        }

        public WxBuilder setNonceStr(String nonceStr) {
            this.nonceStr = nonceStr;
            return this;
        }

        public String getTimeStamp() {
            return timeStamp;
        }

        public WxBuilder setTimeStamp(String timeStamp) {
            this.timeStamp = timeStamp;
            return this;
        }

        public String getSign() {
            return sign;
        }

        public WxBuilder setSign(String sign) {
            this.sign = sign;
            return this;
        }

    }
}

接收微信支付的回调

可以看到上面的代码和支付宝就不一样了,没有用Rxjava或者Handler。

微信支付比较特殊的一个地方是需要我们用一个特殊的Activity来接收回调的信息。这个处理不好很容易就碰到接收不到回调的情况。

我们需要新建一个名叫WXPayEntryActivity的Activity,内容的话可以参考微信SDK里面的。需要注意的有几点:

  • WXPayEntryActivity这个Activity的路径要符合要求,比如APP的包名是com.niubi.company,那这个Activity的路径就需要是com.niubi.company.wxapi.WXPayEntryActivity
  • 这个WXPayEntryActivity当然需要在AndroidManifest文件中注册
<activity
    android:name=".wxapi.WXPayEntryActivity"
    android:exported="true"
    android:launchMode="singleTop"/>

特别需要注意的是上面的exported属性和launchMode属性一定要加上,否则是接收不到回调的

WXPayEntryActivity

  • 我们在支付完成以后,一般是希望直接回到我们自己的应用里面。这个时候我们就需要让上面的WXPayEntryActivity不显示,否则就会闪一下或是停留在这个黑黑页面。
    不显示的话要注意2点,一个是不要有布局,另一个就是要及时的finish掉这个Activity
public class WXPayEntryActivity extends Activity implements IWXAPIEventHandler {
   ...
    /**
     * 发送给微信的回调
     */
    @Override
    public void onResp(BaseResp baseResp) {
        String s = JSON.toJSONString(baseResp);
        this.finish();//关闭这个页面
        if(baseResp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX){
            switch (baseResp.errCode) {
                case BaseResp.ErrCode.ERR_OK://成功
                    //跳转页面即可
                    break;
                case BaseResp.ErrCode.ERR_USER_CANCEL://失败
                    Toast.makeText(this, "取消支付", Toast.LENGTH_LONG).show();
                   //跳转页面即可
                    break;
                default:
                    RxBus.get().post(new RxBusWxBean(0));
                    Toast.makeText(this, baseResp.errCode + "", Toast.LENGTH_LONG).show();
                    break;
    }
}

其他

同样的,微信支付成功与否,也要通过调用自己的服务端来查看,而不能依赖微信的回调状态,这个要切记。

总结

以上就是微信和支付宝的支付接入,基本上按照上述步骤都不会有问题了。特别是微信支付,一定要特别注意文中提到的注意点,那些都是踩过的坑啊!

本文参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,921评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,635评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,393评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,836评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,833评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,685评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,043评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,694评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,671评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,670评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,779评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,424评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,027评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,984评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,214评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,108评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,517评论 2 343