`
dazhilao
  • 浏览: 246022 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

C# Struct的内存布局

    博客分类:
  • C#
阅读更多
问题:请说出以下struct的实例大小以及内存布局
    struct Struct1
    {
        public byte a;
        public short b;
        public string c;
        public int d;
    }

    struct Struct2
    {
        public byte a;
        public long b;
        public byte c;
        public string d;
    }

    struct Struct3
    {
        byte a;
        byte b;
        long c;
    }

    struct Struct4
    {
        byte a;
        long b;
        byte c;
    }


一会再看答案,看看和你的理解是不是有很大的出入?其实struct和class的内存布局都是由StructLayoutAttribute的构造参数:LayoutKind枚举决定的,struct由编译器添加LayoutKind.Sequential,class由编译器添加的是LayoutKind.Auto。而Sequential通过实验数据可以总结如下:

1. 对于不带引用类型的struct:按照定义的顺序排列,内存布局和c,c++规则相同。比如:

Byte a;

Byte b;

Long c;

的大小是 a,b填充4字节,c填充8字节

Byte a

Long c

Byte b

的大小是 a填充8字节,c填充8字节,b填充8字节

2. 对于带有引用类型的struct:大于4字节的字段 -> 引用字段 ->  小于4字节的字段

对于小于4字节的字段按照大小排列,如果大小相同按照定义顺序,内存布局和规则1相同。不过这里有个需要注意的地方就是如果字段还是一个struct类型的,那么这个字段始终排在最后。

所以上面的答案是:

Struct1:c(4) -> d(4) -> b(2) ->a(2)

Struct2:b(8) -> d(4) -> a(1)c(1)填充2字节

Struct3: a(1)b(1)填充2字节 -> c(8)

Struct4:a(1)填充7字节->b(8)->c(1)填充7字节


如果你想亲自动手实验一下的话需要使用SOS.dll进行调试(关于SOS配置和使用入门的文章博客园上有很多)以struct1为例:

Struct1s1 = new Struct1();

s1.a = 1;          

            s1.b = 15;

            s1.c = "c";

            s1.d = 32;


.load sos

已加载扩展C:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\sos.dll

!clrstack -a

PDB symbol for mscorwks.dll not loaded

OS Thread Id: 0x15fc (5628)

ESP       EIP   

0041ee3c 03ba01aa Test_Console.Class12.Main()

    LOCALS:

        0x0041ee84 = 0x01b02b0c

        0x0041ee74 = 0x00000020

        0x0041ee68 = 0x00000000

        0x0041ee50 = 0x00000000


0041f104 6ebd1b4c [GCFrame: 0041f104]


.load sos

已加载扩展C:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\sos.dll

!name2ee *!Test_Console.Struct1 //得到Struct1的方法表地址

PDB symbol for mscorwks.dll not loaded

Module: 6d5d1000 (mscorlib.dll)

--------------------------------------

Module: 00192c5c (Test_Console.exe)

Token: 0x02000012

MethodTable: 00193828

EEClass: 007a45b4

Name: Test_Console.Struct1


!clrstack -a //得到struct1实例的栈上地址

OS Thread Id: 0x1438 (5176)

ESP       EIP   

003eef0c 008f00c9 Test_Console.Class12.Main()

    LOCALS:

        0x003eef1c = 0x01c12b0c


003ef17c 6ebd1b4c [GCFrame: 003ef17c]


!dumpvc 00193828 0x003eef1c //查看值类型的layout

Name: Test_Console.Struct1

MethodTable 00193828

EEClass: 007a45b4

Size: 20(0x14) bytes

Fields:

      MT    Field   Offset                 Type VT     Attr    Value Name

6d84340c  400001c        a          System.Byte  1 instance        1 a

6d83e910  400001d        8         System.Int16  1 instance       15 b

6d8408ec  400001e        0        System.String  0 instance 01c12b0c c

6d842b38  400001f        4         System.Int32  1 instance       32 d


在内存窗口中可以看到内存布局为:

0x003EEF1C  01c12b0c 00000020 0001000f

这里我要说明下使用dumpvc后会给出一个size,这里是20字节,比我们计算的结果多出8个字节,我的理解是因为引用类型有附加的8字节(syncblkindex + methodtableaddress)所以这里的size也加上了8.
分享到:
评论

相关推荐

    C# Struct的内存布局问题解答

    本篇文章将详细探讨C# Struct的内存布局问题。 首先,C#编译器会为Struct添加`LayoutKind.Sequential`属性,而对于Class,则添加`LayoutKind.Auto`。Sequential布局意味着结构体成员会按照声明的顺序依次存储,而...

    C#实现共享内存通讯交互

    结构体的布局必须与内存布局一致,所以通常使用`LayoutKind.Sequential`。 2. **共享内存的创建**: 使用`CreateFileMapping`函数创建一个文件映射对象,然后指定一个空的文件句柄(如`IntPtr.Zero`)以及一些...

    C#中结构体struct和字节数组Byte的之间的转换函数

    在C#编程语言中,当涉及到TCP通信时,由于TCP通信本质上是基于字节流的,因此在处理自定义的数据结构如结构体(struct)时,通常需要将这些结构体转换为字节数组(byte[])进行传输,之后再在接收端还原成原来的...

    C#结构体和C++字节数组的转换

    这里的关键是使用`System.Runtime.InteropServices`命名空间,并通过`[StructLayout]`特性来指定结构体的布局和编码方式。例如: ```csharp using System.Runtime.InteropServices; [StructLayout(LayoutKind....

    C#结构体指针的定义及使用详解

    - `MarshalAs`属性用于指定结构体成员的内存布局和类型转换规则,这对于与非托管代码交互时非常重要。 - 结构体成员的顺序决定了它们在内存中的排列顺序。 #### C#结构体指针的定义 在C#中,指针不是默认支持的...

    C++结构体/函数定义转换C#函数定义/结构体

    - C++的`struct`在C#中也是`struct`,但C#的`struct`是值类型,不支持继承,需要注意内存管理。 2. **结构体转换**: - C++的结构体成员排列可能与C#不同,需要考虑内存对齐问题。 - 在C#中,可能需要使用`...

    C#向C++封送结构体数组[归类].pdf

    这里使用`LayoutKind.Sequential`布局,确保C#结构体与C++结构体的内存布局一致。`MarshalAs`属性用于指定如何封送字段,例如`UnmanagedType.ByValTStr`用于字符串,`UnmanagedType.ByValArray`用于数组。 然而,...

    C#调用C++封装成DLL的结构体及类库.rar

    2. **结构体(Struct)**:C#和C++都支持结构体,但它们在内存布局上有所不同。在C#中,结构体是值类型,而在C++中,结构体可以像类一样使用。当C#调用C++ DLL时,需要确保结构体的内存布局与C++相同,包括对齐方式...

    c# 调用c++对于字符串与复杂结构体的处理

    复杂结构体的处理更为复杂,因为它们涉及到内存布局和生命周期管理。C++/CLI可以创建托管的C++类,这些类可以直接被C#引用。例如,假设有一个C++结构体: ```cpp struct Person { char* name; int age; }; ``` ...

    C#调用C++编写的DLL函数各种参数传递

    在C#中,首先需要定义一个类似的结构体,并使用`[StructLayout(LayoutKind.Sequential)]`属性来确保内存布局与C++中一致: ```csharp public struct NET_INFO_STRUCT { public uint nDurationTime; public double...

    c#结构体和byte之间的转换

    需要注意的是,由于结构体的内存布局是由编译器决定的,因此在不同平台或编译器上可能有所不同。为了确保跨平台兼容性,可以使用`LayoutKind.Explicit`指定每个字段的偏移量。同时,数组的处理需要特别注意,因为...

    网络通信中C、C++结构体转C#结构体.rar

    当我们需要在C或C++系统与C#系统之间进行通信时,可能会遇到一个问题:不同编程语言中的数据类型和内存布局可能存在差异。本文将深入探讨如何在网络通信中将C/C++的结构体转换为C#的结构体。 首先,让我们理解C和...

    C#调用C++ Dll关于结构体数组引用的传递及解析使用的展示代码

    这里的`MarshalAs`属性特别重要,因为它确保了字符数组在内存中的正确布局。 #### C#中调用DLL C#中调用C++ DLL的关键在于正确声明DLL导入函数。以下是具体的实现: ```csharp using System; using System....

    C#, byte operation

    通过手动控制内存布局,可以提高性能,减少垃圾回收的压力。例如,使用结构体(struct)而不是类(class)可以减少对象分配,提高缓存效率。 6. 数据序列化:在C#中,`BinaryFormatter`或自定义序列化器可以将对象...

    C#与C++数据类型对照

    - C#的结构体(struct)可以映射到C++的结构体(struct),但C#的结构体在内存布局上与C++可能存在差异,需要特别注意对齐规则。 4. 引用类型与指针: - C#是垃圾回收的语言,它没有裸指针,而是使用对象引用。当...

    C#中调用C++中动态链接库DLL中的结构体

    C++允许直接操作内存,而C#则通过垃圾回收机制自动管理内存,这在处理结构体时尤其重要,因为C++结构体的布局可能与C#类的布局不同。因此,当C#与C++交互时,需要确保数据类型和内存对齐方式的一致性。 1. **定义互...

    C#调用C++的DLL

    - 对于复杂的数据结构,考虑使用`struct`或`class`作为参数,但要确保大小和布局在两种语言中相同。 - 注意指针和引用的区别,以及它们在C#和C++中的不同处理方式。 通过以上步骤,我们可以成功地在C#中调用C++的...

    C#调用C++生成的DLL,并返回结构体引用或者结构体指针多个值

    C#结构体应该具有相同的字段布局和顺序。例如: ```csharp [StructLayout(LayoutKind.Sequential)] public struct MyStruct { public int Value1; public double Value2; [MarshalAs(UnmanagedType.ByValTStr...

    C#获取硬盘信息(转载的)

    通过本例的学习,我们不仅可以掌握如何使用C#获取硬盘信息的具体方法,还能深入了解C#的数据结构定义、Windows API调用机制、内存布局与数据对齐的重要性,以及在实际开发中如何处理硬件访问权限等问题。这对于从事...

Global site tag (gtag.js) - Google Analytics