Unity与Android交互(3)——需要了解的Andorid基础知识

这篇具有很好参考价值的文章主要介绍了Unity与Android交互(3)——需要了解的Andorid基础知识。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【前言】

在上一篇文章中只是说了如何实现unity和android交互的问题,要了解其中的原理还必须要先了解一些Android的基础知识,了解后也能搞清楚如何接入SDK或者写Native插件。

(以下只是简要介绍,详细的内容需要自己去看链接)

【Android四大组件】

Activitiy

它提供了一个窗口,可以包含用户界面相关的组件,主要用于和用户进行交互,这个窗口通常会填满整个屏幕(可以将Activity理解为Unity中的Canvas)。Android应用程序通常由多个彼此松散绑定的Activity组成,通常,会有一个Activity被指定为MainActivity,启动应用程序时该Activity被激活呈现画面给用户(类似游戏中的主界面),每个Activity可以启动另一个Activity(类似在一个Canvas中打开另一个Canvas),Activity之间的切换基于堆栈机制(这个机制由Android系统实现,游戏中一般是在UIManager中自己实现),Activity之间通过Intent通信。这些Activity需要在AndroidM

在Unity游戏中,通常只有一个Activity,即UnityPlayerActivity。这个Activity会作为游戏的主界面,在启动游戏时被创建并显示,而且在游戏运行期间一直保持活跃状态。UnityPlayerActivity负责加载Unity引擎,并协调游戏界面和游戏逻辑的交互。在Unity中,所有的UI元素、游戏场景、特效等都是通过Unity引擎进行渲染和展示的,因此不需要像原生Android应用那样创建多个Activity来管理多个窗口或界面。

Activity有自己的生命周期,在这个生命周期中进行调用,与Unity的生命周期中的部分关联起来,即可让游戏运行起来。

Unity与Android交互(3)——需要了解的Andorid基础知识

Service

Service非常适用于去执行那些不需要和用户交互而且还要长期运行在后台的任务,例如播放音效、下载文件等。在Android中,后台的运行是完全不依赖UIService一般默认是后台的,其运行时不依赖任何用户界面,即使Activity被销毁,程序被切换到后台或者打开另外一个应用程序,Service仍然可以保存正常运行(这时Acvivity已经停止运行了)。只有当应用程序进程被杀掉时,所有依赖于该进程的Service才会停止运行。

注意,Service是运行在主线程中的,也即生命周期由主线程控制,所有不能在Service中做耗时长的操作,这会导致主线程阻塞,引发引发ANRApplication Not Responding)异常。耗时操作,通常在Service中开子线程来完成的。

在Android系统中,Service的优先级较低,当系统出现内存不足情况时,就有可能会回收掉正在后台运行的Service。如果希望Service可以一直保持运行状态,而不会由于系统内存不足的原因导致被回收,就可以考虑使用前台Service。前台Service会一直有一个正在运行的图标在系统的状态栏显示,下拉状态栏后可以看到更加详细的信息,非常类似于通知的效果。

Content Provider

Android的数据存储方式总共有五种,分别是:Shared Preferences、网络存储、文件存储、外储存储、SQLite。但一般这些存储都只是在单独的一个应用程序之中达到一个数据的共享,有时候我们需要操作其他应用程序的一些数据,就会用到ContentProvider。其以相对安全的方式封装数据(表)并且提供简易的处理机制和统一的访问接口供其他程序调用。而且Android为常见的一些数据提供了默认的ContentProvider(包括音频、视频、图片和通讯录等)。

如果自己的应用程序里的数据需要其他应用程序访问,需要自己实现ContentProvider,通过uri(统一资源定位符,Universal Resource Identifier)来标识哪些数据其他应用程序可以访问,每个资源都有一个uri。其他应用程序通过ContentResolver来访问ContentProvider提供的数据,通过uri来定位自己要访问的数据,还可以通过ContentProvider来监听ContentProvider的数据变化。(UnityAddressable有和这套机制相似的概念)

