Android/Unity大亂鬥-完整雙方集成交互指南

来源:https://www.cnblogs.com/zhimalier/archive/2020/01/13/12187574.html

這是一個很長很長的story!-芝麻粒兒創作 開篇 源碼地址:GitHub 本文目的,將Unity集成到Android端,學完本文後你可以做到 Android任意佈局載入Unity 3D場景 任意操作佈局中的按鈕/3D物品(縮放旋轉等) 互相消息通信(你叼我,我叼你) 自由切換Unity中的場景 動 ...


這是一個很長很長的story!-芝麻粒兒創作

開篇

源碼地址:GitHub

本文目的,將Unity集成到Android端,學完本文後你可以做到

  1. Android任意佈局載入Unity 3D場景
  2. 任意操作佈局中的按鈕/3D物品(縮放旋轉等)
  3. 互相消息通信(你叼我,我叼你)
  4. 自由切換Unity中的場景
  5. 動態載入手機SD卡3D資源,一次開發到處使用。
  6. 在小白麵前裝逼用

公眾號【空名先生】

故事正題

首要任務就是將Unity項目導出來(已經做好了3D的處理,關於通信和動態載入在下麵介紹)

公眾號【空名先生】

敲黑板,重點Export Project一定要勾選,之後點擊最下方的Export 靜等項目導出。

公眾號【空名先生】

導出後的結構感覺好熟悉,就跟Android Studio的項目結構一樣(PS:本來就是)打開Studio 以 moudle的形式導入android工程,第一次可能慢一些慢慢導,去喝杯茶。成功後重要的操作來了。打開剛纔導入的build.gradle文件,首當其衝的就是gradle版本的修改,跟你的studio版本一致。

dependencies {
    classpath 'com.android.tools.build:gradle:3.2.0'
}
我的是3.2 Android Studio,推薦不低於它。
往下走可以找到
apply plugin: 'com.android.application'

改為

apply plugin: 'com.android.library'

因為我們要以library的形式集成,接著再往下麵走就是熟悉的sdkVsersion了,保持和你的anroid項目一致。還有個applicationId,刪掉這個

有的項目遇見UnityAds.aar文件,不影響。其他修改以及遇到的一些坑不再贅述,有問題可以留言。我的如下:

// GENERATED BY UNITY. REMOVE THIS COMMENT TO PREVENT OVERWRITING WHEN EXPORTING AGAIN
buildscript {
    repositories {
        google()
        jcenter()
    }
    dependencies {
        classpath 'com.android.tools.build:gradle:3.2.0'
    }
}
allprojects {
    repositories {
        google()
        jcenter()
        flatDir {
            dirs 'libs'
        }
    }
}
apply plugin: 'com.android.library'
dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
}
android {
    compileSdkVersion 29
    compileOptions {
        sourceCompatibility JavaVersion.VERSION_1_8
        targetCompatibility JavaVersion.VERSION_1_8
    }
    defaultConfig {
        minSdkVersion 21
        targetSdkVersion 29
        ndk {
            abiFilters 'armeabi-v7a', 'x86'
        }
        versionCode 1
        versionName '1.0'
    }
    lintOptions {
        abortOnError false
    }
    aaptOptions {
        noCompress = ['.unity3d', '.ress', '.resource', '.obb']
    }
    buildTypes {
        debug {
            minifyEnabled false
            useProguard false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-unity.txt'
            jniDebuggable true
        }
        release {
            minifyEnabled false
            useProguard false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-unity.txt'
            signingConfig signingConfigs.debug
        }
    }
    packagingOptions {
        doNotStrip '*/armeabi-v7a/*.so'
        doNotStrip '*/x86/*.so'
    }
}

接著打開Unity項目的清單文件AndroidManifest.xml,刪減application節點,刪除intent-filter節點,activity增加內容process(解決某某問題)

