2022年度大赏 | UWA问答精选

UWA每周推送的知识型栏目《厚积薄发 | 技术分享》已经伴随大家走过了304个工作周。精选了2022年十大精彩问答分享给大家,期待2022年UWA问答继续有您的陪伴。


Q1:动态获取URP设置里自定义的RenderFeatures

我们在URP项目中自定义了多个RenderFeatures去实现游戏效果,现在想动态开启和关闭相关的RenderFeatures,请问怎么在代码里动态获取这些Features呢?

A:我们使用反射获取,供参考:

感谢yqsas@UWA问答社区提供了回答

Q2:Skybox的环境照明问题

A场景,通过additive的方式加载B场景。2个场景中都没有任何灯光(包括平行光)、Reflection Probe,且Enviroment Reflections的Intensity Mulitiplier为0,纯靠Enviroment Lighting中的Skybox进行照明。

但是SetActive为第二个场景之后,就会发现照明黑色,如右图所示。重新SetActive为第一个场景,环境照明正确,如左图所示。

如果模式不为Skybox而是Color,则没有上述问题。目前打算自建环境光球谐信息,而不使用unity_SHAr相关数据。有什么比较好的解决方案吗?

A:当把场景B设置为Active的时候,整个Game的Environment的设置就自动切换成场景B的设置了,这时候从FrameDebugger里面可以看到球谐系数变成0了。所以两个模型都黑了。

从Skybox改成Color,起作用的是下图中的3个数值,它们不是0,所以不是黑的。

切换场景A为Active的时候的渲染效果,球谐系数是可以获取到的,所以效果也是正常的,如下图。

所以尝试了一下对场景B进行烘焙,当有了LightingData后,切换到场景B,渲染效果也正常了。变亮了是因为场景B原始设置的Intensity multiplier是5,从FrameDebugger里面看球谐系数不是0了,应该是烘焙后的LihgtingData里面保留了球谐系数。

感谢Xuan@UWA问答社区提供了回答

Q3:NGUI Label自定义材质球无效

想在NGUI下做一个字体溶解Shader,自定义的Shader材质球给Label不起作用,有没有大佬了解这块内容?

A:猜测题主是要在编辑器里面的材质球对象上调整_Threshold的数值,但在Game窗口发现文本没有发生变化。

本质原因是NGUI在对Label进行渲染的时候使用的并不是编辑器里面赋值的材质球,而是在NGUI进行合并DrawCall后动态创建的Material,所以我们需要对这个材质球进行材质球属性设置。

这里可以通过脚本来给实际渲染Label的材质球调整属性达到效果。以下分别是Threshold为0和Threshold为0.4的效果。

PS:这样处理的坏处是,和这个Label在同一个DrawCall的Label都会受到影响,所以需要将这些效果的Label的Depth做特殊处理,和其它的Label不放在同一个DrawCall中。

另外在NGUI的UI DrawCall脚本中,可以打开SHOW_HIDDEN_OBJECTS,这样在编辑器里面是可以看到生成具体的DrawCall对象,也就可以看到它们的材质球属性变化。

从下图可以看到具体的DrawCall,它的材质球名字会在前面加[NGUI]的字样,和编辑器里不是同一个材质球。

感谢Xuan@UWA问答社区提供了回答

Q4:Target API level升级到31后Android 12启动黑屏卡死

当前海外版本有硬性要求:Target API level必须升级到31,升级之后在Android 12机型上启动游戏,Unity闪屏之后卡死,其他Android版本正常。

我们使用的Unity版本:2017.4.27f

其他一些简单测试:去掉闪屏, 导出新的空工程,都会出现启动卡死。

其他一些Unity论坛上的方式尝试均失败:

https://forum.unity.com/threads/unity-2017-and-android-12.1271753/

https://forum.unity.com/threads/android-targeting-api-level-31-makes-the-game-freeze-on-android-12.1237576/

之前我们也遇到升级之后无法安装的问题,然后参照其他解决了安装,只是黑屏无法解决。当前最新尝试Unity 2019版本是正常,初步判断是与以下问题同时修复的:

https://issuetracker.unity3d.com/issues/install-parse-failed-manifest-malformed-errors-are-thrown-when-trying-to-run-an-android-application-with-target-api-level-31

不知道是否有其他大神遇到此问题并解决了?如果有升级之后正常,请告知一下Unity版本号,谢谢。

A1:查了一下这个问题,是因为TelephonyManager的listen函数在Android 12过期了,如果没有授权READ_PHONE_STATE权限,此函数会抛出一个SecurityException。

