纯手工打造Easy支付库

纯手工打造Easy支付库

前言

接触微信支付的开发者都知道,微信支付的所有处理都是在WXPayEntryActivity类里面,按一般开发流程走的话,我们在订单页面发起PayReq支付,微信调起支付页面,支付的结果处理会调用WXPayEntryActivity的onResp方法,根据BaseResp返回的结果码来判断当前的支付结果,整个思路是没错的,关键是有一点,WXPayEntryActivity是一个Activity,虽然我拿到了结果,但是我怎么将这个结果带到我的订单页面呢,因为我的订单在支付成功后是要跳到订单完成页面,有一个笨办法是,直接在WXPayEntryActivity里面跳转订单完成页面,但是,如果业务有10个不同的订单完成页面,那这样处理肯定是不行,那就想第二个办法,一下就想到了,我们可以在每次支付的时候,提前存储一个标记值,然后在WXPayEntryActivity里面根据这个标记值来判断跳转不同的订单完成页面,大功告成,这时候,产品来了说,你这订单完成页面就显示个订单完成,我需要你显示用户支付订单的所有信息,这时候,之前的办法就gg了,因为在WXPayEntryActivity的结果处理里面,我只能拿到微信的结果,但拿不到订单的信息,那只能再想一个办法了,这一下又想到了,用EventBus,我们可以在订单页面进行注册EventBus,然后订阅一个支付结果处理的方法,我们在支付完成的时候用Eventbus post到订单页面来,因为订单信息在订单页面,这时候支付结果和订单信息都有了,那我们可以直接在订单页面的订阅方法里面处理跳转了,但是,还是有缺点,每次增加订单页我都需要在WXPayEntryActivity类里面增加判断和post,感觉有点违背封装。

这么多的思路都是笔者一路踩着坑过来的,虽然都能实现业务,但是代码结构一团糟,就区区支付这方面就动用了一个库,实在是有点浪费,难道就没有一种不用库, 支付又简单,结果处理也简单的办法吗,苦思冥想,终于打造了这款Easy库

知识储备

在了解Easy库的时候,我们首先来认识一个类“ResultReceiver”,来看看官网的解释:第一句就告诉了我们,他是专门接收回调处理结果的一个接口,关键的一点是他实现了Parcelable接口,这也就意味着他能在Intent组件间任意行走

那这个类怎么用呢?我先写个简单点的代码

/**
 * A类
 */
public class A extends Activity {
    public static final int RESULT_REQUEST_CODE = 0x9429;
    ResultBackData backData;
    TextView textView;
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        backData = new ResultBackData();
        textView = new TextView(this);
    }
    public void onClick(View v) {
        Intent intent = new Intent(this, B.class);
        intent.putExtra("resultReceiver", backData);
    }
    class ResultBackData extends ResultReceiver {
        public ResultBackData() {
            super(null);
        }
        @Override
        protected void onReceiveResult(int resultCode, Bundle resultData) {
            super.onReceiveResult(resultCode, resultData);
            if (resultCode == RESULT_REQUEST_CODE) {
                String result = resultData.getString("result");
                textView.setText(result);
            }
        }
    }
}
/**
 *   B类
 */
public class B extends Activity {
    ResultReceiver receiver;
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        receiver = getIntent().getParcelableExtra("resultReceiver");
    }
    public void onClick(View v){
        Bundle b=new Bundle();
        b.putString("result","我是B处理的数据,带给你A来了");
        receiver.send(A.RESULT_REQUEST_CODE,b);
    }
}

从这段简短的代码中可以看到,继承ResultReceiver的ResultBackData类,就好比一个中间件,实现了A与B的数据传输,传输最主要的两个点是实现ResultReceiver的onReceiveResult方法来处理接收,通过ResultReceiver的send实现发送数据,两个毫不相干的Activity由于一个ResultReceiver实现了数据返回,这时候我们可以思考,A类就好比我们的订单类,B类就好比微信的WXPayEntryActivity类,我们完全可以通过ResultReceiver这个纽带从订单类中将ResultReceiver实例传输给WXPayEntryActivity类,待结果处理结束后通过ResultReceiver的send方法返回给订单类,订单类去判断接下来的处理,这完全可以摈弃Eventbus带来的臃肿,有了想法那就开始干。

思路实现

上面的ResultReceiver示例代码是写在Avtivity里面的,如果有10个订单支付页面,难道我还要写10个ResultReceiver?这未免有点太臃肿,让程序员去干重复性的代码,那还不如让去打丑豆,在实施前我们先画一个思维图

画好了思维图,对整个结构一目了然,看看我们的订单页,是多么的清晰,即使来几百个订单页,我们的支付处理也只需要注重发起和接收,然后来看看代码的实现。

代码实现

订单页处理