<application
    //刪減其他
    android:banner="@drawable/app_banner"
    android:isGame="true">
    <activity
        .....
        //刪除下麵兩行-否則造成桌面兩個icon
        //android:label="@string/app_name"
        //android:launchMode="singleTask"
        //增加這行
        android:process="e.unity3d">
        //刪掉intent-filter
        <!--<intent-filter>-->
            <!--<action android:name="android.intent.action.MAIN" />-->
            <!--<category android:name="android.intent.category.LAUNCHER" />-->
            <!--<category android:name="android.intent.category.LEANBACK_LAUNCHER" />-->
        <!--</intent-filter>-->
        <meta-data
            android:name="unityplayer.UnityActivity"
            android:value="true" />
    </activity>
 ......

至此,配置完成,已經可以玩了。

但為了玩的順暢,我們再增加一個自定義控制項(1.用來解決kill問題 2.增強自己的擴展性)。自定義一個java文件集成UnityPlayer(核心關鍵類)

公眾號【空名先生】

public class MUnityPlayer extends UnityPlayer {
    public MUnityPlayer(Context context) {
        super(context);
    }
    @Override
    protected void kill() {
        //super.kill();
        //unity預設一些返回操作等會直接kill掉進程,覆寫kill方法,去掉super.kill, 不讓他kill
    }
}

配置完成,點擊菜單欄的Build,Rebuild Project unity的lib項目中生成aar文件

公眾號【空名先生】公眾號【空名先生】

Unity導出的項目配置完成,配置自己的Android項目,首先將幾個lib......so複製到你自己的項目jniLibs中。

然後在你需要集成的項目build文件中

implementation(name: 'XingFeiUnity', ext: 'aar')
接著將aar複製的你的項目libs文件下,記得重命名刪除後面的“-debug”,否則會出現找不到類的問題。

一波騷操作搞定,跑起來已經基本可以了。但是這就結束了嗎?不可能,騷起來我們就停不下來。

佈局渲染

我們需要在任意佈局載入3d,怎麼個任意法?就是找個view來addview 既不影響3d 還有android原生界面

//xml佈局
<LinearLayout
        android:id="@+id/linear"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_margin="50dp"
        android:orientation="vertical" />

//java文件
mLinear.removeAllViews();
mLinear.addView(mUnityPlayer.getView());
mUnityPlayer.requestFocus();

操作效果請看圖片,全部具體代碼的話 就直接放到Github

公眾號【空名先生】

通信交互

Android調用Unity

//參數二是 Unity中的方法名   參數一是哪個物體掛在了這個C#腳本  參數三 字元串
UnityPlayer.UnitySendMessage("Main Camera", "AndroidCallUnity", "");

公眾號【空名先生】

Unity調用Android 仔細看註釋

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class Manager : MonoBehaviour
{
    private AndroidJavaObject m_androidObj = null;
    public GameObject diqiu;

    void Start()
    {
        //註意-情況不同 com.unity3d.player.UnityPlayer  可能不同,可參考其他博主
        AndroidJavaClass androidClass = new AndroidJavaClass("com.unity3d.player.UnityPlayer");
        m_androidObj = androidClass.GetStatic<AndroidJavaObject>("currentActivity");
        diqiu.SetActive(false);
    }

    //Unity中的某個物體出發此事件
    public void UnityCallAndroid()
    {
        Debug.Log("調用方法");
        if (m_androidObj != null)
        {
            Debug.Log("調用方法進來");
            // 第一個參數是android裡面java代碼的方法名,第二個是攜帶的字元串參數
            m_androidObj.Call("CallAndroid", "我是Unity,我給你發消息了");
        }
    }

    //Android調用Unity-方法名一定要註意
    public void AndroidCallUnity(string json)
    {
        if (diqiu.activeInHierarchy)
        {
            diqiu.SetActive(false);
        }
        else
        {
            diqiu.SetActive(true);
        }
    }
}

場景切換

  • Unity內部自己去處理,就跟玩游戲一樣,讓Unity開發自己去做
  • andorid觸發,unity換場景,這個藉助上面說的消息通信來實現
  • 這還有一個技巧,如果資源不是很多且在一個場景的話,可以讓Unity一次直接渲染出來存在字典里,然後想顯示哪個Android給Unity發消息,Unity根據訂好的消息,展示不同的內容,這個好處就是切換展示速度極快。重點處理一下剛啟動的時候的耗時即可。

動態資源

