C#使用StructLayout特性来控制内存结构的操作代码

 更新时间:2024年10月15日 09:05:36   作者:△曉風殘月〆  
在C#中,内存布局对于性能优化和与非托管代码的互操作性至关重要,StructLayout特性允许开发者控制结构在内存中的布局方式,本文给大家介绍了C#使用StructLayout特性来控制内存结构的操作,需要的朋友可以参考下

C#在调用WInAPI函数时,可能会看到如下的声明

[StructLayout(LayoutKind.Sequential)]
         public struct RECT
         {
             public int Left;
             public int Top;
             public int Right;
             public int Bottom;
         }

在类或者结构体前面带上了

[StructLayout(LayoutKind.Sequential)]

StructLayoutAttribute特性的作用是允许你控制内存中类或结构的数据字段的物理布局。

平常我们在C#代码中使用类或者结构体时,不需要使用此特性。但在与非托管代码时交互,需要使用StructLayoutAttribute特性来控制类型的非托管布局。

StructLayoutAttribute常用构造函数是:

StructLayoutAttribute(System.Runtime.InteropServices.LayoutKind)

System.Runtime.InteropServices.LayoutKind是一个枚举类型,有三个取值。

LayoutKind.Sequential

强制按成员的显示顺序对其进行排列。对于blittable类型,在托管和非托管内存中控制布局。对于non-blittable类型,它会在将类或者结构体封送到非托管代码时控制布局(换言之,如果仅在C#中进行调用,不会做任何操作,在与非托管代码交互时,仅控制送入非托管代码的布局)

LayoutKind.Explicit

控制每个数据成员的精确位置,这会影响托管和非托管代码中的布局,不管是blittable类型还是non-blittable类型。,使用LayoutKind.Explicit时,需要使用FieldOffsetAttribute特性指示类型中每个字段的位置。

默认情况下,编译器会将LayoutKind.Sequential应用到结构体,对于类,需要显式应用LayoutKind.Sequential值。

到这里也就明白了,以后在调用API函数时,如果使用的是结构体,就不再需要下面这句代码了。

[StructLayout(LayoutKind.Sequential)]

下面用示例代码说明一下

这里以获取桌面窗体的宽高为例,需要用到的API函数是

HWND GetDesktopWindow();

BOOL GetWindowRect(HWND hWnd, LPRECT lpRect );

其中LPRECT是指向RECT结构体的指针,RECT结构声明如下:

typedef struct tagRECT {
   LONG left;
   LONG top;
   LONG right;
   LONG bottom;
 } RECT, *PRECT, *NPRECT, *LPRECT;

使用结构体时,调用代码如下:

using System;
using System.Runtime.InteropServices;
 namespace ConsoleApp10
 {
     public struct RECT
     {
         public int Left;
         public int Top;
         public int Right;
         public int Bottom;
     }
 
     class Program
     {
         [DllImport("user32.dll")]
         private static extern IntPtr GetDesktopWindow();
 
         [DllImport("user32.dll", SetLastError = true)]
         private static extern int GetWindowRect(IntPtr hwnd, out RECT rc);
 
         static void Main(string[] args)
         {
             IntPtr hwnd = GetDesktopWindow();
             RECT rect;
             GetWindowRect(hwnd, out rect);
             Console.WriteLine($"Left:{rect.Left},Top:{rect.Top},Right:{rect.Right},Bottom:{rect.Bottom}");
         }
     }
 }

运行结果如下:

下面我们来看一下,把结构体换成类的情况

 using System;
 using System.Runtime.InteropServices;
 
 namespace ConsoleApp10
 {
     public class RECT
     {
         public int Left;
         public int Top;
         public int Right;
         public int Bottom;
    }
 
     class Program
     {
         [DllImport("user32.dll")]
         private static extern IntPtr GetDesktopWindow();
 
         [DllImport("user32.dll", SetLastError = true)]
         private static extern int GetWindowRect(IntPtr hwnd,RECT rc);
 
         static void Main(string[] args)
         {
             IntPtr hwnd = GetDesktopWindow();
             RECT rect = new RECT();
             GetWindowRect(hwnd, rect);
             Console.WriteLine($"Left:{rect.Left},Top:{rect.Top},Right:{rect.Right},Bottom:{rect.Bottom}");
         }
     }
 }

运行结果如下:

运行结果并不正常

把类修改一下,带上[StructLayout(LayoutKind.Sequential)]

     [StructLayout(LayoutKind.Sequential)]
     public class RECT
     {
         public int Left;
         public int Top;
         public int Right;
         public int Bottom;
     }

再次运行,发现结果正常了

最后再看看LayoutKind.Explicit的情况,调用代码如下

 using System;
 using System.Runtime.InteropServices;
 
 namespace ConsoleApp10
 {
     [StructLayout(LayoutKind.Explicit)]
     public class RECT
     {
         [FieldOffset(0)]public int Left; //FieldOffset用于指示类或结构的非托管表示形式中字段的物理位置
         [FieldOffset(4)]public int Top;
         [FieldOffset(8)]public int Right;
         [FieldOffset(12)]public int Bottom;
     }
 
     class Program
     {
         [DllImport("user32.dll")]
         private static extern IntPtr GetDesktopWindow();
 
         [DllImport("user32.dll", SetLastError = true)]
         private static extern int GetWindowRect(IntPtr hwnd,[MarshalAs(UnmanagedType.LPStruct)]RECT rc);
 
         static void Main(string[] args)
         {
             IntPtr hwnd = GetDesktopWindow();
             RECT rect = new RECT();
             GetWindowRect(hwnd, rect);
             Console.WriteLine($"Left:{rect.Left},Top:{rect.Top},Right:{rect.Right},Bottom:{rect.Bottom}");
         }
     }
 }

以上就是C#使用StructLayout特性来控制内存结构的操作代码的详细内容,更多关于C# StructLayout控制内存结构的资料请关注脚本之家其它相关文章!

相关文章

  • C#使用NPOI上传excel

    C#使用NPOI上传excel

    这篇文章主要为大家详细介绍了C#使用NPOI上传excel的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-05-05
  • C#自动删除Word文档空白行和空白页的完整代码

    C#自动删除Word文档空白行和空白页的完整代码

    在处理 Word 文档时,经常会遇到空白行、空表格或空白页的问题,这不仅影响排版美观,还可能导致文档页数冗余,所以本文将介绍如何使用 Spire.Doc for .NET 在 C# 中自动删除 Word 文档的空白行、空表格和空白页,需要的朋友可以参考下
    2025-08-08
  • C#中Property和Attribute的区别实例详解

    C#中Property和Attribute的区别实例详解

    这篇文章主要介绍了C#中Property和Attribute的区别,较为详细的分析了C#中Property和Attribute的功能、定义、区别及使用时的相关注意事项,需要的朋友可以参考下
    2015-06-06
  • C# Winform TextBox控件多行输入方式

    C# Winform TextBox控件多行输入方式

    这篇文章主要介绍了C# Winform TextBox控件多行输入方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • C#操作Word模拟解析HTML标记输出带格式的文本

    C#操作Word模拟解析HTML标记输出带格式的文本

    这篇文章介绍了一种解决Word文档格式设置问题的方案,通过HTML标记化文本内容,利用正则表达式提取关键字,并对Word字符集对象逐字操作来重置格式,感兴趣的小伙伴可以了解下
    2026-01-01
  • c#实现flv解析详解示例

    c#实现flv解析详解示例

    本文提供的解析程序简单的把FLV分解了出来,如果要做FLV的修改操作的话,可以给每个类加个toStream方法,再历遍依次调用就可以写回到文件里了
    2014-01-01
  • c#实现哈夫曼树算法

    c#实现哈夫曼树算法

    这篇文章介绍了c#实现哈夫曼树的实例代码,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-06-06
  • C# 利用代理爬虫网页的实现方法

    C# 利用代理爬虫网页的实现方法

    这篇文章主要介绍了C# 利用代理爬网页的实现方法的相关资料,希望通过本能帮助到大家实现这样的功能,需要的朋友可以参考下
    2017-10-10
  • .NET实现父窗体关闭而不影响子窗体的方法

    .NET实现父窗体关闭而不影响子窗体的方法

    这篇文章主要介绍了.NET实现父窗体关闭而不影响子窗体的方法,很实用的功能,需要的朋友可以参考下
    2014-08-08
  • Unity实现苹果手机Taptic震动

    Unity实现苹果手机Taptic震动

    这篇文章主要介绍了Unity实现苹果手机Taptic震动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-10-10

最新评论