gpt4 book ai didi

c# - 引用字符串 [] 内存泄漏与 NET/COM 互操作

转载 作者:行者123 更新时间:2023-11-30 12:51:56 32 4
gpt4 key购买 nike

我最近发现一个非常奇怪的(对我来说)内存泄漏 IEnumString从 C# 使用的 COM 对象。具体来说,调用 IEnumString.Next 方法时使用的字符串数组已包含上次调用产生的值,这会导致内存泄漏。

IEnumString 在 C# 端看起来像这样:

    [InterfaceType(1)]
[Guid("00000101-0000-0000-C000-000000000046")]
public interface IEnumString
{
void Clone(out IEnumString ppenum);
void RemoteNext(int celt, string[] rgelt, out int pceltFetched);
void Reset();
void Skip(int celt);
}

像这样调用RemoteNext(Next)方法导致泄漏,通过长时间反复运行,看到“Private Bytes”计数器不断上升,验证了这一点。

string[] item = new string[100]; // OBS! Will be re-used for each call!
for (; ; )
{
int fetched;
enumString.RemoteNext(item.Length, item, out fetched);
if (fetched > 0)
{
for (int i = 0; i < fetched; ++i)
{
// do something with item[i]
}
}
else
{
break;
}
}

但是为每次调用创建一个新的字符串 item[] 数组以某种方式使泄漏消失了。

for (; ; )
{
int fetched;
string[] item = new string[100]; // Create a new instance for each call.
enumString.RemoteNext(item.Length, item, out fetched);
if (fetched > 0)
{
for (int i = 0; i < fetched; ++i)
{
// do something with item[i]
}
}
else
{
break;
}
}

第一种情况出了什么问题?我想发生的事情是为 IEnumString.Next 的 rgelt 参数分配的 COM 内存,它应该由调用者释放,但不知何故却没有。

但第二种情况奇怪地有效。

编辑:对于一些额外的信息,这是 RemoteNext 方法在 ILDASM 和 .NET Reflector 中的“实现”。

.method public hidebysig newslot abstract virtual 
instance void RemoteNext([in] int32 celt,
[in][out] string[] marshal( lpwstr[ + 0]) rgelt,
[out] int32& pceltFetched) runtime managed internalcall
{
} // end of method IEnumString::RemoteNext



[MethodImpl(MethodImplOptions.InternalCall, MethodCodeType=MethodCodeType.Runtime)]
void RemoteNext(
[In] int celt,
[In, Out, MarshalAs(UnmanagedType.LPArray, ArraySubType=UnmanagedType.LPWStr, SizeParamIndex=0)] string[] rgelt,
out int pceltFetched);

编辑 2:您还可以通过在调用 RemoteNext 之前简单地将字符串值添加到字符串数组(仅包含空值),使泄漏出现在第二种非泄漏情况中。

string[] item = new string[100]; // Create a new instance for each call.
item[0] = "some string value"; // THIS WILL CAUSE A LEAK
enumString.RemoteNext(item.Length, item, out fetched);

因此,对于编码层而言,项目数组似乎必须为空才能正确释放复制到它的非托管字符串。即便如此,该数组将返回相同的值,即非空数组不会导致返回错误的字符串值,它只会泄漏一些值。

最佳答案

IEnumString - 它只是一个接口(interface)。底层的 COM 对象是什么?这是主要嫌疑人。

查看IEnumString 的非托管声明:

HRESULT Next(
[in] ULONG celt,
[out] LPOLESTR *rgelt,
[out] ULONG *pceltFetched
);

如您所见,第二个参数 rgelt 只是一个指向字符串数组的指针——没什么特别的,但是当您进行托管调用时

string[] item = new string[100]; // Create a new instance for each call.
item[0] = "some string value"; // THIS WILL CAUSE A LEAK
enumString.RemoteNext(item.Length, item, out fetched);

item[0] 中的字符串似乎已转换为 LPOLESTR,但未正确释放。因此试试这个:

string[] item = new string[1];
for (; ; )
{
int fetched;
item[0] = null;
enumString.RemoteNext(1, item, out fetched);
if (fetched == 1)
{
// do something with item[0]
}
else
{
break;
}
}

关于c# - 引用字符串 [] 内存泄漏与 NET/COM 互操作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5662122/

32 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com