態載入資源的問題,因篇幅有限,咱先只提供個思路,Unity支持讀取android設備的存儲文件,讓他們處理即可,然後android發消息告訴他們地址即可

// 參數一是Unity中的物體名稱,參數二Unity中的方法名  參數三路徑字元串
UnityPlayer.UnitySendMessage("AndriodMethodMgr", "CallUnitySetPath", Environment.getExternalStorageDirectory() + "");

啊哈,到這基本就結束了。快了又開心。

坑中帶坑

為了愉快的裝逼,最好還是看一看遇到的這些問題,能至少省幾天時間。

1. 混淆問題,如果你開啟了混淆,切記 切記,把混淆添加進入,這個大坑耽誤我好久啊

2. 如果模型在unity中沒問題,在android端穿幫,可以看看發佈質量,將android的設置成高的

公眾號【空名先生】
3. 如果反覆執行的模型動畫不對,怎麼不對?舉例心臟跳動,這是非常註重動畫的銜接的,如果銜接時間不對會造成心臟動畫的抖動,這會非常的明顯。

可以看動畫的setting 退出時間,退出時間是比例(如下圖),1代表全部動畫,0.5代表動畫使勁按的一般。過度時間前後動畫重疊(好像預設.95?)  可以改成0,如圖設置

公眾號【空名先生】

4. 集成到apk後 申請了橫豎屏 但是apk沒作用,是unity發佈出的設置導致的,再Unity導出的時候 other setting中設置宣傳方向

5. 許可權問題,上面說了會導出一個android項目,你仔細看這個項目的AndroidManifest文件,你會發現也有許可權。

註意,這時候比如你的android項目有許可權A 這個Unity導出的項目沒有許可權A,當你集成合併之後,導致最終的apk沒有許可權A,這並不是我們想看到的;

所以為了 解決這個問題,很簡單,我們把兩個清單文件的許可權保持一致即可,記住啊,否則怎麼哭的都不知道。

6. 還有一個未解決的問題,放到這,有朋友知道的話,感謝指教。

帶有動畫的一個物體,在有的android設備上,動畫表現徵程,但是有的會出現動畫跳動的情況,感覺像是電視的進度條那在跳進度一樣。

結尾

最後,別問我為啥知道這麼多問題,問就是因為自己跪著走過來的。

公眾號【空名先生】


您的分享是我們最大的動力!