Broadcast Receiver

看到广播,不管是在哪,我们都知道是用于做跨模块通信的,在Android中用于应用内多个不同组件之间、不同应用组件之间的消息通信。广播分为有序广播,粘性广播、本地广播、系统广播等。像手机开启、网络状态改变、开始充电、屏幕开启关闭等都会发出系统广播 ,注册了相关广播的app就能知道了,广播被封装在Intent里面,会包含广播类型、广播参数等必要信息。

在发布到Android平台的Unity游戏中,是否需要使用Service、ContentProvider、broadcast receiver,取决于游戏的具体需求。Unity提供了Android Java插件的机制,让我们可以使用Java代码来创建和管理这些组件以实现更加复杂的功能。

【AndroidManifest.xml】

每个Android的应用程序都必须包含一个 AndroidManifest.xml,且文件名是固定的,不能修改。应用程序需要通过它向Android系统提供一些必需的信息,且需要在运行前提供给系统。这些信息包括:

  1. 应用的软件包名称,其通常与代码的命名空间相匹配
  2. 应用的组件,即所需使用的四大组件。每个Activity、Service、Content Provider都需要在文件中进行配置,未配置则不会启动,也即用不了。而Broadcast Receiver的注册分静态注册(在AndroidManifest文件中进行配置)和通过代码动态创建并以调用Context.registerReceiver()的方式注册至系统。静态注册会随系统的启动而一直处于活跃状态,即使程序未运行,只要接收到感兴趣的广播就会触发。
  3. 应用为访问系统或其他应用的受保护部分所需的权限
  4. 应用需要的硬件和软件功能

AndroidManifest文件结构

文件结果官网上说的很详细,不懂的直接在官网上搜即可,这点是必须要看的

Unity如何生成AndroidManifest

首先,Unity有一个默认的AndroidManifest文件,位于:

Unity\Editor\Data\PlaybackEngines\AndroidPlayer\Apk

其次,自己可以根据需要重写AndroidManifest文件:

Unity与Android交互(3)——需要了解的Andorid基础知识

再次,所有接入的插件或SDK可能以后自己的的AndroidManifest文件

最后,在构建应用时,Unity会将上述所有的文件合成一个文件,修改清单,自动向清单添加权限、配置选项、使用的特性和其他信息,生成最终的AndroidManifest文件。

实例AndroidManifest解析

以之前的例子打包出来的apk的文件并添加些其他东西作为例子进行解析

