【真机Bug】异步加载资源未完成访问单例导致资源创建失败

1.错误表现描述

抽卡时,10抽展示界面为A。抽取内容可能是整卡或者碎片,抽到整卡,会有立绘展示和点击详情的按钮。
点击详情后出现详情页B。【此时界面A预制体被销毁,卡片数据进入数据缓存池】
点击页面B的返回按钮,单例的HuanLingRewardController读取DataPool的内容加载界面。
Bug表现:10抽的抽卡展示界面没有出现。
【P.S. 测试发现,问题在游戏引擎平台上没有任何问题,在真机中稳定复现
image.png

2.错误排查阶段

因为界面B没有显示,那么就可能有下面几种情况

  1. 界面B的最大一级预制体没有创建/或者被隐藏
  2. Grid下的每个Item可能没有被创建成功/或者被隐藏

通过对指定卡片的GameObject进行addWatches 监视 和 对于SetActive位置进行断点。
首先排除了,物品未被激活导致不显示的问题。

然后从返回按钮的点击事件回调,追踪了一下上下文结构。在机器适配过程中发现,代码中并没有专门为webgl编写相关的宏分支。那么初步可以判断问题不是出现在机型适配导致的。

接下来,我在CreateItem方法和HuanLingRewardController脚本的Awake和OnEnable阶段断点。
查看堆栈。追踪一下上下文。
image.png
在绘制界面的必经方法中打印了日志(真机调试,用日志输出)
然后。
对比正常创建10抽界面(第一次十抽后显示)和 详情页返回10抽界面(Bug不显示)的日志
image.png
上图为正常加载界面,可以看到碎片都成功加载了。
image.png
上图为不显示界面的情况。发现HuanLingRewardController.awake 阶段在showDrawResult后执行的。
看看awake做了什么事

 private void Awake(){Debug.LogWarning("HuanLingRewardController.Awake()执行了");m_Instance = this;if (!CheckUI()) return;m_LiHuiView.PlayFinished = OnPlayFinished;m_LiHuiView.UpdateActiveTips = OnUpdateActiveTips;m_AutoActiveTipsTweeners = m_AutoActiveTips.GetComponents<UITweener>();m_AutoActiveTips.gameObject.SetActive(false);}

他对Instance实例初始化了。
因为实际上awake在后面执行,所以此时m_Instace == null ,然后在if (!m_Instance) return; 返回了

 public static void ShowDrawResult(GC_SPIRITS_LOTTERY packet, bool playTweenAnimation = true){Debug.LogWarning("ShowDrawResult执行了");if (!m_Instance) return;m_Instance.InnerShowDrawResult(packet, playTweenAnimation);}

3.错误分析

那么为什么作为单例的Controller的awake阶段会在他的静态方法执行后才初始化。
之前的写法是

 public static void ShowLastReward(){Debug.LogWarning("ShowLastReward()执行了");if (m_LastSpiritsLotteryPacket == null) return;ShowUI(SubPage.Reward,true);HuanLingRewardController.ShowDrawResult(m_LastSpiritsLotteryPacket);}

如果ShowUI是同步加载资源的话,是没有问题的但实际上showUI的基类会调用

private static bool DoShowUI(bool bSync, UIPathData pathData, OnOpenUIDelegate delOpenUI = null, object param = null)

 AssetManager.LoadUI(pathData.path, m_instance.LoadUIBundleFinish, pathData);
//主要是调用了LoadUI
pathData.onOpenUI = delOpenUI;
pathData.param = param;
AssetManager.LoadUI(pathData.path, m_instance.LoadUIBundleFinish, pathData);

而LoadAsset是一个异步加载的原型

