c#-LuaInterface内存泄漏问题

  

我发现我在C#/ Lua LuaInterface项目中存在严重的内存泄漏.我用C#编写了一个简单的测试函数,该函数每0.5秒从Lua循环调用一次.我看到Lua的内存使用量在每个循环中都在增加.我最新的C#辅助代码是

  public LuaTable testMemTable()
  {
     LuaTable tabx = m_lua.GetTable("tabx");

     if (tabx != null)
     {
        tabx.Dispose();
        tabx = null;

        GC.Collect();
        GC.WaitForPendingFinalizers();
     }


     m_lua.NewTable("tabx");
     tabx = m_lua.GetTable("tabx");

     for (int i = 0; i < 20000; i++)
        tabx[i] = i * 10;

     return  tabx;
  }

尽管做了tabx.Dispose()和tabx = null然后强行执行GC,我仍然看到内存没有被释放.没有LuaInterface函数可以释放先前分配的表,所以我不知道该怎么做才能释放内存?

Lua边码很简单

while true do

    myAPILibs.testMemTable()

    if tabx ~= nil then
        print(string.format("Size = %d", #tabx))
    else
        print(string.format("Size = nil"))
    end

    print(tabx, tabx[111])

    myAPILibs.sleep(500)

    print("Before ", collectgarbage("count") * 1024)
    collectgarbage("collect")
    print("After ", collectgarbage("count") * 1024)

end

解决我的内存泄漏问题的任何帮助将不胜感激.

再次感谢

杰夫

解决方法:

我需要对LuaInterface稍加抱怨,因为这使这个问题很难解决-在某些情况下,不泄漏内存甚至是不可能的.

LuaInterface(和NLua)中的CLR对象代理没有提供释放Lua引用的终结器.1必须处置每个引用Lua对象的CLR对象.如果您忘记一次,那么Lua对象将永远不会被垃圾回收.

更糟糕的是,您无法正确处理从Lua调用的CLR方法返回的引用.如果您在返回引用之前先处理掉它,那么它就消失了,您将无法再返回它.如果不处理它,则CLR参考会泄漏.您可以执行代码体操来正确处理引用,但是完全没有理由需要您进行此类工作.

现在让我离开我的肥皂盒并解决您的代码.

在所有方法中,都有很多地方会泄漏这些对象.您对GetTable()的使用掩盖了您对其工作方式的误解.每次调用此方法时,都会获得一个新的CLR对象,该对象引用该Lua全局变量引用的Lua表.您对它的用法似乎假设您可以执行m_lua.GetTable(“ tabx”).Dispose()并完成某件事-所有这些操作是创建一个新的CLR引用,然后仅处置该引用.换句话说,这是什么都不做的昂贵方法.

每次调用GetTable()都应进行相应的处理(或使用C#的using块为您进行处理).

要澄清的是,放置CLR LuaTable对象不会破坏Lua表!它所做的只是释放对表的特定CLR参考.

对于引用Lua对象(包括函数)的每个LuaInterface类型,这都是适用的,这是您可能泄漏的另一个地方.

在这里,我将展示每种方法的问题,以及如何将其重写以解决这些问题.

public LuaTable testMemTable()
{
   // This code does nothing.  You get a new reference to the Lua table
   // object, and then you immediately dispose it.  You can remove this
   // whole chunk of code; it's a really expensive no-op.
   LuaTable tabx = m_lua.GetTable("tabx");

   if (tabx != null)
   {
      tabx.Dispose();
      tabx = null;

      GC.Collect();
      GC.WaitForPendingFinalizers();
   }


   m_lua.NewTable("tabx");

   // You create a new CLR object referencing the new Lua table, but you
   // don't dispose this CLR object.
   tabx = m_lua.GetTable("tabx");

   for (int i = 0; i < 20000; i++)
      tabx[i] = i * 10;

   return  tabx;
}

编写此方法的正确方法是:

public void testMemTable()
{
    m_lua.NewTable("tabx");

    using (LuaTable tabx = m_lua.GetTable("tabx")) {
        for (int i = 0; i < 20000; i++) {
            tabx[i] = i * 10;
        }
    }
}

(请注意,由于您从未使用过返回值,因此我将返回类型更改为void.)

public LuaTable getNewTableCSharp()
{
    // You don't dispose the function object.
    var x = lua.GetFunction("getNewTableLua");
    // You don't dispose the table object.
    var retValTab = (LuaTable)x.Call()[0];

    return retValTab;
}

请注意,由于从不重新分配getNewTableLua函数,因此实际上并没有泄漏Lua函数,但是每次调用此函数时,都会泄漏Lua表中包含该函数引用的插槽.

现在麻烦了:因为此函数是从Lua调用并返回对Lua对象的引用,所以您不能修复两个泄漏,只能修复函数泄漏:

public LuaTable getNewTableCSharp()
{
    using (var x = lua.GetFunction("getNewTableLua")) {
        // Still leaks, but you can't do anything about it.
        return (LuaTable)x.Call()[0];
    }
}

回到我的肥皂盒,考虑改用Eluant,它是CLR的一组Lua绑定(非常类似于LuaInterface),只是它解决了内存管??理问题. (免责声明:我是Eluant的作者.)

特别是,它解决了您在这里遇到的问题:

>引用Lua对象的Eluant的CLR对象具有终结器,这些终结器将对Lua引用进行排队,以便稍后发布.如果您忘记放置引用Lua对象的CLR对象,则无论如何最终都会收集它. (但是您仍然应该尽快配置引用,最好使用C#的using块,以确保Lua的GC可以及时收集对象.)
>如果在Lua调用的方法中返回对Lua对象的CLR对象引用,则Eluant将在为Lua返回控制权之前为您处理该引用.

1See here.存在终结器,但如果终结器调用了Lua对象引用,则处理程序不会释放它.换句话说,finalizer根本不执行任何操作.请注意,由于Lua并非线程安全的,因此此时实际释放Lua引用并不安全,但是释放操作可能会在以后排队. LuaInterface不会执行此操作; Eluant does.

相关文章