Xamarin.Android Binding篇
2017-10-11

前言

趁着失业了,闲着没事儿学习了下Xamarin.Android binding,在以往的开发中,我相信很多人都遇到过binding的坑,也不例外,

我也踩了很多雷,好在认识了个大佬,指导了很多  8{U`QQB5X27@C_FO](KQ(4G !

在做这个工作前,我上网搜索了一下关于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工程

image

3.2 将下载的压缩包解压,然后把里面的jar包以及so文件放到以下目录

image

这里的so文件就是为不同的cup架构提供的指令集

3.3 接下来就是设置jar、so文件的属性,按照如下设置

image

image

3.4接下来就是编译,不出意外会出现以下错误

image

错误的意思就是字段名称和类名重复了才导致编译不过,那么接下来怎么办呢,这就需要我们手动配置这些字段名称,毕竟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的地址。

效果图:

IMG_0499

 在调用百度地图的时候,需要配置你的APK签名的sha1值,调试的时候呢,都是用的这个目录下的 C:\Users\HuShuai\AppData\Local\Xamarin\Mono 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\用户名\AppData\Local\Xamarin\Mono for Android\debug.keystore" -alias androiddebugkey -storepass android -keypass android。

 但是这个有时候好像不太管用, 比如我就没有获取到,不如第一种方式简单暴力。

支付宝SDK绑定

有了前面的百度地图绑定示例,后面的两个绑定基本都类似了,无非就是修修改改编译不过的问题。

SDK下载地址:https://doc.open.alipay.com/doc2/detail.htm?treeId=54&articleId=104509&docType=1

But支付宝稍微有点儿不一样:

一开始编译的话,应该会提示以下错误

image

全部提示不能强制转换类RpcException为java.lang.object

按照上一个绑定的经验来绑定的话,完全懵逼了,根本没法弄。But可以这样

将转换的地方,直接修改成 this.Handle

类似下面这样

image

接下来就是把里面转换的地方都改完,然后把里面的内容全部复制,新建一个名字为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>

效果图:

L7_A26CRVNM5V_%`]0I~D~E

微信SDK绑定

微信SDK下载地址:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=11_1

还是按照之前的操作,将jar文件复制到 Jars 文件夹里,设置编译属性为 EmbeddedJar

这里呢说一下博主在绑定过程中 遇到的坑:

                                                                             4Q9@38MVIQ5M57TA]AIV82E

一开始编译的时候,提示名称重复,我心想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源码,

image

我发现是有这个类的,但是为什么绑定没有呢? 我又在绑定项目工程下的目录去查看了生成的类文件,结果也没有叫做 PayReq的类

image

What ? 这咋办呢,然后看了下警告信息,发现是这个类不能正确映射,然后就缺失了,但是这咋办呢?翻了一下午的Xamarin论坛,又全是英文,也没有找到什么有效的解决办法,But 我最前面说认识了个大佬啊,不懂就请教啊。

大佬告诉我你这错误原因可能是 JDK版本 设置的过高的原因,让我改成 JDK1.7,目标编译框架设置成6.0试试

image

大佬就是大佬,一针见血的找到问题,剩下的就是两个小问题,重命名就搞定了,到此编译成功!

<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看了生成的类,是有这个的  !哭泣的脸

image

But并不能影响什么,因为还是可以编译成功!!!

                                                                                      UIOIUMFT@X73XL8PYFXCOP6

image

而且如果我将编译好的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

如果觉得对你有帮助,请帮我点个赞,你的推荐是我学习的动力。

原文地址: http://www.cnblogs.com/CallMeUncle/p/6562440.html