<?xml version="1.0" encoding="utf-8" standalone="no"?>                                                                                      <!-- 安装位置是外部存储 -->                 <!-- 包名,在unity中填写的 --> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" android:compileSdkVersion="32" android:compileSdkVersionCodename="12" android:installLocation="preferExternal" package="com.test.UnityTest" platformBuildVersionCode="32" platformBuildVersionName="12">
    <supports-screens android:anyDensity="true" android:largeScreens="true" android:normalScreens="true" android:smallScreens="true" android:xlargeScreens="true"/>
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> <!-- 读取外部存储权限 -->
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <!-- 写入外部存储权限 -->
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> <!-- 挂载文件系统权限 -->
    <uses-permission android:name="android.permission.WRITE_MEDIA_STORAGE"/> <!-- 写入媒体存储权限 -->
    <uses-feature android:glEsVersion="0x00030000"/>
    <uses-feature android:name="android.hardware.vulkan.version" android:required="false"/>  <!-- uses-feature表示需要使用的硬件,使用vulkan渲染 --> 
    <uses-permission android:name="android.permission.INTERNET"/> <!-- 网络权限 -->
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/> <!-- 定位权限 -->
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/> <!-- 定位权限 -->
    <uses-feature android:name="android.hardware.location.gps" android:required="false"/> <!-- 需要使用gps定位 -->
    <uses-feature android:name="android.hardware.location" android:required="false"/> <!-- 使用定位 -->
    <uses-feature android:name="android.hardware.touchscreen" android:required="false"/> <!-- 使用触摸屏 -->
    <uses-feature android:name="android.hardware.touchscreen.multitouch" android:required="false"/> <!-- 使用多点触摸 -->
    <uses-feature android:name="android.hardware.touchscreen.multitouch.distinct" android:required="false"/>   <!-- 使用多点触摸 -->
    <uses-permission android:name="android.permission.READ_MEDIA_IMAGES"/> <!-- 读取媒体图片权限 -->
    <uses-permission android:name="android.permission.READ_MEDIA_VIDEO"/>   <!-- 读取媒体视频权限 -->
    <uses-permission android:name="android.permission.READ_MEDIA_AUDIO"/>   <!-- 读取媒体音频权限 -->
    <application android:debuggable="true" android:extractNativeLibs="true" android:icon="@mipmap/app_icon" android:label="@string/app_name" android:requestLegacyExternalStorage="true">
        <activity android:configChanges="density|fontScale|keyboard|keyboardHidden|layoutDirection|locale|mcc|mnc|navigation|orientation|screenLayout|screenSize|smallestScreenSize|touchscreen|uiMode" android:exported="true" android:hardwareAccelerated="false" android:launchMode="singleTask" android:name="com.unity3d.player.UnityPlayerActivity" android:resizeableActivity="false" android:screenOrientation="fullSensor" android:theme="@style/UnityThemeSelector"> 
            <intent-filter>
                <action android:name="android.intent.action.MAIN"/> <!-- activity中有这一行表示其被设置为Main Activity 启动应用时首先显示哪一个Activity --> 
                <category android:name="android.intent.category.LAUNCHER"/><!-- 表示activity应该被列入系统的启动器(launcher)(允许用户启动它)。Launcher是安卓系统中的桌面启动器,是桌面UI的统称。 -->
            </intent-filter>
            <meta-data android:name="unityplayer.UnityActivity" android:value="true"/> <!-- meta-data是一种键值对,用于向组件提供配置信息,这里表示该activity是unity的activity -->
            <meta-data android:name="android.notch_support" android:value="true"/> <!-- notch_support表示是否支持刘海屏 -->
        </activity>
        <meta-data android:name="unity.splash-mode" android:value="0"/>     <!-- 0表示不显示splash screen -->
        <meta-data android:name="unity.splash-enable" android:value="true"/> <!-- true表示显示splash screen -->
        <meta-data android:name="unity.allow-resizable-window" android:value="false"/> <!-- false表示不允许改变窗口大小 -->
        <meta-data android:name="notch.config" android:value="portrait|landscape"/> <!-- notch.config表示刘海屏的配置,portrait表示竖屏,landscape表示横屏 -->
        <meta-data android:name="unity.build-id" android:value="2dd52062-f22a-4914-98d2-35cfe8574afc"/>     <!-- build-id表示构建id -->
    </application>
</manifest>

【UI线程】

在Android中UI线程是一个应用程序的主线程, 其在app启动时就会被创建(即每个app都有一个UI线程),Activity中涉及到UI组件的更新必须在主线程中进行(这和UnityAPI只能在主线程中使用是一个道理)。从其他线程的调用如果最终涉及到UI组件,就会报错甚至崩溃。其他线程调用UI线程的组件实际上就是多线程之间的通信,Android 提供了几种途径来从其他线程访问 UI 线程,在Unity中一般只用到Activity.runOnUiThread(Runnable)

runOnUiThread()方法是Activity类的一个方法,可以通过当前Activity对象来调用。该方法接受一个Runnable对象作为参数,该Runnable对象中的run()方法会在主线程中执行。如果当前线程时UI线程,那么会立即执行,如果不是那会发到UI线程的一个事件队列里等待执行。其源码如下:

public final void runOnUiThread(Runnable action) {
    if(Thread.currentThread() != mUiThread) {
        mHandler.post(action);
    } else {
        action.run();
    }
}

