本文面向有一定微信APP支付(以下简称微信支付)接入经验的开发者
首先,我们先要明白微信支付的简单支付流程,流程如下图(灵魂画手):
微信支付简单流程如图:
1、统一下单。为了安全推荐在服务端实现,本地APP只是用来做调起支付,不参与下单逻辑。
2、将参数发送给微信。这一步调用微信的IWXAPI.sendReq来实现。
3、用户在微信进行支付等操作。这步已经离开我们的APP,不做赘述。
4、微信返回支付结果。微信会通过外部调用<你的包名.wxapi.WXPayEntryActivity>这个Activity来达到传输结果的目的。
5、处理结果。
那么,想想我们的目的是什么,是将微信支付剥离成单独Module,不受包名、项目限制的一个单独模块(或者说SDK)。最理想的结果就是:导入此Module,不需要做什么配置就能直接调起微信支付。做过模块化的朋友应该有很深刻的体会。
看上面5步,第1步是后端逻辑,第2步,第3步,第4步是公共逻辑,这三步,我们可以抽离到Module中去实现。第5步是App自身逻辑。接下来就是实现。
第2,3步,有经验的朋友应该不会陌生,这里就不在赘述,重要的是第4步,前面也说到了,我们是想要不受包名、项目的限制。但微信要求却是项目下的指定路径(详见)。唯一桎梏我们的就是这一点了。现在你是不是在想,要是WXPayEntryActivity可以随便放不就好了吗?
为了实现第4步功能,我们今天的猪脚登场了!!!!(卧槽,铺垫了这么久!!不要骂我)
activity-alias!!!!!!!!!!!!!赞美谷歌,赞美太阳,赞美百度
activity-alias顾名思义,他是一个activity代理,他能做什么事情呢?没啥大事,就是仅仅帮你这个Activity多做一个入口而已,类似于........emmmmmm,你们知道钓鱼网站吧,大概就是那个意思。
现在,我们把WXPayEntryActivity放到Module中去,你现在可以改名了,WXPayEntryActivity可以不叫这个名字,什么PayActivity,ResultActivity什么的,随便你来,OK,路径也随便你了,你想怎么来,就怎么来,彻底解放。
在Module的清单文件中加上
Module中真正的回调Activity然后,你在主工程的APP清单文件中加上
主工程的代理入口name:你的ApplicationID.wxapi.WXPayEntryActivity。
exported:true,表示可以被非本APP调用,一般是隐式启动会需要。
targetActivity:目标Activity,就是你需要真实使用的Activity。
好了,你们又会问了。你说了这么多!!证据呢!!!你怎么知道这样就OK呢!!
emmmm,不知道有没有小伙伴看过你支付完成的那一瞬间的Log,大概长成这样
关注点两个:
Log1、红色区域,表示ActivityManager正在启动WXPayEntryActivity,并且还带了extras。他启动的方式是applicationId/.wxapi.WXPayEntryActivity。
2、蓝色区域,发起这个启动的请求是10117进程,我程序是9221,所以,emmmm,肯定是微信在隐式启动我啊。
那么微信是怎么知道这么来启动我呢???我猜测(这个真的只能猜测了!!!!!)。我们在https://open.weixin.qq.com上面填的applicationId被他用来做 发起支付的APP身份验证 和 回调给具体APP的标识。毕竟applicationId是唯一的。Android系统内部也是通过这个来区分同的应用。“/”后面那部分就简单了,他指具体那个Activity,因为微信规定了路径,他只需要按照他规定的路径来隐式启动Activity就好了。
所以他在官网上才会规定必须是固定路径,且exported为true。因为不这样,他隐式启动不了啊!!魂淡!!
于是,我们使用代理的模式,向外部暴露出他想要的路径,他就能成功的启动我们,并发送结果。
然后。。。就完了。这个剥离的难点就在WXPayEntryActivity这个上面,现在我们使用代理的方式解决这个问题,剩下的就是完善Module逻辑。
注解:包名、applicationId:Android Studio出来之前包名以前被用来当作唯一身份标识并且也被用来当作路径的其实部分。在那个时候:包名=applicationId。然后Android Studio出来一段时间以后,突然发现可以单独的设置applicationId了,这个时候applicationId就被单独得用来做唯一身份标识。而包名就是仅仅是manifest文件中统一路径的一个东西。不再用来作为唯一标识。这个时候:报名≠applicationId。但是由于习惯了包名的叫法,且一般新建项目的时候,这两者都一样。现在大家说的包名大部分就是applicationId。是不是很绕???我也觉得是啊!!所以麻烦以后你们都开始叫applicationId好不好!!!!!
在完结之前放个我单独抽取的Module(SDK???)
在Application中初始化 在支付的地方这样调用使用的时候真的只需要这两句代码,不要额外配置什么,唯一注意的是使用 的主项目的包名和签名与你提交开放平台的一致。
对了!!顺便一说,微信的那个WXEntryActivity也可以按照这种思路来修改,以后再也不用在项目中单独的为微信创建这俩Activity了。
我自己写的demo地址:
https://github.com/EricMo/lib_wxpay
网友评论