在开始讲解前,我把最终的效果先贴出来,然后跟着结果一步步去解说

      PayWeixin pay=new PayWeixin();
      pay.setAppid("")
      pay.setNoncestr("")
      pay.setOrdercode("")
      pay.setPackag("")
      pay.setPrepayid("")
      pay.setSign("")
      pay.setTimestamp("")
 EasyPayShare.getInstance().doPayWx(this, pay, new ShareCallBack() {
            @Override
            public void onSuccess(String result) {
                Toast.makeText(PayActivity.this, "支付成功", Toast.LENGTH_SHORT).show();
            }
            @Override
            public void onCanceled(String result) {
                Toast.makeText(PayActivity.this, "支付取消", Toast.LENGTH_SHORT).show();
            }
            @Override
            public void onFailed(String result) {
                Toast.makeText(PayActivity.this, "支付失败", Toast.LENGTH_SHORT).show();
            }
        });

初始化微信

写一个管理类EasyPayShare,专门处理微信的初始化和支付(因为该库还有支付宝支付、微信分享和微信登录,该文章就拿一个点来讲,其他思路都是一样实现),

public class EasyPayShare {
private static IWXAPI wxApi;
    private EasyPayShare() {
    }
    private static class InnerClass {
        static EasyPayShare Instance = new EasyPayShare();
    }
    public static EasyPayShare getInstance() {
        return InnerClass.Instance;
    }
    public void registWeixin(Context context, @NonNull String appId) {
        wxApi = WXAPIFactory.createWXAPI(context, appId, true);
        wxApi.registerApp(appId);
    }
    public IWXAPI getWxApi() {
        return wxApi;
    }
    /**
     * 
     * @param context  上下文
     * @param payWeixin  预支付订单
     * @param callBack  回调的引用
     */
  public void doPayWx(Context context,PayWeixin.JsonDataBean payWeixin, ShareCallBack callBack) {
        IWeixin share = new WxShare(context);
        share.pay(payWeixin, callBack);
    }
}

我们只需要在Application里面去初始化

    @Override
    public void onCreate() {
        super.onCreate();
    EasyPayShare.getInstance().registWeixin(this, "微信AppId");
    }

回调的接口

该接口是订单页与ReceiverResult通讯的纽带,因为支付的结果只有三种,所以我们定义了三个方法

/**
 * @author wangqi
 * @since 2017/11/15 18:16
 */
public interface ShareCallBack {
    void onSuccess(String result);
    void onCanceled(String result);
    void onFailed(String result);
}

ResultReceiver的处理

EasyPayShare的doPayWx方法里面调用了WxShare的pay方法,将预支付订单和ResultReceiver通过Intent传递给WxPayEntryActivity类,onReceiveResult接收处理后的结果,通过不同的requestCode来区分支付是失败还是取消还是成功

/**
 * @author wangqi
 * @since 2017/11/15 18:14
 */
public class WxShare extends ResultReceiver implements IWeixin {
    public static final String KEY_SHARE_LISTENER = "key_share_listener";
    public static final String KEY_PAY = "key_pay";
    private WeakReference<Context> contextWeakRef;
    private ShareCallBack callback;
    public WxShare(Context context) {
        super(null);
        contextWeakRef = new WeakReference<>(context);
    }
    @Override
    public void pay(PayWeixin.JsonDataBean payWeixin, ShareCallBack callback) {
        this.callback = callback;
        Context context = contextWeakRef.get();//
        if (context != null) {//
            context.startActivity(new Intent(context, WXPayEntryActivity.class)
                    .putExtra(KEY_SHARE_LISTENER, this)
                    .putExtra(KEY_PAY, payWeixin)
            );
        }
    }
    @Override
    protected void onReceiveResult(int resultCode, Bundle resultData) {
        String result;
        if (resultData == null) {
            result = "null";
        } else {
            result = resultData.getString("result", "null");
        }
        if (resultCode == IWeixin.RESULT_SUCCESS) {
            if (callback != null) {
                callback.onSuccess(result);
            }
        } else if (resultCode == IWeixin.RESULT_CANCELD) {
            if (callback != null) {
                callback.onCanceled(result);
            }
        } else {
            if (callback != null) {
                callback.onFailed(result);
            }
        }
    }
}

WxPayEntryActivity页面的处理

整个WxPayEntryActivity页面十分的简单,拿到预支付订单支付,然后通过ResultReceiver把结果返回,这个地方我让WxPayEntryActivity去继承了EasyWxPayEntryActivity,这样,开发者在使用该库的时候,只需要在特定的包名位置继承EasyWxPayEntryActivity就行了

/**
 * @author wangqi
 * @since 2017/11/18 15:25
 */
