前言
趁着失业了,闲着没事儿学习了下Xamarin.Android binding,在以往的开发中,我相信很多人都遇到过binding的坑,也不例外,
在做这个工作前,我上网搜索了一下关于binding的文章,也有,但是比较少,而且文章都是一两年前,很多SDK都更新了很多代了,
按照他们之前的一些做法就会有问题。
我尝试绑定了微信SDK、支付宝SDK、百度地图SDK,在大神帮助下,一步一步绑定成功了。
而且我相信这几个SDK在开发中用到的概率也还是很大的,所以特此分享下绑定遇到的坑,也为了更好的记录自己遇到的问题。
话不多说,直接上代码。
百度地图SDK绑定
1.准备工作:下载百度android开发包,毕竟工欲善其事必先利其器,地址:http://lbsyun.baidu.com/sdk/download?selected=mapsdk_basicmap,mapsdk_searchfunction,mapsdk_lbscloudsearch,mapsdk_calculationtool
2.开发环境,这里就不讲解开发环境搭建了。我这儿的环境的是VS2017+win10,后续的两个绑定都是如此。
3.Binding:
3.1 新建安卓bingding工程
3.2 将下载的压缩包解压,然后把里面的jar包以及so文件放到以下目录
这里的so文件就是为不同的cup架构提供的指令集
3.3 接下来就是设置jar、so文件的属性,按照如下设置
3.4接下来就是编译,不出意外会出现以下错误
错误的意思就是字段名称和类名重复了才导致编译不过,那么接下来怎么办呢,这就需要我们手动配置这些字段名称,毕竟VS再智能也会有绑定不对的时候啊!
打开文件Metadata.xml文件,加入以下代码。
<attr path="/api/package[@name='com.baidu.location']/class[@name='Address']/field[@name='address']" name="name">AddressInfo</attr> <attr path="/api/package[@name='com.baidu.mapapi']/class[@name='VersionInfo']/field[@name='VERSION_INFO']" name="name">VersionInformation</attr> <attr path="/api/package[@name='com.baidu.platform.comapi.map']/class[@name='B']/field[@name='b']" name="name">BField</attr> <attr path="/api/package[@name='com.baidu.platform.comapi.map']/class[@name='E']/field[@name='e']" name="name">EField</attr>这里的意思呢就是讲这些字段名称重命名,Metadata.xml还可以去除某些类或字段,设置某些函数的返回值之类的。Metadata.xml更多的用法的话,可以参考https://developer.xamarin.com/guides/android/advanced_topics/binding-a-java-library/customizing-bindings/java-bindings-metadata/。主要全是英文,我英文能力也差,估计翻译出来大家都会吐槽,还不如自己看。
然后再编译就没有问题了,说一下,百度地图我编译的环境是在jdk1.8+安卓目标平台7.0下编译的。
然后我照着百度地图提供的DEMO写了个示例
关键代码:
SDKInitializer.Initialize(ApplicationContext); SetContentView(Resource.Layout.Main); mapView = FindViewById<MapView>(Resource.Id.mapView); currentLocationMode = LocationMode.Normal; var baiduMap = mapView.Map; //定义Maker坐标点 LatLng point = new LatLng(30.691359, 104.052236); //构建Marker图标 BitmapDescriptor bitmap = BitmapDescriptorFactory.FromResource(Resource.Drawable.marker); //构建MarkerOption,用于在地图上添加Marker OverlayOptions option = new MarkerOptions().InvokePosition(point).InvokeIcon(bitmap); //在地图上添加Marker,并显示 baiduMap.AddOverlay(option); baiduMap.MyLocationEnabled = true;更多的用法的话,参照百度地图的API
代码的话,别急,会在最后给出GITHub的地址。
效果图:
在调用百度地图的时候,需要配置你的APK签名的sha1值,调试的时候呢,都是用的这个目录下的 C:UsersHuShuaiAppDataLocalXamarinMono for Android
debug.keystore来签名的,然后需要获取它的sha1值,推荐一个好用的工具,而且非常简单的,详情请看: http://bbs.lbsyun.baidu.com/forum.php?mod=viewthread&tid=106461&qq-pf-to=pcqq.group.
可以通过命令行来获取 Keytool -list -v -keystore "C:Users用户名AppDataLocalXamarinMono for Androiddebug.keystore" -alias androiddebugkey -storepass android -keypass android。
但是这个有时候好像不太管用, 比如我就没有获取到,不如第一种方式简单暴力。
支付宝SDK绑定
有了前面的百度地图绑定示例,后面的两个绑定基本都类似了,无非就是修修改改编译不过的问题。
SDK下载地址:https://doc.open.alipay.com/doc2/detail.htm?treeId=54&articleId=104509&docType=1
But支付宝稍微有点儿不一样:
一开始编译的话,应该会提示以下错误
全部提示不能强制转换类RpcException为java.lang.object
按照上一个绑定的经验来绑定的话,完全懵逼了,根本没法弄。But可以这样
将转换的地方,直接修改成 this.Handle
类似下面这样
接下来就是把里面转换的地方都改完,然后把里面的内容全部复制,新建一个名字为RpcException的类
然后把内容粘贴进去,新建的类放在 Additions 这个文件夹中。
然后就是在Metadata.xml里面加入一下代码
<remove-node path="/api/package[@name='com.alipay.android.phone.mrpc.core']/class[@name='RpcException']" />这里的意思呢就是删除banding自动给我们生成的类,而我们的 Additions 文件夹呢则是存放用户自定义的内容,在编译的时候则会把我们添加的内容一起编译进去。
然后就是自己写了一个支付demo,只能跳转到支付界面以及调用H5页面支付界面,没有测试是否能够真的支付成功,主要是没有如果要测试的话,还需要部署套测试环境才行。如果那位测试可以支付的话,请给博主留言,谢谢了。
调用H5支付页面的话,需要在AndroidManifest.xml文件中加入以下节点:
<activity android:name="com.alipay.sdk.app.H5PayActivity" android:configChanges="orientation|keyboardHidden|navigation" android:exported="false" android:screenOrientation="behind"></activity> <activity android:name="com.alipay.sdk.auth.AuthActivity" android:configChanges="orientation|keyboardHidden|navigation" android:exported="false" android:screenOrientation="behind"></activity>效果图:
微信SDK绑定
微信SDK下载地址:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=11_1
还是按照之前的操作,将jar文件复制到 Jars 文件夹里,设置编译属性为 EmbeddedJar
这里呢说一下博主在绑定过程中 遇到的坑:
一开始编译的时候,提示名称重复,我心想so easy,这难道不我啊,毕竟有两个绑定的经验了,然后就再Metadata.xml添加了如下两句代码:
<attr path="/api/package[@name='com.tencent.mm.opensdk.modelbase']/class[@name='BaseResp']/field[@name='errCode']" name="managedName">errCode</attr> <attr path="/api/package[@name='com.tencent.mm.opensdk.modelmsg']/class[@name='WXMediaMessage']/field[@name='mediaObject']" name="managedName">mediaObject</attr>
而且也如预期一样编译成功了,心想这不是多难啊。然后就想照着这个 文章 这样写个支付Demo,写着写着,准备用到 PayReq 这个类的时候,结果发现如何引用都找不到它,这就尴尬了。
我反编译看了下微信SDK源码,
我发现是有这个类的,但是为什么绑定没有呢? 我又在绑定项目工程下的目录去查看了生成的类文件,结果也没有叫做 PayReq 的类
What ? 这咋办呢,然后看了下警告信息,发现是这个类不能正确映射,然后就缺失了,但是这咋办呢?翻了一下午的Xamarin论坛,又全是英文,也没有找到什么有效的解决办法,But 我最前面说认识了个大佬啊,不懂就请教啊。
大佬告诉我你这错误原因可能是 JDK版本 设置的过高的原因,让我改成 JDK1.7,目标编译框架设置成6.0试试
大佬就是大佬,一针见血的找到问题,剩下的就是两个小问题,重命名就搞定了,到此编译成功!
<attr path="/api/package[@name='com.tencent.mm.opensdk.modelbase']/class[@name='BaseResp']/field[@name='errCode']" name="managedName">errCode</attr> <attr path="/api/package[@name='com.tencent.mm.opensdk.modelmsg']/class[@name='WXMediaMessage']/field[@name='mediaObject']" name="managedName">mediaObject</attr>
但是又遇到一个奇葩问题,在实例化 PayReq 对象时,VS还是找不到这个类,我用ILSpay看了生成的类,是有这个的 !
But并不能影响什么,因为还是可以编译成功!!!
而且如果我将编译好的dll 文件单独放到 安卓工程里面去引用的话,编译是没有问题的,这个时候VS也能知道这个类,在对象浏览器中也能看到这个类,我试过将两个工程里面的obj、bin目录下的文件都删除,然后重新编译,结果还是这个问题,直接引用这个binding工程找不到PayReq这个类,但是却能编译成功。不知道是不是VS17的一个Bug,如果那位园友遇到过这个问题,也可以留言说一下。
微信支付后回调Activity 代码:
using System; using Android.App; using Android.Content; using Android.OS; using Com.Tencent.MM.Opensdk.Modelbase; using Com.Tencent.MM.Opensdk.Openapi; using Android.Util; using Android.Widget; using Com.Tencent.MM.Opensdk.Constants; namespace XamarinWeiXinDemo { /// <summary> /// 微信支付回调Activity /// </summary> public class WXPayEntryActivity : Activity, IWXAPIEventHandler { private IWXAPI api; public void onCreate(Bundle savedInstanceState) { base.OnCreate(savedInstanceState); //SetContentView(R.layout.pay_result); //布局是可以自定义 api = WXAPIFactory.CreateWXAPI(this, "App_ID"); api.HandleIntent(Intent, this); } protected override void OnNewIntent(Intent intent) { base.OnNewIntent(intent); Intent = intent; api.HandleIntent(intent, this); } public void OnReq(BaseReq p0) { } public void OnResp(BaseResp p0) { Log.Debug("微信支付回调", "onPayFinish, errCode = " + p0.errCode); if (p0.Type == ConstantsAPI.CommandPayByWx) { //0 成功 展示成功页面 //-1 错误 可能的原因:签名错误、未注册APPID、项目设置APPID不正确、注册的APPID与设置的不匹配、其他异常等。 //-2 用户取消 无需处理。发生场景:用户不支付了,点击取消,返回APP。 if (p0.errCode == 0) { //支付成功逻辑 Toast.MakeText(this, "支付成功", ToastLength.Long).Show(); } else { //支付失败 Toast.MakeText(this, "支付失败", ToastLength.Long).Show(); } Finish(); } } } }
最后
三个绑定demo都已经传至GitHub
百度地图:https://github.com/HuUncle/Xamarin.Android-BaiDuMapSDKBinding
微信:https://github.com/HuUncle/Xamarin.Android-WeiXinSDKBindingDemo
支付宝:https://github.com/HuUncle/Xamarin.Android-AlipaySDKBindingDemo
如果觉得对你有帮助,请帮我点个赞,你的推荐是我学习的动力。
转载请注明出处 IT胡小帅: http://www.cnblogs.com/CallMeUncle/p/6562440.html