2025-01-06 Unity 使用 Tip2 —— Windows、Android、WebGL 打包记录

文章目录

  • 1 Windows
  • 2 Android
    • 2.1 横版 / 竖版游戏
    • 2.2 API 最低版本
    • 2.3 目标帧率
      • 2.3.1 targetFrameRate
      • 2.3.2 vSyncCount
      • 2.3.3 Unity 默认设置以及推荐设置
      • 2.3.4 Unity 帧率托管
  • 3 WebGL
    • 3.1 平台限制
    • 3.2 打包报错记录 1
    • 3.3 打包报错记录 2

​ 最近尝试将写的小游戏打包,主要平台包括 Windows、Android 和 WebGL,以下是一些打包过程记录。

  • Unity 版本:6000.0.26f1c1

1 Windows

​ 不得不说,Windows 平台是真好,打包一切顺利,没有任何卡壳,太感动了〒▽〒。

​ 点击 Build And Run,直接无脑生成 exe。

image-20250106033655380

​ 小游戏的话不需要全屏,所以我设置了 Windowed。

image-20250106033350591

2 Android

​ 需要做一些处理,没有那么顺利。

2.1 横版 / 竖版游戏

​ 需要设置游戏是横版还是竖版,我将 Default Orientation 设置了 Auto Rotation,即自动翻转,并将 Allowed Orientations for Auto Rotation 仅设置 Landscape Right/Landscape Left,即可以横版左右翻转。

image-20250106034030506

​ 以下是相关设置说明:

  • Default Orientation:指定应用程序窗口在设备屏幕内的方向。

    可选项包括:

    1. Portrait:竖屏显示。
    2. Portrait Upside Down:竖屏倒立显示。
    3. Landscape Right:横屏显示(手机右侧朝下)。
    4. Landscape Left:横屏显示(手机左侧向下)。
    5. Auto Rotation:自动旋转。
  • Auto Rotation Behavior:自动旋转方式。

    可选项包括:

    1. User:用户可以锁定设备的自动旋转方向设置以关闭自动旋转。
    2. Sensor:依据设备传感器数据进行旋转,用户无法进行干预。

2.2 API 最低版本

​ Minimum API Level 必须手动设置,一般设置为 API Level 33,或者依据 Build 提示设为其指定值。我这里依据 Build 提示,设置为 API Level 34。

image-20250106035322270

​ 有意思的是,Unity 默认会选很古老的 API,例如 Android 8.0(API Level 26)。

​ 但奇怪的是,在 Unity Hub 下载 Android 模块时,往往只下载了 API Level 33 ~ 35,这就非常搞鬼了,不手动设置的话,基本第一次 Build 都会报错,因为压根没装那么低版本的 API。

image-20250106035718633

2.3 目标帧率

​ 通过 targetFrameRate(软件计时、手动设置)和 vSyncCount(硬件支持、自动设置)两种方式进行控制,只会同时生效其中一种(不考虑 VR 设备):

  • vSyncCount = 0 时,targetFrameRate 生效。
  • vSyncCount != 0 时,vSyncCount 生效。

2.3.1 targetFrameRate

targetFrameRate 是指定 Unity 尝试渲染游戏的目标帧速率,需要在程序运行之初使用代码进行设置。参考官方文档:https://docs.unity3d.com/6000.0/Documentation/ScriptReference/Application-targetFrameRate.html。

​ 不做任何手动设置时,Android 固定 30 fps 渲染(即 targetFrameRate = 30),以节省电池电量,与显示器的本机刷新率无关。

说明:

  1. targetFrameRate 只是帧率上限,实际帧率只会比这个值低或等于。

    不是说设置为 120 fps 后,老年机也能跑 120 帧。

  2. targetFrameRate = -1,表示帧率不封顶,帧数能跑多高就会跑多高。

  3. 受到显示器最大刷新率的限制,实际帧率不会超过显示器的本机刷新率。

    如果 targetFrameRate = 120,但是手机屏幕最高仅支持 90 帧,那么实际帧率最大也只能是 90。

  4. 在 Android 和 iOS 上,targetFrameRate 将向下舍入到能够整除显示器当前刷新率的最大值。

    例如,在 60Hz Android 显示器上运行时,且设置 targetFrameRate = 25 ,则设备帧率最高为 20fps,因为 20 是小于 25 且可以整除 60 的最高数字。

  5. 在 Windows 和 Web 上,不建议使用 targetFrameRate ,因为 targetFrameRate 基于软件计时,容易出现轻微卡顿现象。

