7、与iOS、Android的交互 高级篇

本文属于「Unity与iOS、Android平台的整合」系列文章之一,转载请注明出处。主要讲解如何优雅地实现Unity与iOS、Android的交互。

零、前言

市面上提供的SDK基本是原生iOS、Android,在Unity接入过程中,往往不是简简单单地一次调用,而是成套的原生API接入,需要一套完整的解决方案来处理这些问题。
在此,将自己的方案提出,意为抛砖引玉,如果大家有其他思路还望留言交流。
该解决方案的是对基础原理进行的拓展。
4、与iOS、Android的交互 理论篇
5、与iOS、Android的交互 实践篇——主动调用
6、与iOS、Android的交互 实践篇——传递参数

一、需求分析

实际项目使用中,我们希望

  • 使用C#代码调用相关功能
  • 在调用时不需要区分具体平台
  • 在编辑器模式下进行模拟测试
  • 实现复杂类型参数相互传递
  • 实现监听
  • 实现回调
  • ……

二、方案设计

<<<需求:

  • 使用C#代码调用相关功能

>>>方案:

将代码根据职责进行分层:

  • 上层逻辑(C#)
  • 供上层逻辑调用的接口(C#)
  • 调用SDK的接口(C++、Java)
  • SDK API(OC、Java)

<<<需求:

  • 在调用时不需要区分具体平台
  • 在编辑器模式下进行模拟测试

>>>方案:

将上一个需求的方案进行扩充:

  • 供上层逻辑调用的接口(C#),使用编译开关在函数内部根据平台区别调用
    public void Login()
    {
        Login_();
    }
    #if UNITY_EDITOR
    private static void Login_()
    {
        Debug.Log("SDK Login");
    }
    #elif UNITY_IOS
    [DllImport("__Internal")]private static extern void Login_();
    #elif UNITY_ANDROID
private static void Login_() 
{
    using (AndroidJavaClass jc = new AndroidJavaClass("xxx.xxx.xxx))
    {
        jc.CallStatic("Login_"); 
    }
}
    #endif

<<<需求:

  • 实现复杂类型参数相互传递

>>>方案:

  • 使用Json序列化、反序列化相关类型
  • 相互传递JSON字符串
public class Message
{
    public int id;
    public string name;
    public static string ToJson(Message msg)
    {
        return Json.Serialize(msg);
    }
    public static Message FromJson(string json)
    {
        return Json.DeSerialize<Message>(json);
    }
}
public void SetLoginInfo(Message msg)
{
    SetLoginInfo_(Message.ToJson(msg));
}
#if UNITY_EDITOR
private static void SetLoginInfo_(string msg)
{
    Debug.Log("SDK SetLoginInfo" + msg);
}
#elif UNITY_IOS
[DllImport("__Internal")]private static extern void SetLoginInfo_(string msg);
#elif UNITY_ANDROID
private static void SetLoginInfo_()
{
    using (AndroidJavaClass jc = new AndroidJavaClass("xxx.xxx.xxx))
    {
        jc.CallStatic("SetLoginInfo_", msg); 
    }
}
#endif
public Message GetLoginInfo()
{
    Message.FromJson(GetLoginInfo_());
}
#if UNITY_EDITOR
private static string GetLoginInfo_()
{
    return "{\"id\":1,\"name\":\"abc\"}";
}
#elif UNITY_IOS
[DllImport("__Internal")]private static extern string GetLoginInfo_();
#elif UNITY_ANDROID
private static string GetLoginInfo_()
{
    using (AndroidJavaClass jc = new AndroidJavaClass("xxx.xxx.xxx))
    {
        return jc.CallStatic<string>("GetLoginInfo_"); 
    }
}
#endif

<<<需求:

  • 实现监听

>>>方案:

  • 供上层逻辑调用的接口(C#),提供监听
  • 调用SDK的接口(C++、Java),注册原生API的监听
  • 使用UnitySendMessage调用指定C#方法从而触发C#层的监听
  • 使用唯一的GameObject来负责接收UnitySendMessage
    供上层逻辑调用的接口(C#)
public class XXXSDKEventHandler : Monobehaviour
{
    public event Action onLogin;
    public void OnLogin()
    {
        if(onLogin != null)
        {
            onLogin();
        }
    }
}
public class XXXSDKManager
{
    private XXXSDKEventHandler handler;
    public XXXSDKManager()
    {
        handler = new GameObject("XXXSDKEventHandler").AddComponent<XXXSDKEventHandler>();
        Object.DontDestroyOnLoad(handler.gameObject);
    }
    public event Action onLogin
    {
        add { handler.onLogin += value; }
        remove { handler.onLogin -= value; }
    }
}

调用SDK的接口(C++)

void OnLogin(){
    UnitySendMessage("XXXSDKEventHandler", "OnLogin", "");
}

调用SDK的接口(Java)

void OnLogin(){
    UnityPlayer.UnitySendMessage("XXXSDKEventHandler", "OnLogin", "");
}

<<<需求:

  • 实现回调

>>>方案:

  • 供上层逻辑调用的接口(C#),缓存回调并分配回调ID
  • 调用SDK的接口(C++、Java),接收回调ID
  • 使用UnitySendMessage调用指定C#方法从而触发C#层的回调
  • 使用唯一的GameObject来负责接收UnitySendMessage
public class XXXSDKEventHandler : Monobehaviour
{
    private int cba_key = 0;
    private Dictionary<int, Action> cbas = new Dictionary<int, Action>();
    public int AddCallbackAction(Action action)
    {
        cbas.Add(cba_key, action);
        return cba_key++;
    }
    private void Receiver(string jsonMessage)
    {
        JsonObject jo = Json.DeserializeObject<JsonObject>(jsonMessage);
        int key = int.Parse(jo["cba_key"].ToString());
        if (!cbas.ContainsKey(key)) { return; }
        Action<bool> action = cbas[key];
        if (action != null) { action(result); }
        cbas.Remove(key);
    }
    #region 延时回调
    Queue<string> callbackQueue = new Queue<string>();
    private void MessageHandle(string json)
    {
        JsonObject jo = Json.DeserializeObject<JsonObject>(json);
        string funcName = jo["funcName"].ToString();
        string message = jo["message"].ToString();
        this.callbackQueue.Enqueue(funcName);
        this.callbackQueue.Enqueue(message);
    }
    void Update()
    {
        while (true)
        {
            yield return null;
            while (callbackQueue.Count >= 2)
            {
                SendMessage(this.callbackQueue.Dequeue(), this.callbackQueue.Dequeue());
            }
        }
    }
    #endregion
}
public class XXXSDKManager
{
    private XXXSDKEventHandler handler;
    public XXXSDKManager()
    {
        handler = new GameObject("XXXSDKEventHandler").AddComponent<XXXSDKEventHandler>();
        Object.DontDestroyOnLoad(handler.gameObject);
    }
    public Login(Action onFinish)
    {
        Login_(handle.AddCallbackAction(onFinish));
    }
    #if UNITY_EDITOR
    private static void Login_(int callbackID)
    {
        Debug.Log("SDK Login");
        handler.SendMessage("Receiver","{\"funcName\":\"Receiver\",\"message\":\"{\\\"cba_key\\\":\\\"" + callbackID + "\\\"}\"}");
    }
    #elif UNITY_IOS
    [DllImport("__Internal")]private static extern void Login_(int callbackID);
    #elif UNITY_ANDROID
    private static void Login_(int callbackID) 
    {
        using (AndroidJavaClass jc = new AndroidJavaClass("xxx.xxx.xxx))
        {
            jc.CallStatic("Login_", callbackID); 
        }
    }
}

SDK的回调接口(C++)

void Login_(int callbackID){
    //处理xxx,最后调用以下
    UnitySendMessage("XXXSDKEventHandler", "OnLogin", [NSString stringWithFormat:@"{\"funcName\":\"Receiver\",\"message\":\"{\\\"cba_key\\\":\\\"%d\\\"}\"}", callbackID]);
}

SDK的回调接口(Java)

void Login_(int callbackID){
    //处理xxx,最后调用以下
    UnityPlayer.UnitySendMessage("XXXSDKEventHandler", "OnLogin", "{\"funcName\":\"Receiver\",\"message\":\"{\\\"cba_key\\\":\\\"" + callbackID + "\\\"}\"}");
}

三、收个尾

以上内容为Unity与iOS/Android原生SDK交互的整套解决方案,能够解决复杂的原生SDK交互需求,在项目中也经历过实践,亲测可用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,783评论 5 472
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,396评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,834评论 0 333
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,036评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,035评论 5 362
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,242评论 1 278
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,727评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,376评论 0 255
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,508评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,415评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,463评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,140评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,734评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,809评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,028评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,521评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,119评论 2 341

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,566评论 18 139
  • 前言 最近在实验室做了一个项目,用到了蓝牙通讯和U3D的交互,都有很多坑,如:IOS与Unity3D界面之间的跳转...
    Larrycal阅读 23,594评论 39 53
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,537评论 18 399
  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 6,340评论 0 17