国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C# > 正文

C# Struct的內存布局問題解答

2020-01-24 03:05:00
字體:
來源:轉載
供稿:網友

問題:請說出以下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.

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 阳新县| 鹰潭市| 永城市| 抚顺市| 恩平市| 吉安市| 城固县| 安溪县| 原平市| 宝应县| 济源市| 泰州市| 沙雅县| 大埔县| 南城县| 沁阳市| 江都市| 平南县| 兴和县| 隆子县| 南召县| 莱阳市| 山西省| 阳原县| 大城县| 河南省| 宜丰县| 临猗县| 嘉义县| 华阴市| 洪雅县| 昆山市| 白水县| 乌兰浩特市| 静乐县| 辽阳县| 黔南| 塔城市| 浪卡子县| 英吉沙县| 富平县|