多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
## 第三方支付 app中支付功能集成了微信支付和支付宝支付两大主流支付模式。 [TOC=3,3] ### 微信支付接入流程 &emsp;&emsp; 注1:微信SDK改成通过Gradle的方式发布到jcenter,包名做了相应修改,从原来的com.tencent.mm.sdk修改为com.tencent.mm.opensdk,需要开发者修改对应的import语句。 * 申请你的AppID &emsp;&emsp; 应用包名:是在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); ~~~ * 调起支付 &emsp;&emsp;商户服务器生成支付订单,先调用统一下单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;//签名 ~~~ * 支付结果回调 &emsp;&emsp;在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值列表: | 名称 | 描述 | 解决方案 | | --- | --- | --- | | &emsp;&emsp; 0&emsp;&emsp;| 成功 | 展示成功页面 | | -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&timestamp=2016-08-15%2012%3A12%3A15&version=1.0&sign=MsbylYkCzlfYLy9PeRwUUIg9nZPeN9SfXPNavUCroGKR5Kqvx0nEnd3eRmKxJuthNUx4ERCXe552EV9PfwexqW%2B1wbKOdYtDIb4%2B7PL3Pc94RZL0zKaWcaY3tSL89%2FuAVUsQuFqEJdhIukuKygrXucvejOUgTCfoUdwTi7z%2BZzQ%3D ~~~ * 支付结果获取和处理 调用pay方法支付后,将通过2种途径获得支付结果: 同步返回 &emsp;&emsp;商户应用客户端通过当前调用支付的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(); }; }; ~~~ 异步通知 &emsp;&emsp;商户需要提供一个http协议的接口,包含在请求支付的入参中,其key对应notify_url。支付宝服务器在支付完成后,会以POST方式调用notify_url传输数据。 详细文档参考支付宝开发平台[https://docs.open.alipay.com/204/105296/](https://docs.open.alipay.com/204/105296/)