​ 最初不知道如何设置的时候,导出 APK 在手机上运行时只有 30 帧超级卡,我还以为是自己性能优化没做好 hh。

2.3.2 vSyncCount

vSyncCount 表示每帧之间应传递的垂直同步数,即游戏允许在帧之间传递的屏幕刷新次数。范围是 0~ 4 内的整数,默认为 1。参考官方文档:https://docs.unity3d.com/6000.0/Documentation/ScriptReference/QualitySettings-vSyncCount.html。

​ 假设目标设备显示器的刷新率为 90 fps,当 vSyncCount > 0 时,游戏目标帧为 90 / vSyncCount

vSyncCount 能够在 “Project Settings” -> “Quality”-> “VSync Count” 中进行设置:

  • Don’t Sync:即 vSyncCount = 0
  • Every V Blank:即 vSyncCount = 1
  • Every Second V Blank:即 vSyncCount = 2
image-20250106043532111

说明:

  1. 在 Android 和 iOS 上,vSyncCount 始终被忽略,因为移动设备不允许不同步渲染(使用 targetFrameRate 控制帧速率)。
  2. 在 Windows 和 Web 上,建议使用 vSyncCount,因为其基于硬件同步机制,能够产生完全无卡顿的输出。

2.3.3 Unity 默认设置以及推荐设置

平台Unity 默认设置推荐设置
WindowsvSyncCount = 0, targetFrameRate = -1
(帧率不封顶,不同步渲染)
默认即可
Android / IOStargetFrameRate = 30
(固定 30 帧渲染)
targetFrameRate = 60
WebGLvSyncCount = 1
(垂直同步,以本机显示刷新率渲染)
默认即可

​ 实现代码:

public class GameStart : MonoBehaviour
{
    private void Awake()
    {
#if PLATFORM_ANDROID
        QualitySettings.vSyncCount  = 0;
        Application.targetFrameRate = 60;
#endif
    }
}

2.3.4 Unity 帧率托管

​ 在 “Project Settings” -> “Player”-> “Resolution and Presentation” -> “Resolution” 中,可勾选 “Optimized Frame Pacing”,从而让 Unity 均匀分布帧,以减少帧速率的差异并创造更流畅的体验,提供更平滑的帧率表现。

image-20250106045000717

​ 但参考该文:https://blog.csdn.net/m0_63261863/article/details/143192236,勾选可能导致游戏降频,建议结合自己情况勾选。

3 WebGL

​ WebGL 平台限制是真多,导出过程真的快吐了。

​ 但优点是链接点开即玩,所以没有办法。

3.1 平台限制

  1. 不支持访问本地文件。

    对资产数据和 AssetBundle 的网络请求会缓存在浏览器缓存中。

  2. 不支持多线程。

    不支持 C# System.Threading 命名空间中的任何内容。

  3. 不支持System.Net命名空间内的 .NET 网络类。

  4. 不支持 Unity 内部 AudioSource 与 AudioClip 的部分 API。

    这将导致,在 Unity 编辑器中使用 Addressables 模拟真机 AB 包加载音频时,无法正确播放。只能使用 Use Asset Database,或者将音频文件放在 Resources 文件夹下(不建议这样做)。

image-20250106050823395
  1. 不允许使用 System.Reflection.Emit 动态生成代码。

  2. 不支持同步加载资源,只能异步加载。

3.2 打包报错记录 1

​ 一次偶然的机会,看到 Low 选项,不自觉设置为了 Medium。

image-20250106051828069

​ 于是 WebGL 下打包报错,获取不到单例的非公共构造函数。