public class EasyWxPayEntryActivity extends Activity implements IWXAPIEventHandler {
    Bundle bundle;
    ResultReceiver listener;
    PayWeixin payWeixin;
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        EasyPayShare.getInstance().getWxApi().handleIntent(getIntent(), this);
        // 判断是否安装了微信客户端
        if (!EasyPayShare.getInstance().getWxApi().isWXAppInstalled()) {
            Toast.makeText(this, "您还未安装微信客户端!", Toast.LENGTH_SHORT).show();
            finish();
        }
        bundle = getIntent().getExtras();
        listener = bundle.getParcelable(WxShare.KEY_SHARE_LISTENER);
        payWeixin = (PayWeixin) bundle.getSerializable(WxShare.KEY_PAY);
        pay();
    }
    @Override
    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);
        setIntent(intent);
        EasyPayShare.getInstance().getWxApi().handleIntent(intent, this);
    }
    private void pay() {
        PayReq req = new PayReq();
        req.appId = payWeixin.getAppid();
        req.partnerId = payWeixin.getPartnerid();
        req.prepayId = payWeixin.getPrepayid();
        req.packageValue = payWeixin.getPackag();
        req.nonceStr = payWeixin.getNoncestr();
        req.timeStamp = payWeixin.getTimestamp();
        req.sign = payWeixin.getSign();
        EasyPayShare.getInstance().getWxApi().sendReq(req);
    }
    @Override
    public void onReq(BaseReq baseReq) {
    }
    @Override
    public void onResp(BaseResp resp) {
        switch (resp.errCode) {
            case BaseResp.ErrCode.ERR_OK:
                listener.send(IWeixin.RESULT_SUCCESS, bundle);
                break;
            case BaseResp.ErrCode.ERR_USER_CANCEL:
                listener.send(IWeixin.RESULT_CANCELD, bundle);
                break;
            default:
                listener.send(IWeixin.RESULT_FAIL, bundle);
                break;
        }
        finish();
    }
}

结果返回

支付的结果通过ResultReceiver send到WxShare页面的onReceiveResult页面,通过requestCode来区分不同的结果,然后将订单页传递过来的ShareCallBack回调给订单页,整个思路的处理非常的简单和Easy化

总结

代码量下来并不多,总之,一个好的思路会给开发带来很多的方便。多说一句,平时在使用框架的时候不能只专注使用,我们要多关注关注框架内部的实现,也就是看源码,一个好的思路能给开发者带来很大的启发 源码:https://github.com/MRwangqi/EasyPayShare

原文发布于微信公众号 - codelang(codelang)

原文发表时间:2017-11-26

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏GIS讲堂

Arcgis andoid开发之应用百度地图接口实现精准定位与显示

怀着激动、兴奋的心情,在这个漫天柳絮的季节写下了这片博文,为什么呢,因为困扰我很久的一个技术性的问题得到了解决,发次博文,供大家参观、学习,同时,也以慰藉我长期...

1323
来自专栏GIS讲堂

Arcgis for Androd API开发系列教程(一)——地图显示与GPS定位

序:最近呢,工作鸭梨不是怎么大,对于自己爱折腾的想法又冒出了水面,开始自己的android开发的学习之旅。但是呢,本人是做GIS的,所以呢,就打算从这方面入手看...

1655
来自专栏developerHaoz 的安卓之旅

Android 录音功能直接拿去用

这个类可以说是这个包的核心了,如果理解了这个 Service,录音这一块基本就没什么问题了。

6033
来自专栏分享达人秀

Intent 属性详解(上)

Android应用将会根据Intent来启动指定组件,至于到底启动哪个组件,则取决于Intent的各属性。本期将详细介绍Intent的各属性值,以及 A...

23010
来自专栏james大数据架构

列表视图(ListView和ListActivity)

在ListView中显示网络图片  ImageView 类虽然有一个 setImageUri 方法,但不能直接接受一个由网络地址生成的uri作为参数从而显示图片...

3317
来自专栏向治洪

nfc近场通信

NFC简介: Near Field Communication 近场通信,是一种数据传输技术。 与wifi、蓝牙、红外线等数据传输技术的一个主要差异就是有...

3849
来自专栏Android-JessYan

我一行代码都不写实现Toolbar!你却还在封装BaseActivity?

原文地址: http://www.jianshu.com/p/75a5c24174b2 qq群:301733278

1744
来自专栏刘望舒

感受LiveData与ViewModel结合之美

虽说这篇是说LiveData与ViewModel,但是或多或少都有涉及另外一个组件:Lifecycles 。它们连同Room都是在17年谷歌IO大会推出的,当时...

1412
来自专栏向治洪

nfc开发

    很多Android设备已经支持NFC(近距离无线通讯技术)了。本文就以实例的方式,为大家介绍如何在Android系统中进行NFC开发。 Andro...

4435
来自专栏向治洪

讯飞语音

、你需要android手机应用开发基础 2、科大讯飞语音识别SDK android版 3、科大讯飞语音识别开发API文档 4、android手机 关于科大讯飞S...

31810

扫码关注云+社区

领取腾讯云代金券