前文说过,Unity并不使用Android来渲染UI界面,所以Unity有自己的主线程,和Android的UI线程不是同一个线程,但我们可以在UI线程中通过UnitySendMessage发消息给Unity主线程(如上一篇文章所示)。像播放广告、浏览网页等,一般用的是Android UI组件是实现,注意要在UI线程中进行一些处理,需要用到runOnUiThread。

【arr与jar】

Android工程中, app最终被编译打包成能在Android设备上运行的APK文件,Android library在目录结构上与Android App相同。

Unity与Android交互(3)——需要了解的Andorid基础知识

其包含构建APP所需的一切(如源代码、资源文件、Android Manifest),但在构建时被编译成供其它Android App依赖的Android Archive (AAR)文件,也即多个app可以使用同一个arr文件(类似Unity中的Package)。

一个arr文件包含了源代码、资源文件、Android Manifest等,将arr文件解压出来,可以得到只包含源代码的Java Archive(JAR)文件。

Unity与Android交互(3)——需要了解的Andorid基础知识

【静态链接库与动态链接库】 

静态库和动态库都是从其他工程里build出来的可重定位目标文件,里面都是二进制的代码,可以被链接生成可执行文件。

在构建形成可执行文件的链接步骤中,会将自己的代码和从静态链接库拷贝出来的代码(仅需要拷贝自己代码使用的那部分库代码,没用的不拷贝)形成一个整体的代码,对动态链接库只会拷贝一些重定位和符号表信息。

在程序运行时,静态链接库中的代码和自己的代码作为一个整体被加载到内存中,而动态链接库在使用时根据重定位和符号表信息加载库并找到需要调用的函数。

因此,对于同样的一份代码,使用静态库构建出来的可执行文件比用动态库的大,多个不同的可执行文件使用了同样的静态库,那么内存中会有多份静态库的代码,而动态库只会有一份。

Linux/Unix 系统里静态库扩展名一般是 .a(archive),动态库扩展名一般是 .so(share object)

Windows 系统里 VC 编译器用的静态库扩展名一般是 .lib,动态库扩展名一般是 .dll(dynamic link library)

Android so文件

Android基于Linux Kernl,开发Android应用时,有时候Java层的编码不能满足实现需求,就需要到C/C++实现后生成的so文件,再用System.loadLibrary()加载进行调用。常见的场景如:加解密算法,音视频编解码等。不同的CPU架构对所执行的二进制文件的规范是不同的,所以在生成so文件时,需要考虑适配市面上不同手机CPU架构。文章来源地址https://www.toymoban.com/news/detail-496117.html

【SDK的生成】

  1. 编写源代码:一般使用C/C++编写需要实现的功能代码,方便跨平台
  2. 编译源代码:使用编译器将源代码编译成库文件,如.so或.a文件
  3. 创建调用接口:如果是面向Android平台,需要Java Native Interface(JNI)创建Java接口,以便Java代码能够使用库文件中的函数和方法
  4. 打包SDK:如果是面向Android平台,将库文件和写好的Java接口打包成Android SDK,以便其他开发者使用
  5. 测试SDK:对生成的SDK进行测试,确保其能够正常运行
  6. 撰写SDK开发、API接口说明等文档
  7. 发布SDK

