可以使用UnmanagedMemory.LPTStr而不是.ByValTStr导致内存损坏吗?为什么?

时间:2013-03-01 19:30:59

标签: c# .net memory pinvoke dllimport

我们在Windows窗体应用程序中有一个树视图,使用以下代码使用相应的文件图标显示文件。我的问题是调用GetIcon()似乎会破坏我的内存,因为我开始得到各种程序崩溃,这是我在调用之后无法用调试器捕获的。 将managedType.LPTStr更改为managedType.ByValTStr时,该计划有效。这是一个真正的解决方案还是只是掩盖了问题?

此代码似乎在我们的上一个产品版本中工作,我看不到任何已更改的内容。使用.NET 4.0。我只在发布模式下看到问题。

[DllImport("Shell32.dll")]
private static extern int  SHGetFileInfo(string pszPath, uint dwFileAttributes, out SHFILEINFO psfi, uint cbfileInfo, SHGFI uFlags);

 [StructLayout(LayoutKind.Sequential)]
        private struct SHFILEINFO
        {
            public SHFILEINFO(bool b)
            {
                hIcon=IntPtr.Zero;
                iIcon=0;
                dwAttributes=0;
                                szDisplayName = "";
                                szTypeName = "";
            }

            public IntPtr hIcon;
            public int iIcon;
            public uint dwAttributes;

                        [MarshalAs(UnmanagedType.LPTStr, SizeConst = 260)]//works if .ByValTStr is used instead
                        public string szDisplayName;
                        [MarshalAs(UnmanagedType.LPTStr, SizeConst = 80)]//works if .ByValTStr is used instead
                        public string szTypeName;
        };

 public static Icon GetIcon(string strPath, bool bSmall)
        {
            SHFILEINFO info = new SHFILEINFO(true);
            int cbFileInfo = Marshal.SizeOf(info);
            SHGFI flags;
            if (bSmall)
                flags = SHGFI.Icon|SHGFI.SmallIcon|SHGFI.UseFileAttributes;
            else
                flags = SHGFI.Icon|SHGFI.LargeIcon|SHGFI.UseFileAttributes;

            SHGetFileInfo(strPath, 256, out info,(uint)cbFileInfo, flags);
            return Icon.FromHandle(info.hIcon);
        }

2 个答案:

答案 0 :(得分:0)

嗯,这不是结构中的正确LPStr,所以你不能试图把它作为一个整理并期望它起作用:

typedef struct _SHFILEINFO {
  HICON hIcon;
  int   iIcon;
  DWORD dwAttributes;
  TCHAR szDisplayName[MAX_PATH];
  TCHAR szTypeName[80];
} SHFILEINFO;
  • LPTStr当你分配了一个特殊的内存块来保存这个字符串时(通常是Marshal.AllocHGlobal或类似的),你使用string,然后你复制了ByValTStr 1}}到那个非托管的内存区域。

  • {{1}}当您按字面意思传递实际字符串时,而不是通过引用内存中的其他区域。

struct需要正确的值,而不是指针。

答案 1 :(得分:0)

我意识到这是一个老问题,但这有助于我解决一个似乎突然开始突然爆发的崩溃。在通过Windows Update推出.NET 4.5.2更新后,我似乎开始遇到这些问题。 LPTStr在更新之前工作,ByValTStr之后工作。