​ 报错代码:FrameworkException: Non-Public Constructor() not found! in Framework.Toolkits.PoolKit.SingletonObjectPool`1[Framework.Toolkits.AudioKit.AudioPlayer] at Framework.Toolkits.SingletonKit.SingletonCreator.CreateNonPublicConstructorObject[T] () [0x00000] in <00000000000000000000000000000000>:0

image-20250106052012118

​ 查原因后,发现 Strip Engine Code 等级为 Medium 及以上时,会对反射的代码产生影响。而恰好,我的单例是通过反射获取私有无参构造函数创建的(官方链接:https://docs.unity3d.com/6000.0/Documentation/Manual/class-PlayerSettingsWebGL.html)。解决方案就是将 Strip Engine Code 等级调回 Low 即可。

image-20250106052249345
  • Strip Engine Code:仅适用于 IL2CPP 脚本后端。

    勾选会剔除应用程序不使用的 DLL,以减少内置播放器的大小,建议勾选。

  • Managed Stripping Level:控制剔除程度。

    1. Minimal

      使用此选项以剥离类库、UnityEngine、Windows 运行时程序集,并复制所有其他程序集。

    2. Low

      删除无法访问的托管代码,减少构建大小和 Mono/IL2CPP 构建时间。

    3. Medium

      运行 UnityLinker 以减少代码大小,剔除范围大于 Low 选项,且某些反射代码路径的行为可能不同(支持自定义 link.xml 文件)。

    4. High

      运行 UnityLinker 进一步减少代码大小,剔除范围大于 Mediumm 选项,某些反射代码路径的行为可能不同,同时某些方法的托管代码调试可能不再起作用(支持自定义 link.xml 文件)。

3.3 打包报错记录 2

​ 有次无意间不满意新写的代码,直接将老版本代码(unitypackage)覆盖导入 Unity,之后再进行 WebGL 打包时发生如下报错:

Build completed with a result of 'Failed' in 129 seconds (129224 ms)
Building Library\Bee\artifacts\WebGL\build\debug_WebGL_wasm\build.js failed with output:

P:\Unity Project\Mine\Survivor Game 2.0>set MYDIR=S:\Unity Editor\6000.0.26f1c1\Editor\Data\PlaybackEngines\WebGLSupport\BuildTools\Emscripten\emscripten\ 

P:\Unity Project\Mine\Survivor Game 2.0>goto FOUND_MYDIR 
wasm-ld: warning: function signature mismatch: UxmlFactory__ctor_m87E7DA1E842A9B4B4A45132392EDA2868D62B8A2
>>> defined as () -> void in P:/Unity Project/Mine/Survivor Game 2.0/Library/Bee/artifacts/WebGL/il2cppOutput/build/GameAssembly.a(3yoe1qmioyer.o)
>>> defined as (i32, i32) -> void in lto.tmp

wasm-ld: warning: function signature mismatch: VisualChangesProcessor_get_elementBuilder_m759F6B67F45FB60E9B1E4C646F26689258317B1E
>>> defined as () -> void in P:/Unity Project/Mine/Survivor Game 2.0/Library/Bee/artifacts/WebGL/il2cppOutput/build/GameAssembly.a(lsyvkdbleliz.o)
>>> defined as (i32, i32) -> i32 in lto.tmp

wasm-ld: warning: function signature mismatch: Area__ctor_m6128DC659857264FD4FBF06C01E0C2F04EAA4105
>>> defined as () -> void in P:/Unity Project/Mine/Survivor Game 2.0/Library/Bee/artifacts/WebGL/il2cppOutput/build/GameAssembly.a(kkef0xumaj1j.o)
>>> defined as (i32, i32, i32) -> void in lto.tmp

wasm-ld: warning: function signature mismatch: SingleQueryMatcher_IsInUse_mF17B2C22806E732E50055D4F6AC8A0E3ABEB4B99
>>> defined as () -> void in P:/Unity Project/Mine/Survivor Game 2.0/Library/Bee/artifacts/WebGL/il2cppOutput/build/GameAssembly.a(z4arkogp7up0.o)
>>> defined as (i32, i32) -> i32 in lto.tmp

...

​ 最初认为是项目中部分插件不支持 WebGL 平台,倒腾一番后,又重装了 WebGL 打包模块,依然是同样的报错。

​ 解决方案:删除该项目下的 Library 文件夹,Unity 重新打开项目,没有报错。

image-20250106053758010

​ 分析原因:Library 中缓存的以前打包的结果,导致代码链接失败。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/949443.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Deep blind super-resolution for hyperspectral images_译文

关键词&#xff1a; 高光谱图像 盲超分辨率 退化模型 深度学习 摘要 目前单张高光谱图像超分辨率的深度学习方法都是非盲方法&#xff0c;采用简单的双三次退化模型。这些模型泛化性能较差&#xff0c;无法处理未知的退化。此外&#xff0c;RGB图像的盲超分辨率方法忽略了高光…

Visual studio code编写简单记事本exe笔记

安装扩展cmake tools c/c c/c Extension pack CMakeLists.txt cmake_minimum_required(VERSION 3.20) project(NotepadApp)set(CMAKE_CXX_STANDARD 17) set(CMAKE_CXX_STANDARD_REQUIRED ON)# Windows specific settings if(WIN32)set(CMAKE_WINDOWS_EXPORT_ALL_SYMBOLS ON)s…

Java100道面试题

1.JVM内存结构 1. 方法区&#xff08;Method Area&#xff09; 方法区是JVM内存结构的一部分&#xff0c;用于存放类的相关信息&#xff0c;包括&#xff1a; 类的结构&#xff08;字段、方法、常量池等&#xff09;。字段和方法的描述&#xff0c;如名称、类型、访问修饰符…

【数电尾灯设计】2022-8-16

缘由数电尾灯设计问题&#xff0c;求解答--CSDN问答 从题目可以列出 000 100 010 111-----------4进制 000 100 010 110 001 101 011 111-----------8进制 由列出可知用16进制芯片的3个引脚可以获得8进制推导出4进制从而可用逻辑处理为4进制实现尾灯功能。之上第一步实现了尾灯…

安卓14无法安装应用解决历程

客户手机基本情况&#xff1a; 安卓14&#xff0c;对应的 targetSdkVersion 34 前天遇到了安卓14适配问题&#xff0c;客户发来的截图是这样的 描述&#xff1a;无法安装我们公司的B应用。 型号&#xff1a;三星google美版 解决步骤&#xff1a; 1、寻找其他安卓14手机测试…

【WRF数据准备】ECMWF 49r1: Soil Variables的变化及WRF模型修正

目录 ECMWF 49r1: Change in soil variablesECMWF 49r1更新的背景土壤变量的主要变化对WRF使用者的影响Github中描述ERA5 Vtable 下载另:原始 IFS 数据-ECMWF 服务器参考2024年12月12日,ECMWF 发布了 49r1 版本的业务 IFS。ECMWF在其49r1循环版本中对土壤变量进行了重要更新。…

一机多实例:如何在一台机器上高效运行多个 MySQL 服务

前言 在实际开发和测试环境中&#xff0c;我们经常需要运行多个 MySQL 实例来模拟不同的数据库环境。例如&#xff0c;在一台服务器上运行多个数据库服务以节约硬件资源&#xff0c;或者同时运行不同版本的 MySQL 进行功能兼容性测试。MySQL 本身支持通过配置多实例运行&#…

源代码编译安装X11及相关库、vim,配置vim(1)

一、目录结构 如下。 所有X11及相关库装到mybuild&#xff0c;源代码下载到src下&#xff0c;解压&#xff0c;进入&#xff0c;编译安装。编译时指定--prefix到相同的目录&#xff0c;即上图中mybuild。 ./configure --prefixpwd/../../mybuild [CFLAGS"-I/path/to/X11…

5. CSS引入方式

5.1 CSS的三种样式 按照 CSS 样式书写的位置(或者引入的方式)&#xff0c;CSS样式表可以分为三大类&#xff1a; 1.行内样式表&#xff08;行内式&#xff09; 2.内部样式表&#xff08;嵌入式&#xff09; 3. 外部样式表&#xff08;链接式&#xff09; 5.2 内部样式表 …

【C++】构造函数与析构函数

写在前面 构造函数与析构函数都是属于类的默认成员函数&#xff01; 默认成员函数是程序猿不显示声明定义&#xff0c;编译器会中生成。 构造函数和析构函数的知识需要建立在有初步类与对象的基础之上的&#xff0c;关于类与对象不才在前面笔记中有详细的介绍&#xff1a;点我…

WPF区域导航+导航参数使用+路由守卫+导航日志

背景&#xff1a;使用ContentControl控件实现区域导航是有Mvvm框架的WPF都能使用的&#xff0c;不限于Prism 主要是将ContenControl控件的Content内容在ViewModel中切换成不同的用户控件 下面是MainViewModel&#xff1a; private object body;public object Body {get { retu…

Unity中 Xlua使用整理(一)

1.安装: 从GitHub上下载Xlua源码 Tencent/xLua: xLua is a lua programming solution for C# ( Unity, .Net, Mono) , it supports android, ios, windows, linux, osx, etc. (github.com) 下载Xlua压缩包&#xff0c;并解压将Aseet文件夹中的Xlua和Plugins文件夹复制到Unit…

Matlab仿真径向受压圆盘光弹图像

Matlab仿真径向受压圆盘光弹图像-十步相移法 主要参数 % 定义圆盘参数 R 15; % 圆盘半径&#xff0c;单位&#xff1a;mm h 5; % 圆盘厚度&#xff0c;单位&#xff1a;mm P 300; % 径向受压载荷大小&#xff0c;单位&#xff…

基于Django的学校智能图书馆借书归还订阅管理系统

完整源码项目包获取→点击文章末尾名片&#xff01;

【设计模式-2】23 种设计模式的分类和功能

在软件工程领域&#xff0c;设计模式是解决常见设计问题的经典方案。1994 年&#xff0c;Erich Gamma、Richard Helm、Ralph Johnson 和 John Vlissides&#xff08;四人帮&#xff0c;GoF&#xff09;在《设计模式&#xff1a;可复用面向对象软件的基础》一书中系统性地总结了…

阿里云代理商热销产品推荐

在数字化浪潮的推动下&#xff0c;企业对于云计算的依赖日益加深。阿里云&#xff0c;作为中国领先的云计算服务提供商&#xff0c;为企业提供了丰富多样的云产品和服务。本文将聚焦于阿里云代理商热销产品推荐&#xff0c;探讨其如何帮助企业高效利用云资源&#xff0c;加速数…

[redux] 异步逻辑的两种写法

createAsyncThunk | Redux Toolkit 第一种, extraReducers 普通的reducers只能写同步代码 异步必须得用中间件的形式,就是异步代码调用完有结果了, 再调用同步的reducer, 大概这么理解, 第一种怎么用呢? 先用一个异步函数 const fetchUserById createAsyncThunk(users/fet…

在Java中使用有符号类型模拟无符号整数的技巧

有符号整数和无符号整数 有符号整数&#xff1a;可以表示正数、负数和零。例如&#xff0c;Java中的 byte 类型是有符号的&#xff0c;其范围是 -128 到 127.无符号整数&#xff1a;只能表示非负数&#xff08;即零和正数&#xff09;。例如&#xff0c;无符号 byte 应该表示的…

51单片机——8*8LED点阵

LED 点阵的行则为发光二极管的阳极&#xff0c;LED 点阵的列则为发光二极管的阴极 根据 LED 发光二极管导通原理&#xff0c;当阳极为高电平&#xff0c;阴极为低电平则点亮&#xff0c;否则熄灭。 因此通过单片机P0口可控制点阵列&#xff0c;74HC595可控制点阵行 11 脚 SR…

Flutter:邀请海报,Widget转图片,保存相册

记录下&#xff0c;把页面红色区域内的内容&#xff0c;转成图片后保存到相册的功能 依赖 # 生成二维码 qr_flutter: ^4.1.0 # 保存图片 image_gallery_saver_plus: ^3.0.5view import package:demo/common/index.dart; import package:ducafe_ui_core/ducafe_ui_core.dart; i…