而Unity在启用了自带的音频系统的情况下,恰巧在启动时机会去调用这个方法以实现“在用户接电话时游戏静音”的功能,抛出的异常影响了后续的流程导致卡死。

论坛上有人遇到了类似的问题,但是表现为崩溃:

https://forum.unity.com/threads/android-12-telephony-crash.1287986/

项目能升级引擎的话,可以试试这里提到的修复的版本:

https://issuetracker.unity3d.com/issues/android-player-crashing-in-fmod-when-targetting-sdk-level-31

如果项目不能升级引擎,也有一个解决办法:

1. 反编译classes.jar

2. 修改UnityPlayer类的addPhoneCallListener实现,判断if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) return;据说Android 12开始不需要自己处理静音了

3. 再编回classes.jar

另外可以试试以下这个反混淆工具,自己重命名类名变量名之后再反编译。

https://github.com/FabricMC/Enigma

感谢littlesome@UWA问答社区提供了回答

A2:不升级引擎Unity2017,使用JByteMod修改classes.jar的addPhoneCallListener接口,就可以正常使用。

做法就是把addPhoneCallListener所有Code都删掉,我是把

这段代码的OpCode插入

感谢yang@UWA问答社区提供了回答

A3:我用2018 4.30f1编了一个classes.jar的包,音频使用Wwise,视频用的Avpro。方法就是按照楼上@yang 提供的方案判断SDK版本小于31则return。

实测替换就可以解决问题 分享给还在纠结的人:

链接:https://pan.baidu.com/s/1Issrh8QRx1A-VppIT8ZDOQ

提取码:1111

感谢liwei@UWA问答社区提供了回答

A4:用2018.4.25版本按@yang提供的方案也可以解决:

链接:https://pan.baidu.com/s/14r_4GPHONQXvc23I66qm-w

提取码:otr0

感谢lyd@UWA问答社区提供了回答

A5:Unity 2017.4.20f2按照上面用JByteMod修改jar的办法解决了启动黑屏的问题,摸索着插入成功了,对某些插入不熟悉,我是jar反编译了去看别的地方类似的写法,然后在JByteMod中找到对应的右键编辑查看插入属性。大体按照顺序一个个写上去,label 0插入完之后再插入if_icmplt 0。

感谢RdUGz39WR5Cx@UWA问答社区提供了回答

Q5:计算大文件MD5耗时问题

在计算大文件MD5的时候,存在耗时严重问题,大概2分钟,在手机上接受不了,有大佬有方法吗?

测试发现:改Buffer大小到1MB,由2200毫秒变成了1980毫秒,优化效果并不明显。

https://itecnote.com/tecnote/c-the-fastest-way-to-create-a-checksum-for-large-files-in-c/

A:可以尝试使用xxHash算法,对比过性能数据,比MD5算法快很多。

https://github.com/uranium62/xxHash

https://github.com/Cyan4973/xxHash

感谢马三小伙儿@UWA问答社区提供了回答

Q6:资源打包关系依赖树

想做包体资源分析,大家有什么好的树显示工具或者思路推荐吗?有比较好的开源方案也可以。最简单就像N叉树一样,比如root一个文件名,然后展开整个树结构。

A1:我自己做了一个,供参考。都是用Unity自己的IMGUI最基本的接口去实现。

EditorWindows

GUI.Box

GUI.BeginGroup

GUI.Label

Handles.DrawBezier

Handles.DrawWireDisc

TreeView

基本上,组织好各个AssetBundle的依赖关系其实是很好呈现的。

感谢黄程@UWA问答社区提供了回答

A2:推荐一款比较好用的插件,不止有依赖树,还有其他打包的资源数据可供分析:

https://assetstore.unity.com/packages/tools/utilities/build-report-tool-8162

感谢郑骁@UWA问答社区提供了回答

Q7:关于函数参数使用Lambda表达式的疑问

关于函数参数使用Lambda表达式的疑问:

写法一:_socket.BeginSend(data, offset, len, SocketFlags.None, out _socketError, new AsyncCallback(OnSendData), _socket);

写法二:_socket.BeginSend(data, offset, len, SocketFlags.None, out _socketError, OnSendData, _socket);

请问写法二本质同第一种是一样的?编译器会帮忙new一个AsyncCallback?或者OnSendData指向的是函数的地址,没有new的开销?

A:我构筑了两个类似的方法(省略了前后实现)以验证两种写法是否有差别。

编译后,使用dnSpy工具查看dll文件,发现IL代码中都会有new的开销,即两种写法本质上是完全一致的。

感谢Faust@UWA问答社区提供了回答

Q8:预制物嵌套导致AssetBundleName修改后对母预制物丢失引用

