## 第三方支付
app中支付功能集成了微信支付和支付宝支付两大主流支付模式。
[TOC=3,3]
### 微信支付接入流程
   注1:微信SDK改成通过Gradle的方式发布到jcenter,包名做了相应修改,从原来的com.tencent.mm.sdk修改为com.tencent.mm.opensdk,需要开发者修改对应的import语句。
* 申请你的AppID
   应用包名:是在APP项目配置文件AndroidManifest.xml中声明的package值,项目中的 package="com.yuanfengbbc.shop"。
![](https://box.kancloud.cn/222688d70e6e693e187dddfdd7a47eb2_647x453.png)
* 下载SDK及API文档
在build.gradle文件中,添加如下依赖即可:
~~~
dependencies {
compile 'com.tencent.mm.opensdk:wechat-sdk-android-with-mta:+'
}
~~~
或
~~~
dependencies {
compile 'com.tencent.mm.opensdk:wechat-sdk-android-without-mta:+'
}
~~~
(其中,前者包含统计功能)
3. 注册APPID
商户APP工程中引入微信JAR包,调用API前,需要先向微信注册您的APPID,代码如下:
~~~
IWXAPI api = WXAPIFactory.createWXAPI(this, null);
~~~
将该app注册到微信
WX_APP_ID在com.yuanfeng.ecdemo.yf.util的包下的Contants.java文件下。
~~~
api.registerApp(Contants.WX_APP_ID);
~~~
* 调起支付
  商户服务器生成支付订单,先调用统一下单API生成预付单,获取到prepay_id后将参数再次签名传输给APP发起支付。以下是调起微信支付的关键代码:
~~~
IWXAPI api;
PayReq request = new PayReq();
request.appId = "wxd930ea5d5a258f4f";
request.partnerId = "1900000109";
request.prepayId= "1101000000140415649af9fc314aa427",;
request.packageValue = "Sign=WXPay";
request.nonceStr= "1101000000140429eb40476f8896f4c9";
request.timeStamp= "1398746574";
request.sign= "7FFECB600D7157C5AA49810D2D8F28BC2811827B";
api.sendReq(request);
~~~
注:如果遇到支付不成功,可能是后台返回的签名的问题,需要自己手动签名
~~~
StringBuilder sb = new StringBuilder();
sb.append("appid=");
sb.append(Constans.WX_APP_ID);
sb.append('&');
sb.append("noncestr=");
sb.append(beanWx.getNonce_str());
sb.append('&');
sb.append("package=");
sb.append(req.packageValue);
sb.append('&');
sb.append("partnerid=");
sb.append(beanWx.getMch_id());
sb.append('&');
sb.append("prepayid=");
sb.append(beanWx.getPrepay_id());
sb.append('&');
sb.append("timestamp=");
sb.append(String.valueOf(System.currentTimeMillis()/ 1000 ));
sb.append('&');
sb.append("key=");
sb.append("fa45be83eaeb9e4da1661ba9c9e7cad9");
String appSign = MD5Util.encrypt(sb.toString());
req.sign = appSign;//签名
~~~
* 支付结果回调
  在com.yuanfengbbc.shop.wxapi包路径中实现WXPayEntryActivity类(包名或类名不一致会造成无法回调),在WXPayEntryActivity类中实现onResp函数,支付完成后,微信APP会返回到商户APP并回调onResp函数,开发者需要在该函数中接收通知,判断返回错误码,如果支付成功则去后台查询支付结果再展示用户实际支付结果。注意一定不能以客户端返回作为用户支付的结果,应以服务器端的接收的支付通知或查询API返回的结果为准。代码示例如下:
~~~
@Override
public void onResp(BaseResp resp) {
if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX) {//微信支付
PAY_CALLBACK_FLAG = resp.errCode;
LogUtil.i("resp.errStr====" + resp.errStr);
if (PAY_CALLBACK_FLAG == WXPayEntryActivity.SUCESS) {
currentStatus.setText("支付成功");
Constans.NUM_WAIT_PAY_ORDER = true;
Constans.NUM_WAIT_SEND = true;
} else {
ORDER_NUMBER = null;
currentStatus.setText("支付失败");
}
}
}
~~~
回调中errCode值列表:
| 名称 | 描述 | 解决方案 |
| --- | --- | --- |
|    0  | 成功 | 展示成功页面 |
| -1 | 错误 | 可能的原因:签名错误、未注册APPID、项目设置APPID不正确、注册的APPID与设置的不匹配、其他异常等。 |
| -2 | 用户取消 | 发生场景:用户不支付了,点击取消,返回APP。|
详细文档参考微信开放平台 [https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=8_5](https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=8_5)
### 支付宝支付接入流程
* 导入开发资源
将alipaySdk-xxxxxxxx.jar包放入商户应用工程的libs目录下,如下图。
![](https://box.kancloud.cn/668a02c351084375487422b81334afc2_516x78.png)
在app module下的build.gradle下手动添加依赖,如下代码所示:
~~~
dependencies {
......
compile files('libs/alipaySdk-20170725.jar')
......
}
~~~
* 修改Manifest
在应用工程的AndroidManifest.xml文件里面添加声明:
~~~
<activity
android:name="com.alipay.sdk.app.H5PayActivity"
android:configChanges="orientation|keyboardHidden|navigation|screenSize"
android:exported="false"
android:screenOrientation="behind"
android:windowSoftInputMode="adjustResize|stateHidden" >
</activity>
<activity
android:name="com.alipay.sdk.app.H5AuthActivity"
android:configChanges="orientation|keyboardHidden|navigation"
android:exported="false"
android:screenOrientation="behind"
android:windowSoftInputMode="adjustResize|stateHidden" >
</activity>
~~~
和权限声明:
~~~
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
~~~
* 添加混淆规则
在应用工程的proguard-project.txt里添加以下相关规则:
~~~
-keep class com.alipay.android.app.IAlixPay{*;}
-keep class com.alipay.android.app.IAlixPay$Stub{*;}
-keep class com.alipay.android.app.IRemoteServiceCallback{*;}
-keep class com.alipay.android.app.IRemoteServiceCallback$Stub{*;}
-keep class com.alipay.sdk.app.PayTask{ public *;}
-keep class com.alipay.sdk.app.AuthTask{ public *;}
-keep class com.alipay.sdk.app.H5PayCallback {
<fields>;
<methods>;
}
-keep class com.alipay.android.phone.mrpc.core.** { *; }
-keep class com.alipay.apmobilesecuritysdk.** { *; }
-keep class com.alipay.mobile.framework.service.annotation.** { *; }
-keep class com.alipay.mobilesecuritysdk.face.** { *; }
-keep class com.alipay.tscenter.biz.rpc.** { *; }
-keep class org.json.alipay.** { *; }
-keep class com.alipay.tscenter.** { *; }
-keep class com.ta.utdid2.** { *;}
-keep class com.ut.device.** { *;}
~~~
至此,开发包开发资源导入完成。
* 支付接口调用
需要在新线程中调用支付接口。(可参考alipay_demo实现)
PayTask对象主要为商户提供订单支付、查询功能,及获取当前开发包版本号。
获取PayTask支付对象调用支付(支付行为需要在独立的非ui线程中执行),代码示例:
~~~
//订单信息
final String payInfo = object.getJSONObject("data").getString("orderString");
Runnable payRun = new Runnable() {
@Override
public void run() {
// 构造PayTask 对象
PayTask alipay = new PayTask(ActivityOrderPayMethod.this);
// 调用支付接口,获取支付结果
String result = alipay.pay(payInfo, true);
Message msg = new Message();
msg.what = SDK_PAY_FLAG;
msg.obj = result;
aliPayResultHandler.sendMessage(msg);
}
};
// 必须异步调用
Thread payThread = new Thread(payRun);
payThread.start();
~~~
| 参数名称 | 参数说明 |
| --- | --- |
| String orderInfo | app支付请求参数字符串,主要包含商户的订单信息,key=value形式,以&连接。 |
| boolean isShowPayLoading | 用户在商户app内部点击付款,是否需要一个loading做为在钱包唤起之前的过渡,这个值设置为true,将会在调用pay接口的时候直接唤起一个loading,直到唤起H5支付页面或者唤起外部的钱包付款页面loading才消失。(建议将该值设置为true,优化点击付款到支付唤起支付页面的过渡过程。) |
orderStr示例如下,参数说明见"请求参数说明",orderStr的获取必须来源于服务端:
~~~
app_id=2015052600090779&biz_content=%7B%22timeout_express%22%3A%2230m%22%2C%22seller_id%22%3A%22%22%2C%22product_code%22%3A%22QUICK_MSECURITY_PAY%22%2C%22total_amount%22%3A%220.02%22%2C%22subject%22%3A%221%22%2C%22body%22%3A%22%E6%88%91%E6%98%AF%E6%B5%8B%E8%AF%95%E6%95%B0%E6%8D%AE%22%2C%22out_trade_no%22%3A%22314VYGIAGG7ZOYY%22%7D&charset=utf-8&method=alipay.trade.app.pay&sign_type=RSA2×tamp=2016-08-15%2012%3A12%3A15&version=1.0&sign=MsbylYkCzlfYLy9PeRwUUIg9nZPeN9SfXPNavUCroGKR5Kqvx0nEnd3eRmKxJuthNUx4ERCXe552EV9PfwexqW%2B1wbKOdYtDIb4%2B7PL3Pc94RZL0zKaWcaY3tSL89%2FuAVUsQuFqEJdhIukuKygrXucvejOUgTCfoUdwTi7z%2BZzQ%3D
~~~
* 支付结果获取和处理
调用pay方法支付后,将通过2种途径获得支付结果:
同步返回
  商户应用客户端通过当前调用支付的Activity的Handler对象,通过它的回调函数获取支付结果。(可参考alipay_demo实现)代码示例:
~~~
private Handler mHandler = new Handler() {
public void handleMessage(Message msg) {
PayResult payResult = new PayResult((String) msg.obj);
String resultStatus = payResult.getResultStatus();
};
};
~~~
异步通知
  商户需要提供一个http协议的接口,包含在请求支付的入参中,其key对应notify_url。支付宝服务器在支付完成后,会以POST方式调用notify_url传输数据。
详细文档参考支付宝开发平台[https://docs.open.alipay.com/204/105296/](https://docs.open.alipay.com/204/105296/)