记一次线上报错 GList AddChildAt NullReferenceException

文章目录

问题描述

后台日志大量报错,去主干看无法复现

c# exception:System.NullReferenceException: Object reference not set to an instance of an object. at FairyGUI.GCompone

nt.AddChildAt (FairyGUI.GObject child, System.Int32 index) [0x00000] in <00000000000000000000000000000000>:0 at Fairy

GUI.GList.AddChildAt (FairyGUI.GObject child, System.Int32 index) [0x00000] in <00000000000000000000000000000000>:0 a

t FairyGUI.GComponent.AddChild (FairyGUI.GObject child) [0x00000] in <00000000000000000000000000000000>:0 at FairyGUI

.GList.set_numItems...

分析

看逻辑是没有问题的,但是在调用numItems时c#层报错。看FairyGUI的代码,

C# 复制代码
public int numItems
{
    get
    {
        if (_virtual)
            return _numItems;
        else
            return _children.Count;
    }
    set
    {
        if (_virtual)
        {
            if (itemRenderer == null)
                throw new Exception("FairyGUI: Set itemRenderer first!");

            _numItems = value;
            if (_loop)
                _realNumItems = _numItems * 6;//设置6倍数量,用于循环滚动
            else
                _realNumItems = _numItems;

            //_virtualItems的设计是只增不减的
            int oldCount = _virtualItems.Count;
            if (_realNumItems > oldCount)
            {
                for (int i = oldCount; i < _realNumItems; i++)
                {
                    ItemInfo ii = new ItemInfo();
                    ii.size = _itemSize;

                    _virtualItems.Add(ii);
                }
            }
            else
            {
                for (int i = _realNumItems; i < oldCount; i++)
                    _virtualItems[i].selected = false;
            }

            if (_virtualListChanged != 0)
                Timers.inst.Remove(this.RefreshVirtualList);
            //立即刷新
            this.RefreshVirtualList(null);
        }
        else
        {
            int cnt = _children.Count;
            if (value > cnt)
            {
                for (int i = cnt; i < value; i++)
                {
                    if (itemProvider == null)
                        AddItemFromPool();
                    else
                        AddItemFromPool(itemProvider(i));
                }
            }
            else
            {
                RemoveChildrenToPool(value, cnt);
            }

            if (itemRenderer != null)
            {
                for (int i = 0; i < value; i++)
                    itemRenderer(i, GetChildAt(i));
            }
        }
    }
}

AddChild 发生在 AddItemFromPool 中

C# 复制代码
/// <summary>
/// Add a item to list, same as GetFromPool+AddChild
/// </summary>
/// <returns>Item object</returns>
public GObject AddItemFromPool()
{
    GObject obj = GetFromPool(null);

    return AddChild(obj);
}

/// <summary>
/// Add a item to list, same as GetFromPool+AddChild
/// </summary>
/// <param name="url">Item resource url</param>
/// <returns>Item object</returns>
public GObject AddItemFromPool(string url)
{
    GObject obj = GetFromPool(url);

    return AddChild(obj);
}

AddChild 的参数为空引用, 看GetFromPool如何拿到

c# 复制代码
public GObject GetFromPool(string url)
{
    if (string.IsNullOrEmpty(url))
        url = _defaultItem;

    GObject ret = _pool.GetObject(url);
    if (ret != null)
        ret.visible = true;
    return ret;
}

最终定位到GObjectPool.GetObject,对象是从对象池池子中取出的,没取到则创建一个

c# 复制代码
        public GObject GetObject(string url)
        {
            url = UIPackage.NormalizeURL(url);
            if (url == null)
                return null;

            Queue<GObject> arr;
            if (_pool.TryGetValue(url, out arr)
                && arr.Count > 0)
                return arr.Dequeue();

            GObject obj = UIPackage.CreateObjectFromURL(url);
            if (obj != null)
            {
                if (initCallback != null)
                    initCallback(obj);
            }

            return obj;
        }

从上面的代码可以得出报错的原因

  1. 可能池子里拿出来的对象在lua层已经被销毁了
  2. 可能创建对象没有成功,资源存在问题

在lua代码报错的地方加入以下定位代码

lua 复制代码
    local list = self._listview
    local obj, nullNum, lossNum
    for i = list.numChildren + 1, infoLen do -- 先处理已经销毁了的
        obj, nullNum, lossNum = self:getItmeFromPool(list)
        if obj then
            list:AddChild(obj)
        else
            if SDKCtrl then
                SDKCtrl:reportError(" warning 上报定位  get null tiems:" .. nullNum .. "  get Disposed times: " .. lossNum)
            end
        end
    end

    list.numItems = infoLen
    list:ResizeToFit(infoLen)
lua 复制代码
function mod:getItmeFromPool(list)
    local obj
    local lossNum, nullNum = 0, 0
    while lossNum < 100 and nullNum < 100 do
        obj = list:GetFromPool()

        if not obj then
            nullNum = nullNum + 1
        elseif obj.isDisposed then
            lossNum = lossNum + 1
        else
            break
        end
    end

    return obj, nullNum, lossNum
end

结果

定位的代码更新出去后,日志中出现了 warning 上报定位 get null tiems:100 get Disposed times: 0。

item对象没有创建成功,是分支的列表item资源存在问题,分支找到对应的包重新发布下,问题解决了。应该是主干同步分支的时候没有发布该包。

总结

GList添加item时候,会从对象池中取一个已存在对象复用或者创建新的对象,如果没有找到对应的资源或者对象池中的对象已经销毁过,会导致创建失败,报错 AddChildAt NullReferenceException。

相关推荐
人才程序员11 小时前
【C++拓展】vs2022使用SQlite3
c语言·开发语言·数据库·c++·qt·ui·sqlite
Xing201716 小时前
unity打包ios Xcode问题记录
unity·游戏引擎
龙哥·三年风水18 小时前
openresty(nginx)+lua+kafka实现日志搜集系统
kafka·lua·openresty
MasterNeverDown18 小时前
WPF 使用iconfont
hadoop·ui·wpf
不惑_1 天前
深度学习 · 手撕 DeepLearning4J ,用Java实现手写数字识别 (附UI效果展示)
java·深度学习·ui
墨笺染尘缘1 天前
Unity——鼠标是否在某个圆形Image范围内
unity·c#·游戏引擎
Хайде1 天前
Qt Desiogn生成的ui文件转化为h文件
ui
Thomas_YXQ1 天前
Unity3D项目开发中的资源加密详解
游戏·3d·unity·unity3d·游戏开发
资深设备全生命周期管理2 天前
以Python 做服务器,N Robot 做客户端,小小UI,拿捏
服务器·python·ui
杀死一只知更鸟debug2 天前
Unity自学之旅05
unity·游戏引擎