if 组件是否存在_UE4 UMG简介+Slate组件问题排查

Slate 组件问题排查总结简介
首先是一个工作中遇到的BUG: 用slua添加子节点到父节点上的时候,第二次打开无法显示对应的子节点Widget。对应Lua代码如下

  1. local comboBox = ui_manager.ShowUI(ui_manager.UI_Config.ui_coupon_combobox,2,price,buyUIInfo.shopInfo.id)
  2. if comboBox then
  3. self:AttachChildWindow("ScaleBox_Coupon",comboBox)
  4. end


因为我们引入了UI对象的内存池概念,很自然而然的想到了缓存住的控件是不是释放有问题。检查slua的release代码之后发现没有任何问题。基于ScaleBox 只能有1个子物体这一个机制,在猜测BUG原因的时候,尝试用CanvasPanel替换ScaleBox组件。发现果然能解决这个问题。
但是也带来了2个新问题:
1、ScaleBox子物体在第一次关闭和第二次打开的时候经历了什么?
2、为什么CanvasPanel就可以而ScaleBox不可以?
为了解决以上疑问,开启我们的排查过程。
为了能使大家更好的了解这个流程,先对Slate组件创建销毁流程进行介绍。Slate组件创建销毁流程
1、将我们日常使用的一个组件,ScaleBox拆出来看(如下图):有如下的对应关系。里面包含了 UScaleBox,UScaleBoxSlot,SscaleBox

ae36ab3a246d680c42276be61523630b.png


2、创建过程

346a1e3ee589193772a961f8b153a775.png


3、销毁过程

cc8bcf0c1bd19dd8185f3578576bb45e.png

问题排查
针对ScaleBox,进入断点调试。
在第一次关闭界面的时候,发现UscaleBox Release 有正常跑到(下面代码部分)。

  1. void UScaleBox::ReleaseSlateResources(bool bReleaseChildren)
  2. {
  3. Super::ReleaseSlateResources(bReleaseChildren);
  4. MyScaleBox.Reset();
  5. }

但是在第二次打开的时候,发现没有进入RebuildWidget(Swidget的实例化部分,也就是下面代码部分)

  1. TSharedRef<SWidget> UScaleBox::RebuildWidget()
  2. {
  3. MyScaleBox = SNew(SScaleBox)
  4. .SingleLayoutPass(bSingleLayoutPass);
  5. if ( GetChildrenCount() > 0 )
  6. {
  7. CastChecked<UScaleBoxSlot>(GetContentSlot())->BuildSlot(MyScaleBox.ToSharedRef());
  8. }
  9. return MyScaleBox.ToSharedRef();
  10. }