public static void LoadAsset(BundleType type, string name, Action<IAssetRef, object> callback, object param){if (string.IsNullOrEmpty(name))return;
#if USE_ABBundleTask task = new BundleTask(OnLoadRemoteAssetFinished);task.AddParam(param);task.AddParam(callback);task.Add(type, name);LoadBundle(task, AssetLoader.LoadQueueType.KEYRES);
#elseif (RemoteBundleManager.IsRemoteBundle(type, name)){BundleTask task = new BundleTask(OnLoadRemoteAssetFinished);task.AddParam(param);task.AddParam(callback);task.Add(type, name);LoadBundle(task, AssetLoader.LoadQueueType.KEYRES);}else if (Zeus.Framework.Asset.LocalAssetStatus.Ready == Zeus.Framework.Asset.AssetManager.GetAssetStatus(GetAssetPath(type, name), GetAssetType(type))){if (callback != null)callback(Zeus.Framework.Asset.AssetManager.LoadAsset(GetAssetPath(type, name), GetAssetType(type)), param);}else{Zeus.Framework.Asset.AssetManager.LoadAssetAsync(GetAssetPath(type, name), GetAssetType(type), callback, param);}
#endif}

说明代码逻辑是先生成的窗口预制体然后,去缓存池中读上一次抽卡保存的结果,然后加载数据刷新面板。
所以现在是异步加载未完成然后就调用了后续的静态方法导致被迫中止。

4.问题解决

所以

HuanLingRewardController.ShowDrawResult(m_LastSpiritsLotteryPacket);

刷数据的逻辑应该放在ShowUI 执行成功的回调函数里。
于是,给方法新增标记,判断是否是第二次加载

HuanLingRewardController.ShowDrawResult(m_LastSpiritsLotteryPacket, false);

ShowUI.cs中

public static void ShowUI(SubPage defaultPage,bool isSecondEnter = false){// 针对绘卷UI进行特殊处理if (StoryScrollMainView.GetInstance()){StoryScrollMainView.GetInstance().NeedShowMenu = false;}if (StoryScrollSubView.GetInstance()){StoryScrollSubView.GetInstance().NeedShowMenu = false;}UIManager.ShowUI(UIInfo.HuanLingRoot, (isSuccess, _) =>{if (isSuccess && Instance){Instance.OnShow(defaultPage);}if (isSecondEnter){HuanLingRewardController.ShowDrawResult(m_LastSpiritsLotteryPacket, false);}});}

在UIManager.ShowUI 执行成功的回调中加入上述代码即可。
这样就保证了在界面预制体加载完成后,才会走刷数据的流程。
出现这样的问题也是因为,手机端异步加载资源的速度受到网络延迟,服务器结点的影响很大。用同步加载的逻辑去思考异步功能肯定是不行的。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++——模版

前言&#xff1a;哈喽小伙伴们好久不见&#xff0c;这是2024年的第一篇博文&#xff0c;我们将继续C的学习&#xff0c;今天这篇文章&#xff0c;我们来习一下——模版。 目录 一.什么是模版 二.模版分类 1.函数模版 2.类模板 总结 一.什么是模版 说起模版&#xff0c;我们…

线索二叉树

线索二叉树即从前、中、后序三种遍历中其中一种来看&#xff0c;树中的左右孩子都不会是空着的&#xff0c;都会指向对应的前驱和后驱。 以中序遍历为例&#xff0c;二叉树线索化过程如下&#xff1a; 先是树的结构 typedef struct ThreadNode{Elemetype data;struct ThreadNo…

微服务面试题之套路一

面试题 一、你的项目是从SpringBoot演进到微服务架构的&#xff0c;你在此过程中有调研过哪些技术&#xff0c;怎么调研落地的? 微服务通信框架&#xff1a; 需要选择适合项目的微服务通信框架&#xff0c;如Dubbo、Spring Cloud或gRPC Feign RestTemplate 等。调研方式可以是…

高性能通信之Netty

一, 同步IO(BIO)模型的架构 一般针对性能不高的情况下可以使用. 二,异步IO(NIO)模型的架构 多路复用(epoll模型):

【LeetCode:124. 二叉树中的最大路径和 + 二叉树+递归】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

前端开发人员如何做好SEO

前端开发人员如何做好SEO SEO工作不仅限于专业人员。前端开发者也可以在日常开发中实施一些代码层面的SEO优化。 以下是一些前端常用的SEO方法&#xff1a; 设置合理的title、keywords、description title、keywords、description对SEO至关重要&#xff0c;需贴合页面内容编…

Codeforces Round 931 (Div. 2) (A~B)

比赛&#xff1a;Codeforces Round 931 (Div. 2) (A~B) 目录&#xff1a;A B A题&#xff1a;Too Min Too Max 标签: 构造算法&#xff08;constructive algorithms&#xff09;贪心&#xff08;greedy&#xff09;数学&#xff08;math&#xff09; 题目大意 对数组 a 找到…

【力扣hot100】刷题笔记Day19

前言 回溯回溯回溯&#xff01;早上整理档案竟然用了桶排序&#xff0c;不愧是算法狂魔们 79. 单词搜索 - 力扣&#xff08;LeetCode&#xff09; DFS class Solution:def exist(self, board: List[List[str]], word: str) -> bool:m, n len(board), len(board[0])# used…

mysql timestamp转换为datetime

MySQL timestamp转换为datetime的方法 1. 流程概述 在MySQL中&#xff0c;timestamp和datetime是两种不同的数据类型。timestamp存储了日期和时间&#xff0c;并且会自动更新&#xff0c;可以用于记录数据的创建和修改时间。datetime则是一个固定的日期和时间&#xff0c;不会自…

谈谈高并发系统的设计方法论

谈谈高并发系统的设计方法论 何为高并发系统&#xff1f;什么是并发&#xff08;Conurrent&#xff09;&#xff1f;什么是高并发&#xff08;Hight Concurrnet&#xff09;&#xff1f;高并发的衡量指标有哪些&#xff1f; 实现高并发系统的两大板块高并发系统应用程序侧的设计…

腾讯云学生服务器使用教程_申请腾讯云学生机详细流程

2024年腾讯云学生服务器优惠活动「云校园」&#xff0c;学生服务器优惠价格&#xff1a;轻量应用服务器2核2G学生价30元3个月、58元6个月、112元一年&#xff0c;轻量应用服务器4核8G配置191.1元3个月、352.8元6个月、646.8元一年&#xff0c;CVM云服务器2核4G配置842.4元一年&…

还在用Jenkins?快来试试这款简而轻的自动部署软件!

最近发现了一个比 Jenkins 使用更简单的项目构建和部署工具&#xff0c;完全可以满足个人以及一些小企业的需求&#xff0c;分享一下。 Jpom 是一款 Java 开发的简单轻量的低侵入式在线构建、自动部署、日常运维、项目监控软件。 日常开发中&#xff0c;Jpom 可以解决下面这些…

Nginx的多线程支持探究

文章中心思想: Nginx本身并不直接支持多线程处理模型。它采用的是基于事件驱动的单线程或多进程架构,而非多线程模型。然而,通过Nginx的模块和第三方扩展,可以实现类似多线程的并发处理效果。 详细说明: Nginx,作为一款高性能的Web服务器和反向代理服务器,其架构和并发…

章节二、three.js开发入门与调试设置02;

一、轨道控制器查看物体&#xff1b; 1、基本概念 轨道控制器&#xff08;OrbitControls&#xff09;可以使得相机围绕目标进行轨道运动&#xff1b; 2、代码样例 // 七、创建轨道控制器&#xff08;相机围绕着物体捕捉视角&#xff09; const controls new OrbitControls(c…

吴恩达机器学习全课程笔记第五篇

目录 前言 P80-P85 添加数据 迁移学习 机器学习项目的完整周期 公平、偏见与伦理 P86-P95 倾斜数据集的误差指标 决策树模型 测量纯度 选择拆分方式增益 使用分类特征的一种独热编码 连续的有价值特征 回归树 前言 这是吴恩达机器学习笔记的第五篇&#xff0c…

《2023跨境电商投诉大数据报告》发布|亚马逊 天猫国际 考拉海购 敦煌网 阿里巴巴

2023年&#xff0c;跨境电商API接口天猫国际、京东国际和抖音全球购以其强大的品牌影响力和市场占有率&#xff0c;稳坐行业前三的位置。同时&#xff0c;各大跨境电商平台消费纠纷问题层出不穷。依据国内知名网络消费纠纷调解平台“电诉宝”&#xff08;315.100EC.CN&#xff…

javaEE--后端环境变量配置

目录 pre 文件准备 最终运行成功结果 后端运行步骤 1.修改setenv文件 2.运行setenv&#xff0c;设置环境变量 3.查看jdk版本 4.修改mysql文件夹下的my文件 前端运行步骤 1.nodejs环境配置 2.查看node和npm版本 3.下载并运行npm 4.注册登录 pre 文件准备 最终运行…

VR转接器:破解虚拟与现实边界的革命性设备

VR转接器&#xff0c;这一革命性的设备&#xff0c;为虚拟现实体验带来了前所未有的自由度。它巧妙地连接了虚拟与现实&#xff0c;使得用户在享受VR眼镜带来的奇幻世界的同时&#xff0c;也能自由地在现实世界中活动。这一设计的诞生&#xff0c;不仅解决了VR眼镜续航的瓶颈问…

2、云原生安全之可视化界面rancher的部署

文章目录 1、rancher的部署1.1、安装rancher1.2、配置k8s2、部署helm3、容器安全工具neuvector此时已经部署好了k8s,使用rancher来管理 rancher简化了使用k8s的流程,可以图形化管理k8s。 参考: https://blog.51cto.com/u_15343792/5000311https://docs.rancher.cn/docs/ra…

你们团队是否有RocketMQ创建Topic、GID创建规范呢

这里是weihubeats,觉得文章不错可以关注公众号小奏技术 背景 早期在使用RocketMQ的时候&#xff0c;系统和开发人员不算多。所以topic的创建会非常随意&#xff0c;各种千奇百怪的topic 比如: order_topic、ORDER_TOPIC、order-topic 各种奇奇怪怪的风格&#xff0c;用_的&a…