更多相關文章
  • 前提要述:參考書籍《MySQL必知必會》 利用空閑時間快速回顧一些資料庫基礎。 4.1 連接 在最初安裝MySQL,可能會要求你輸入一個管理登錄(通常為root)和一個口令(密碼)。 連接MySQL需要以下信息: 主機名(電腦名)——如果連接到本地MySQL伺服器,為localhost; 埠(如 ...
  • 前提要述:參考書籍《MySQL必知必會》 《MySQL必知必會》是先講了查詢,但是沒有記錄就無法查詢,所以先將如何添加數據。 表已經知道怎麼創建了,隨便創兩張。 5.1 插入數據 MySQL使用 INSERT來插入(或添加)行(記錄)到資料庫表中。插入可用以下幾種方式使用: 插入完整的行(記錄); ...
  • #!/bin/bash echo "修改主機名" hostnamectl set-hostname wangxfa hostname sleep 1 echo "查看並關閉防火牆" systemctl status firewalld systemctl stop firewalld systemc ...
  • 屬性順序錯誤 一般情況下欄位類型要放在前面,限制參數放在後面,UNSIGNEDZEROFILL 之間沒有先後順序,主鍵 KEY 和 auto_increment 要放在UNSIGNED ZEROFILL 後面 否則報錯 。設置為主鍵的欄位沒有 deafault參數,not null 一般放在最後面。 ...
  • # 註釋內容 -- 註釋內容 -- 創建資料庫 king CREATE DATABASE king; -- 查看當前伺服器下有哪些資料庫 SHOW DATABASES; SHOW SCHEMAS; -- 查看king資料庫的詳細信息 SHOW CREATE DATABASE king; -- 創建數 ...
  • 提到Redis,大家肯定都聽過,並且應該都在項目中或多或少的使用過,也許你覺得Redis用起來挺簡單的呀,但如果有人問你下麵的幾個問題(比如同事或者面試官),你能回答的上來嗎? 1. 什麼是Redis? 2. Redis能存儲哪幾種數據結構? 3. Redis有幾種持久化機制?它們的優缺點分別是什麼 ...
  • 前提要述:參考書籍《MySQL必知必會》 2.1 MySQL簡介 2.1.1 什麼是MySQL MySQL是一種關係資料庫管理系統。負責資料庫中數據的存儲,檢索,管理和處理。 2.1.2 MySQL的優點 成本低——MySQL是開源的,一般可以免費使用。 性能——MySQL執行快。 可信賴,體積小。 ...
  • 個人博客 "http://www.milovetingting.cn" AOP之AspectJ的簡單使用 AOP的定義 AOP為Aspect Oriented Programming的縮寫,意為:面向切麵編程,通過預編譯方式和運行期間動態代理實現程式功能的統一維護的一種技術。 以上關於AOP的定義引 ...
一周排行
  • 微信公眾號: "Dotnet9" ,網站: "Dotnet9" ,問題或建議: "請網站留言" , 如果對您有所幫助: "歡迎贊賞" 。 .NET CORE(C ) WPF 抽屜式菜單 閱讀導航 1. 本文背景 2. 代碼實現 3. 本文參考 4. 源碼 1. 本文背景 使用簡單動畫實現抽屜式菜單 ...
  • 在上面abp(net core)+easyui+efcore實現倉儲管理系統——ABP WebAPI與EasyUI結合增刪改查之八(三十四) 文章的學習之後。我們通過前面的八篇文章已經學習了通過WebAPI介面與控制器去實現新增、刪除與修改功能。接下來,我們要在控制器中實現查詢功能。 ...
  • 1.選中項目-->屬性-->生成-->選中 XML文檔文件(xml路徑和該項目相同) 2.選擇生成序列化程式集:自動/開 ...
  • 本筆記摘抄自:https://www.cnblogs.com/PatrickLiu/p/7640873.html,記錄一下學習過程以備後續查用。 一、引言 從今天開始我們開始講結構型設計模式,結構型設計模式有如下幾種:適配器模式、橋接模式、裝飾模式、組合模式、外觀模式、享元模式、代理模式。 創建型設 ...
  • C 中 ConfigureAwait 相關答疑FAQ 在前段時間經常看到園子里有一些文章討論到 ConfigureAwait,剛好今天在微軟官方博客看到了 "Stephen Toub" 前不久的一篇答疑 ConfigureAwait 的一篇文章,想翻譯過來。 原文地址:https://devblog ...
  • 想要實現二維數組中根據某個欄位排序,一般可以通過數組迴圈對比的方式實現。這裡介紹一種更簡單的方法,直接通過PHP函數實現。array_multisort() :可以用來一次對多個數組進行排序,或者根據某一維或多維對多維數組進行排序。詳細介紹可參考PHP手冊:https://www.php.net/m ...
  • 常用的軟體: 播放器: cloundMusic(網易雲音樂) https://music.163.com/#/download PotPlayer(一款強大的視頻播放器) https://daumpotplayer.com/download/ ACDsee(ACDsee圖片編輯器免費版) https ...
  • 發現問題 在一次偶然中,在爬取某個網站時,老方法,打開調試工具查看請求方式,請求攔截,是否是非同步載入,不亦樂乎,當我以為這個網站非常簡單的時候,發現二級網頁的地址和源碼不對應 Ajax非同步載入?源碼也是這樣的 而且這些鏈接直... ...
  • 準備年後要跳槽,所以最近一直再看面試題,並且把收集到的面試題整理了以下發到博客上,希望對大家有所幫助。 首先是集合類的面試題 1. HashMap 排序題,上機題。 已知一個 HashMap<Integer,User>集合, User 有 name(String)和 age(int)屬性。請寫一個方 ...
  • JVM體繫結構圖 Native Interface(本地介面) Java本地介面(Java Native Interface (JNI))允許運行在Java虛擬機(Java Virtual Machine (JVM))上的代碼調用本地程式和類庫,或者被它們調用,這些程式和類庫可以是其它語言編寫的,比 ...
x