这是为什么呢? 原来,在UWidget的BUILD过程中,有一个查找共享指针MyWidget的操作。这个指针虽然被UscaleBox Reset了一次。但是在ScaleBox 第二次打开时候,这个指针依然存在引用。所以该实例并没有被销毁。那么在第二次打开的时候,由于没有走RebuildWidget。导致了子类UScaleBox 的指针引用已经被销毁,在AddChild时候调用添加了个空Object。(下图代码部分)

  1. TSharedRef<SWidget> UWidget::TakeWidget_Private(ConstructMethodType ConstructMethod)
  2. {
  3. bool bNewlyCreated = false;
  4. TSharedPtr<SWidget> PublicWidget;
  5. // If the underlying widget doesn't exist we need to construct and cache the widget for the first run.
  6. if (!MyWidget.IsValid())
  7. {
  8. PublicWidget = RebuildWidget();
  9. #if !(UE_BUILD_SHIPPING || UE_BUILD_TEST)
  10. ensureMsgf(PublicWidget.Get() != &SNullWidget::NullWidget.Get(), TEXT("Don't return SNullWidget from RebuildWidget, because we mutate the state of the return. Return a SSpacer if you need to return a no-op widget."));
  11. #endif
  12. MyWidget = PublicWidget;
  13. bNewlyCreated = true;
  14. }
  15. else
  16. {
  17. PublicWidget = MyWidget.Pin();
  18. }


那这个指针是被谁Hold住了呢? 经过排查与调试,发现是被UScaleBoxSlot给Hold住了导致了这份资源没有被释放。

67e2662d69f58f339b076215e4d206e5.png

8ff9f848bb5b50804fbc6aa7abae17a9.png


为什么在UScaleBox中的共享指针已经释放了,但是UScaleBoxSlot所持有的相同指针没有被释放呢?请看Slot释放的地方(下面代码模块):

  1. // If the child is a UserWidget, we should let it manage it's own slate resources instead of forcing a clear here. This fixes issues such as UE-39106
  2. if (PanelSlot->Content && !PanelSlot->Content->IsA<UUserWidget>())
  3. {
  4. const bool bReleaseChildren = true;
  5. PanelSlot->ReleaseSlateResources(bReleaseChildren);
  6. }


UE的注释写的很清楚了,我们所Add 的子物体,正是UserWidget 界面。在UE-39106版本中,让UserWidget 的Slate释放控制交给了上层。在这里没有自动释放。下面是UE 39106的改动链接。https://issues.unrealengine.com/issue/UE-39106

fb4791e6faab9bd2b22599d1e490f7f0.png

虚幻引擎UE为了补锅一个slate 被释放的问题,导致了这个新问题。 我们上层缓存了这份实例,销毁子物体的时候调用了RemoveFromParent。在ScaleBox被销毁的时候,因为其子物体为userwidget,而不去释放PanelSlot的SlateResourse。那么在第二次打开的时候,因为PanelSlot中共享指针的存在,没有New一份新的子物体实例。
OK,问题定位了。那么之前提到的第二个问题,为什么换成CanvasPanel就没有这个问题了呢?,既然查到了在ScaleBox 中是 ScaleBoxSlot 的指针Hold住了这份资源,那么我们去CanvasPanel对应的Slot 看一下。(下面代码)

  1. void UCanvasPanelSlot::BuildSlot(TSharedRef<SConstraintCanvas> Canvas)
  2. {
  3. Slot = &Canvas->AddSlot()
  4. [
  5. Content == nullptr ? SNullWidget::NullWidget : Content->TakeWidget()
  6. ];
  7. SynchronizeProperties();
  8. }


明显可以看出:原来CanvasPanelSlot 在Build过程中没有缓存那份实例!只是做了常规引用!在上面有贴出过ScaleBoxSlot build 过程的代码。所以就是ScaleBoxSlot 存住了这份指针,而CanvasPanelSlot 并没有。所以在都没有Release对应掉这份资源的情况下,CanvasPanel会走rebuild过程,而ScaleBox不会!
到这里,我们基本上就定位出了问题全部原因所在。解决办法
这里想到了4种解决方案,都能解决这个问题,这里简单说一下:
1、不判断指针是否还存在引用,全部走Rebuild过程。
2、在Remove判断是否为UserWidget过程中,对本身PanelSlot做一次释放。
3、删除UScaleBoxSlot中对SscaleBox指针的引用。
4、将UScaleBoxSlot中的共享引用改为弱引用。
第一种风险较大,因为所有UMG都会走的创建过程,不太保险。
第二种风险小于第一种,但是也在所有UMG都会走的创建过程,不太保险。
第三种和第四种个人感觉都是有效的方案,总的来看还是第四种改动小并优雅一点。
所以目前采用了第四种。

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

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

相关文章

创业型公司的产品经理应该知道的事情

可能因为最近有一本书叫做<<人人都是产品经理>>&#xff0c;所以产品经理这个词变得很fashion了&#xff0c;产品经理在一些成熟型的公司&#xff0c;确实是一个很重要的职位。因为成熟型的公司产品的开发的流程都已经很完善&#xff0c;所以产品经理是其中一个不可…

设置框开始隐藏状态html5,小猿圈分享HTML5中form如何关闭自动完成功能的方法

WEB前端现在是时下较火的编程语言之一&#xff0c;但是对于怎么学习或者学习哪些内容很多朋友都是不了解的&#xff0c;针对以上内容小猿圈web前端讲师总结了form如何关闭自动完成功能的方法&#xff1f;希望对你的前端学习有一定的帮助。什么是HTML5的form自动完成功能&#x…

浏览器登录_经常用浏览器自动登录忘记了密码?教你一键查看网页星号密码

不知道大家有没有出现这种情况&#xff0c;因为一直用的网页自动填写密码来登录&#xff0c;所以有时候甚至把密码给忘了相信有的小伙伴有可能就会出现这种情况哈&#xff0c;今天小林君来教你个超简单的方法&#xff0c;不用安装任何软件&#xff0c;就可以一键查看网页上隐藏…

控件属性动作

控件应该定义属性而不是公共字段&#xff0c;因为可视化设计器在属性浏览器中显示属性&#xff0c;而不显示字段。属性就像智能字段。属性通常具有带访问函数的专用数据成员&#xff0c;在语法上属性被作为类的字段进行访问。&#xff08;虽然属性可以具有不同的访问级别&#…

python连接数据库并编写调用函数_Python使用pyodbc访问数据库操作方法详解

本文实例讲述了Python使用pyodbc访问数据库操作方法。分享给大家供大家参考&#xff0c;具体如下&#xff1a;1、连接数据库1)直接连接数据库和创建一个游标(cursor)cnxn pyodbc.connect(DRIVER{SQL Server};SERVERlocalhost;DATABASEtestdb;UIDme;PWDpass)cursor cnxn.cursor(…

Mysql 5.5的编译安装 在ubuntu 10平台上面

(一)安装mysql5.5的要求 编译安装mysql5.5与5.1还是有一点不同&#xff0c;因为mysql现在用cmake来作编译工程工具。 这是与mysql5.1最大不同的地方&#xff0c;因此在安装mysql5.5时所要求的工具包也不同。发现用cmake编译mysql比以前的速度还是快了不少。 &#xff08;1&a…

html图片轮播怎么做的,CSS3制作轮播图的一种方法

轮播图&#xff0c;网页上经常能看得见&#xff0c;画面比较精美&#xff0c;下面是纯CSS3的轮播图的一种下面是style部分&#xff1a;这几行都能明白吧*{margin:0;padding:0;}a{text-decoration:none}li{list-style:none;}设计宽度不要超过轮播图片的总宽度再加不到第一张1张图…

怎么让textarea占满整个td高度没用_家里没发现虫子,怎么才能确定是被什么害虫咬了?...

在家里被虫子咬&#xff0c;还发痒&#xff0c;主要有三种可能性。蚊子、跳蚤、臭虫都有可能&#xff0c;那么如何才能确定是被哪种害虫叮咬&#xff1f;接下来带大家了解三种害虫叮咬的区别&#xff01;首先是蚊子&#xff0c;早上起来发现身上有包&#xff0c;先看有几处&…

PLSQL 笔记

oracle的左连接或右连接 以下是解释&#xff0c;自己研究下&#xff1a; ------------------------------------------------------------------- 数据表的连接有: 1、内连接(自然连接): 只有两个表相匹配的行才能在结果集中出现 2、外连接: 包括 &#xff08;1&#xff09;左外…

上位机与1200组态步骤_组态王与 I/O 设备

组态王软件是一种通用的工业监控软件&#xff0c;它将过程控制设计、现场操作以及工厂资源管理融于一体&#xff0c;将一个企业内部的各种生产系统和应用以及信息交流汇集在一起&#xff0c;实现最优化管理。它基于Microsoft Windows XP/Win7/Win8/Win10/WinServer 系列操作系统…

MySQL安装错误: unknown option '--skip-federated'

mysql启动时出现以下错误&#xff1a; [ERROR] /usr/local/mysql/libexec/mysqld: unknown option --skip-federated [ERROR] Aborting [Note] /usr/local/mysql/libexec/mysqld: Shutdown complete 只要将/etc/my.cnf文件中的skip-federated注释掉即可 通过查看/var/log/mysql…

华为鸿蒙发布2.0,华为做到了!鸿蒙2.0正式发布,苹果安卓有危机?

孩子的梦想总是天真烂漫&#xff0c;无论年龄大小&#xff0c;每个人都有属于自己的梦想&#xff0c;儿童节刚过华为天真的梦就照进了现实。6月2日晚间&#xff0c;鸿蒙2.0版本发布会如期而至&#xff0c;余承东正式向全世界宣布这款挑战安卓和 iOS的商用移动操作系统&#xff…

【IE大叔的嘴歪眼斜】之—— 由hasLayout引发的临床CSS Bug表

IE大叔这嘴歪眼斜的毛病不是一天两天了&#xff0c;集体拉出来测试时候&#xff0c;明明大家都在微笑&#xff0c;就丫一副呲牙咧嘴的...... 哎&#xff0c;没办法&#xff0c;谁让咱国内市面上都是这种呲牙咧嘴的浏览器呢.....(关注IE678死亡速度) 查阅&#xff0c;翻读&#…

greenplum 查询出来的数字加减日期_Python实践代码总结第5集(日期相关处理)

英文的月份转数字及数字转英文import calendar# 数字转月份的简写calendar.month_abbr[12]--> Dec# 简写月份转数字list(calendar.month_abbr).index(Dec)--> 12# 数字转月份的全写calendar.month_name[12]--> December# 月份转数字list(calendar.month_name).index(D…

静态代码和动态代码的区别_无代码和低代码有哪些区别

代码是大多数软件程序和应用程序的骨干。代码是大多数软件程序和应用程序的骨干。每行代码充当一条指令&#xff1a;采用一种逐步性的逻辑机制&#xff0c;以便计算机、服务器和其他机器执行操作。想创建那些指令&#xff0c;就要知道如何编写代码&#xff0c;这项宝贵的技能有…

备注html网页代码,备注.html · dengzhao/prd_zhangyao - Gitee.com

&#xfeff;备注$axure.utils.getTransparentGifPath function() { return resources/images/transparent.gif; };$axure.utils.getOtherPath function() { return resources/Other.html; };$axure.utils.getReloadPath function() { return resources/reload.html; };备注…

mysqld与mysqld_safe的区别

直接运行mysqld程序来启动MySQL服务的方法很少见&#xff0c;mysqld_safe脚本会在启动MySQL服务器后继续监控其运行情况&#xff0c;并在其死机时重新启动它。用mysqld_safe脚本来启动MySQL服务器的做法在BSD风格的unix系统上很常见&#xff0c;非BSD风格的UNIX系统中的mysql.s…

python词频统计代码_机器学习必备宝典-《统计学习方法》的python代码实现及课件...

《统计学习方法》可以说是机器学习的入门宝典&#xff0c;许多机器学习培训班、互联网企业的面试、笔试题目&#xff0c;很多都参考这本书。本站根据网上资料用python复现了课程内容&#xff0c;并提供本书的代码实现、课件下载。《统计学习方法》简介《统计学习方法》全面系统…

Oracle中用rownum替代Top函数的方法

今天写一个方法&#xff0c;主要功能是从数据库中根据条件查出第一条信息。以前用sql server的时候&#xff0c;我记得TOP方法还是非常简单实用的。 方法是&#xff1a;select top n [列名] from [表名] where [查询条件] 这个方法想必像我这样的新人也会非常熟悉&#xff0c;所…

view类不响应自定义消息_安卓平台如何给控件添加自定义操作?

在安卓应用设计和开发过程中&#xff0c;设计人员为了界面简洁、有独特的交互方式&#xff0c;可能会为控件设计特殊的操作手势&#xff0c;例如消息列表中单指按住消息向左滑删除消息&#xff1b;系统顶部的通知单指向左滑可以关闭通知等。这些操作对于普通用户非常方便&#…