到了这里,关于Unity与Android交互(3)——需要了解的Andorid基础知识的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 百度二面是什么级别的,Unity进阶技巧 - RectTransform详解,android开发需要哪些基础

    三、Pivot和Anchor的结合 在了解了Pivot和Anchor分别是什么后,我们就来看看Unity是如何使用这个两个东西来控制UI的布局 第1种情况:两个锚点重合时我们先来看看两个锚点重合时的情况,这种情况是我们最常用也是最容易理解的方式 我们将Anchor锚点放在黑框的正中间,然后将

    2024年04月12日
    浏览(51)
  • 简单了解 HTTP 基础知识

    HTTP(Hypertext Transfer Protocol,超文本传输协议)是用于在网络上传输数据的一种协议,对于网络开发人员来说,理解这一协议是至关重要的。由于其广泛的应用,除了在网页应用传输数据之外,它还被应用于物联网(IoT)中的数据和命令传输。 HTTP 协议的第一个版本只有一个方

    2024年04月17日
    浏览(53)
  • 快速了解机器视觉(CV)基础知识

    最近再查一些基础知识的时候看见了几篇文章写得很棒(在这篇文章的结束我会给出参考链接),然后我把他们整合了一下,跟大家分享,希望能有帮助: a.图片分类 b.目标定位 c.语义分割 d.实例分割 ①语义鸿沟(semantic gap) 人类可以轻松地从图像中识别出目标,而计算机看到

    2024年02月09日
    浏览(83)
  • Android 开发中需要了解的 Gradle 知识

    作者:wkxjc Gradle 是一个基于 Groovy 的构建工具,用于构建 Android 应用程序。在 Android 开发中,了解 Gradle 是非常重要的,因为它是 Android Studio 默认的构建工具,可以帮助我们管理依赖项、构建应用程序、运行测试等。 本文将介绍 Android 开发中需要了解的一些 Gradle 知识,包括

    2024年02月13日
    浏览(31)
  • 一起来了解Git的基础知识叭~~

    $ git init Initialized empty Git repository $ ls -al to see git $ touch file01.txt Init a empty file01.txt file $ git status To see modifiyed status $ git add . remove all modifyed files to index from workspace $ git add file.txt(FILENAME) : remove single modifyed file to index from workspace $ git commit -m “Your annotation” commit index file to reposit

    2023年04月27日
    浏览(47)
  • 【大数据】了解 YARN 架构的基础知识

    Hadoop YARN( Y et A nother R esource N egotiator)将 Hadoop 的存储单元即 HDFS(Hadoop 分布式文件系统)与各种处理工具编织在一起。 在 Hadoop 1.0 版本,也称为 MRV1 ( MapReduce Version 1 ),MapReduce 执行处理和资源管理功能。它由一个 作业跟踪器 ( Job Tracker )组成,它是唯一的主机。作业

    2024年01月21日
    浏览(65)
  • 【敬伟ps教程】PS基础知识了解

    选中油漆桶工具 取色器中选择颜色 按住 Shift 点击灰色区域即可 或者直接右键灰色区域,选择颜色即可 如果面板被推拽的很乱,想恢复原来位置,可以点击:窗口–工作区–复位基本功能 当然工作区也可以自己新建,窗口–工作区–新建工作区,新建的工作区就会出现在右

    2023年04月11日
    浏览(62)
  • UI及交互设计知识整理(超详细 基础篇)

    首先,解决一个问题 UI 是 User Interface 的缩写,即界面设计。 其中 Interface 中的 inter 又有“ 在一起,交互 ”之义,所以UI设计中也包括了交互设计。 用更通俗易懂的话来说就是 符合用户需求的界面设计 。 然后开始我们的 设计基础知识介绍 这是包括项目经理、ui设计师、交

    2023年04月25日
    浏览(82)
  • 【Unity美术】Unity工程师对3D模型需要达到的了解【二】

    👨‍💻个人主页 :@元宇宙-秩沅 👨‍💻 hallo 欢迎 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍💻 本文由 秩沅 原创 👨‍💻 收录于专栏 :Unity基础实战 模型导入规范文档入口, .坐标轴规范: 人物面朝向为Z轴正方向,Y轴正方向为头顶方向,X轴正方向为人物右侧 一般式用

    2024年02月03日
    浏览(41)
  • 【Unity美术】Unity工程师对3D模型需要达到的了解【一】

    👨‍💻个人主页 :@元宇宙-秩沅 👨‍💻 hallo 欢迎 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍💻 本文由 秩沅 原创 👨‍💻 收录于专栏 :Unity基础实战 模型导入规范文档入口, .坐标轴规范: 人物面朝向为Z轴正方向,Y轴正方向为头顶方向,X轴正方向为人物右侧 一般式用

    2024年02月03日
    浏览(50)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包