Unity 2020.3.16预制物嵌套时,子预制物引用的图片AssetBundleName修改后,母预制物会丢失引用。

举例来说,预制物A中有个预制物B,然后预制物B上的RawImage引用图片C。ABC三个打到不同AssetBundle中。

首次打包,加载全部AssetBundle,实例化A,A显示正常。

修改图片C包名,再次打包,A所在包不会有变动。但是加载全部AssetBundle,实例化A,A会丢失C的引用。

反编译AssetBundle会发现,实际预制物A所在资源包数据中有图片C的引用数据,但是因为二次打包A包无变化,就没有更新C所在包的数据。

这个问题升级Unity是否可以解决?或者在当前版本是否可以避开?

反编译AssetBundle会发现A所在Bundle会直接以External References形式关联到图片C的地址,并且AssetBundle也会依赖到图片C所在Bundle(但是不依赖到嵌套Prefab B所在Bundle)。

Prefab B重新关联图片D再打包,A引用会正常刷新。但是仅仅修改图片C的BundleName再打包,不会触发A重新打包。

AssetBundle的Manifest显示的A和B依赖关系是不正确的,显示还是A依赖B,B依赖C,和实际解包出来的不一样。

现在已经用追踪Prefab嵌套树,外加资源BundleName监视的流程暂时解决了打包问题。但是还是希望能获得更规范的解决方案。

A1:Unity Prefab嵌套目前只处理了Editor部分,打包AssetBundle时,会将Subprefab的序列化文件部分copy一份到Rootprefab,其实就等于AssetBundle环境下,嵌套Prefab不生效。

感谢郑骁@UWA问答社区提供了回答

A2:Unity在增量打包的时候,会计算AssetFileHash,在这里计算的时候,只考虑了嵌套的Prefab,但是实际打AssetBundle时,使用的是嵌套Prefab的引用项,我们当前做法是修改Unity的源码,在计算AssetFileHash时就将嵌套Prefab展开。

感谢顾中一@UWA问答社区提供了回答

Q9:multi_compile的Keyword是不是需要主动加入到SVC里面去

multi_compile的Keyword是不是需要主动加入到SVC里面去?

A:对于一个Shader资源来说,在项目进行打包构建时,multi_compile定义的关键字会把Shader中含有该关键字但实际未使用的变体也进行构建,而shader_feature定义的关键字则不会。

但当我们项目中使用SVC收集变体时,并不是所有multi_compile定义的变体都需要主动加入到SVC中,只有我们实际用到的需要收集。

进行实验如下:

实验构建场景,通过SVC收集变体、打成AssetBundle包。在场景中提前加载并Warmup,再实例化一个用到相关Shader中变体“FOG_EXP2”的预制体。(变体“FOG_EXP2”是multi_compile关键字定义的。)

情况一:SVC中没有包含变体“FOG_EXP2”。此时会在实例化时触发Shader.CreateGPUProgram(相当于回到该SVC所引用的Shader中去加载了),不满足我们收集变体并预热、从而降低游戏过程中Shader加载耗时的需求。

情况二:SVC中收集了变体“FOG_EXP2”。实例化时没有触发Shader.CreateGPUProgram,说明该变体被正常Warmup了。

结论是,对于包体构建是没有区别的,SVC打包时会依赖对应的Shader,multi_compile定义的关键字自然都会参与构建;对于变体预热,只要是需要用到的变体,必须收集到SVC中并Warmup后,才不会在实例化渲染时触发Shader.CreateGPUProgram。

感谢Faust@UWA问答社区提供了回答

Q10:Xcode工程中,如何通过Object-C代码反调Unity侧的C#代码

在iOS平台下,IL2CPP导出的Xcode工程中Object-C调用Unity方法是通过SendMessage实现的:

请问在Mac平台下IL2CPP方式导出的Mac工程,如何通过Object-C代码反调Unity侧的C#代码?也是通过SendMessage的方式吗?但是我没找到相关的接口。

A:用SendMessage是可以实现的,但是效率不好。可以参考我这个Object-C回调Unity。把你需要的接口,写成函数指针,在Object-C里注册,需要时做回调。

可以参考《Unity与Object-C交互》

感谢廖武兴@UWA问答社区提供了回答

今天的分享就到这里。当然,生有涯而知无涯。在漫漫的开发周期中,您看到的这些问题也许都只是冰山一角,我们早已在UWA问答网站上准备了更多的技术话题等你一起来探索和分享。欢迎热爱进步的你加入,也许你的方法恰能解别人的燃眉之急;而他山之“石”,也能攻你之“玉”。

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

推荐阅读更多精彩内容