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

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

1.接入支付的库

2.接受服务端的订单信息,发起调用支付宝和微信

3.接收支付宝和微信的回调

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

我们需要去官网下载最新的DEMO和SDK:https://docs.open.alipay.com/54/104509/

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

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

调用支付宝SDK的方法发起支付

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

//下面的orderInfo就是咱自己的服务端返回的订单信息,里面除了订单ID等,还有签名等安全信息
//使用方式基本按照支付宝的DEMO里面就行了

final Runnable payRunnable = new Runnable() {
 @Override
	public void run() {
    PayTask alipay = new PayTask(PayDemoActivity.this);
    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();

由上面的调用可见,支付宝是通过消息机制来接收回调的,所以我们得在Handler的消息中接收回调信息。

private Handler mHandler = new Handler() {
	    @SuppressWarnings("unused")
 	   public void handleMessage(Message msg) {
  	      switch (msg.what) {
        case SDK_PAY_FLAG: {
            //这里接收支付宝的回调信息
            //需要注意的是,支付结果一定要调用自己的服务端来确定,不能通过支付宝的回调结果来判断
            break;
        }
        default:
            break;
        }
    };
};

需要注意的是,支付结果一定要调用自己的服务端来确定,不能通过支付宝的回调结果来判断!

其他

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

已经安装过支付宝,会直接调用支付宝支付
没有安装支付宝,会调起支付宝的H5页面支付
以上就是支付宝的接入了,步骤还是比较简单的,也没有什么坑。下面的微信支付就有坑了…

微信支付接入
接入微信的SDK

https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=8_5

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

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

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

调用微信SDK发起支付

String content = ...    (这个是服务端返回的订单信息)
//需要一个注册微信支付的APPID
IWXAPI api = WXAPIFactory.createWXAPI(mContext, APPID);
JSONObject json = new JSONObject(content); 
PayReq req = new PayReq();
req.appId           = json.getString("appid");
req.partnerId       = json.getString("partnerid");
req.prepayId        = json.getString("prepayid");
req.nonceStr        = json.getString("noncestr");
req.timeStamp       = json.getString("timestamp");
req.packageValue    = json.getString("package");
req.sign            = json.getString("sign");
api.sendReq(req); //这里就发起调用微信支付了

接收微信支付的回调

可以看到上面的代码和支付宝就不一样了,没有用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不显示,否则就会闪一下或是停留在这个黑黑页面。
不显示的话要注意2点,一个是不要有布局,另一个就是要及时的finish掉这个Activity

public class WXPayEntryActivity extends Activity implements IWXAPIEventHandler{
    
    ...

    @Override
    public void onResp(BaseResp resp) {
        Log.d(TAG, "onPayFinish, errCode = " + resp.errCode);

        if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX) {
            <!--AlertDialog.Builder builder = new AlertDialog.Builder(this);-->
            <!--builder.setTitle(R.string.app_tip);-->
            <!--builder.setMessage(getString(R.string.pay_result_callback_msg, String.valueOf(resp.errCode)));-->
            <!--builder.show();-->
            //这里肯定不能是像上面的DEMO一样弹出对话框了,而是通知我们发起支付调用的页面
            //然后及时finish掉这个页面,贴个伪代码:
            sendPayNotice()
            finish();
        }
    }
}

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