Marshal.PtrToStructure throwing AccessViolationException

本文关键字:AccessViolationException throwing PtrToStructure Marshal | 更新日期: 2023-09-27 18:26:00

我得到了这个结构:

    [StructLayout(LayoutKind.Sequential)]
    public struct IS
    {
    public UInt32 ID; 
    public UInt32 Quality; 
    public UInt32 Flags;
    public UInt32 Flags2;     
    public UInt32 ContainerSlots; 
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)]
    public Int32[] ItemStatType;  
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)]
    public UInt32[] ItemStatValue;    
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)]
    public Int32[] ItemStatUnk1;    
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)]
    public Int32[] ItemStatUnk2;       
    public UInt32 ScalingStatDistribution; 
    public UInt32 DamageType;      
    public UInt32 Delay;      
    public float RangedModRange;
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 5)]
    public Int32[] SpellId;          
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 5)]
    public Int32[] SpellTrigger;       
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 5)]
    public Int32[] SpellCharges;  
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 5)]
    public Int32[] SpellCooldown;   
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 5)]
    public Int32[] SpellCategory;     
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 5)]
    public Int32[] SpellCategoryCooldown;
    public UInt32 Bonding; 
    public string Name;       
    public string Name2;                  
    public string Name3;         
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 3)]
    public UInt32[] Color;  
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 3)]
    public UInt32[] Content;
};

我正在尝试从文件中读取字节,并使用Marshal和GCHandle将这些字节复制到上面的struct,我的代码如下:

reader = BinaryReader.FromFile(fileName);
m_rows = new List<IS>();
int size = Marshal.SizeOf(typeof(IS));
if(reader.BaseStream.Length < size)
  return;
byte[] buffer = new byte[size];
buffer = reader.ReadBytes(size);
GCHandle handle = GCHandle.Alloc(buffer, GCHandleType.Pinned);
m_rows.Add((IS)Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(IS)));
handle.Free();

但我得到了AccessViolationException : attempt to read or write protected memory

我不知道为什么会抛出这个异常。

Marshal.PtrToStructure throwing AccessViolationException

我没有立即看到这个错误,并编写了一个小测试程序来重新解决这个问题。使用二进制搜索来找到问题,重复注释一半的字段,直到我将其缩小到:

[StructLayout(LayoutKind.Sequential)]
public struct IS {
    public string Name;
}

这是不可行的,pinvokemarshaller假定字符串的默认封送处理来自C字符串char*。这不可能纠正从文件中读取的数据,因为它永远不能包含有效的指针。当试图取消引用指针时,会触发AccessViolation。

这个问题中没有任何提示可以猜测字符串实际上是如何序列化到文件的。正常方式为:

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
public struct IS {
    [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 42)]
    public string Name;
};

如有必要,请使用十六进制查看器来计算SizeConst的正确值。如果编码不正常(不是系统默认页面),则必须将其声明为byte[],并使用正确的编码进行转换。

您可以理解,访问违规是由于试图读取未分配的内存或正在释放的内存,您可能需要检查以下堆栈溢出日志:

Marshal.PtrToStructure触发时发生AccessViolationException

这是指托管结构和本机结构的大小之间的差异作为问题的原因,本质上,您需要在编组期间提供偏移量,以匹配结构的托管分配和本机分配之间的差异。

也请检查此过帐,其中用户已添加偏移

在循环中使用方法Marshal.PtrToStructure时发生访问冲突异常。

解决方案的另一个链接:

http://www.codeproject.com/Questions/585390/AccessplusViolationplusException

如果这没有帮助,那么使用windbg调试这些问题很容易,我可以列出详细信息,以备不时之需。此外,在VS中启用Win32 Aces违规异常,它会在换行异常处中断,并提供一些附加信息