当前位置: 首页 > article >正文

.net Span类型和Memory类型

.NET 中 Span 类型和 Memory 类型的深度剖析

在 .NET 编程的世界里,高效处理内存是提升程序性能的关键。Span<T>Memory<T> 类型的出现,为开发者提供了强大而灵活的工具,用于高效地访问和操作连续内存区域。今天,我们就来深入探讨这两种类型及其相关概念。

核心类型介绍

1. Span<T>

Span<T> 代表一块连续的、不可变长度的内存区域,可直接读写其中的元素。它可以在栈上声明,也能指向堆上分配的数据或其他内存位置。这种设计使得在不复制数据的情况下,能高效处理内存区域,尤其适用于处理大型数据结构、高性能计算以及与操作系统交互的场景。

2. ReadOnlySpan<T>

ReadOnlySpan<T>Span<T> 的只读版本,提供对连续内存区域的只读访问。当不需要修改内存内容时,使用它能确保数据的安全性,同时享受与 Span<T> 类似的性能优势。

3. Memory<T>ReadOnlyMemory<T>

System.Memory<T>System.ReadOnlyMemory<T> 类似于 Span<T>,但它们增加了表示更复杂内存所有权的能力。可以引用来自非托管内存、网络流、管道或内存映射文件等来源的数据。此外,它们支持跨线程和异步操作,可通过 IMemoryOwner<T> 接口表示所有权。

4. MemoryManager<T>

用于创建和管理 Memory<T>ReadOnlyMemory<T> 实例。开发者可通过实现 MemoryManager<T> 自定义内存的分配和释放方式。

5. IMemoryOwner<T>

作为 MemoryManager<T> 的一个更简单的实现,表示临时拥有一段可释放内存的实体。可通过 MemoryPool<T> 获取此类实例,当内存不再需要时,可通过 Dispose() 方法释放资源。

6. MemoryPool<T>

内存池类,用于高效地分配和回收小块内存,避免频繁的内存分配和垃圾回收带来的开销。可以从内存池中租借 Memory<T> 实例,完成后归还给池。

7. Memory<T>.Pin() 方法和 GCHandle

当需要将 Span<T>Memory<T> 转换为不受垃圾回收影响、可暴露给非托管代码的指针时,可使用 .Pin() 方法获取 MemoryHandle,或直接使用 GCHandle 类来固定内存,常用于与非托管代码(如 C/C++ 互操作)的内存共享。

代码示例

Span<T>ReadOnlySpan<T> 示例

// 创建一个整数数组
int[] array = { 1, 2, 3, 4, 5 };// 从数组创建 Span<T>
Span<int> span = new Span<int>(array);// 访问 Span 中的元素
Console.WriteLine(span[0]); // 输出: 1
span[0] = 10; // 修改 Span 中的元素会影响原始数组// 使用 Slice 方法获取 Span 的子集
Span<int> subSpan = span.Slice(1, 3); // 从索引 1 开始,长度为 3 的子集: { 10, 2, 3 }
foreach (var item in subSpan)
{Console.WriteLine(item);
}// 创建 ReadOnlySpan<T>
ReadOnlySpan<int> readOnlySpan = span.AsReadOnly();
Console.WriteLine(readOnlySpan[0]); // 输出: 10
// readOnlySpan[0] = 20; // 这会编译错误,因为 ReadOnlySpan<T> 是只读的

从这个示例可以看出,Span<T> 就像是对数组内存的一个“视图”,修改 Span<T> 中的元素会直接影响原始数组。而 ReadOnlySpan<T> 则提供了只读访问,保证了数据的安全性。

Memory<T>ReadOnlyMemory<T> 示例

// 创建一个整数数组
int[] array = { 1, 2, 3, 4, 5 };// 从数组创建 Memory<T>
Memory<int> memory = MemoryMarshal.CreateFromArray(array);// 访问 Memory 中的元素(通过 Span 属性)
Span<int> spanFromMemory = memory.Span;
Console.WriteLine(spanFromMemory[0]); // 输出: 1
spanFromMemory[0] = 100; // 修改 Span 中的元素会影响原始数组和 Memory// 创建 ReadOnlyMemory<T>
ReadOnlyMemory<int> readOnlyMemory = memory;
ReadOnlySpan<int> readOnlySpan = readOnlyMemory.Span;
Console.WriteLine(readOnlySpan[0]); // 输出: 100
// readOnlySpan[0] = 200; // 这会编译错误,因为 ReadOnlySpan<T> 是只读的// 使用 Memory 的 Slice 方法
Memory<int> subMemory = memory.Slice(1, 3); // { 100, 2, 3 }
Span<int> subSpan = subMemory.Span;
foreach (var item in subSpan)
{Console.WriteLine(item);
}

Memory<T> 同样可以方便地访问和操作内存,但它更适合处理复杂的内存场景,尤其是涉及异步和跨线程操作。

Span<T>.Clear 方法示例

int[] array = { 1, 2, 3, 4, 5 };
Span<int> span = new Span<int>(array);
span.Clear();
Console.WriteLine(string.Join(", ", array)); // 输出 0, 0, 0, 0, 0

Span<T>.Clear 方法能快速将 Span<T> 中的所有元素设置为默认值,这在需要清空内存数据时非常实用。

MemoryMarshal 类示例

byte[] byteArray = { 1, 2, 3 };
Span<int> intSpan = MemoryMarshal.Cast<byte, int>(byteArray.AsSpan()).Slice(0, byteArray.Length / sizeof(int));
Console.WriteLine(intSpan[0]); // 输出 67305985,这是 1, 2, 3 的字节表示转换为 int 的结果

MemoryMarshal 类提供了一组静态方法,用于在 Span<T>Memory<T> 和其他类型之间进行转换和操作,大大增强了内存操作的灵活性。

Span<T> 与字符串示例

string str = "Hello";
ReadOnlySpan<char> charSpan = str.AsSpan();
Console.WriteLine(charSpan[0]); // 输出 Hchar[] charArray = new char[5];
Span<char> writableCharSpan = charArray.AsSpan();
writableCharSpan.Fill('A');
string newStr = new string(writableCharSpan);
Console.WriteLine(newStr); // 输出 AAAAA

通过 MemoryMarshalText.Encoding 类,可以在 Span<char> 和字符串之间进行转换,方便处理字符串相关的内存操作。

异步编程和跨线程操作

Memory<T>ReadOnlyMemory<T> 非常适合用于异步编程和跨线程操作,因为它们是引用类型,可以安全地跨越方法边界和线程边界。而 Span<T> 是值类型,生命周期和范围受到限制,不适合直接用于这些场景。

以下是一个使用 Memory<T>ReadOnlyMemory<T> 进行异步编程和跨线程操作的示例代码:

class Program
{static async Task Main(){// 创建一个整数数组int[] array = { 1, 2, 3, 4, 5 };// 从数组创建 Memory<T>Memory<int> memory = array.AsMemory();// 异步方法,接受 ReadOnlyMemory<T> 作为参数await ProcessMemoryAsync(memory);// 跨线程操作Task threadTask = Task.Run(() => ProcessMemoryOnThread(memory));await threadTask;}static async Task ProcessMemoryAsync(ReadOnlyMemory<int> readOnlyMemory){// 使用 ReadOnlySpan 来处理内存区域ReadOnlySpan<int> span = readOnlyMemory.Span;foreach (var item in span){// 模拟异步操作,例如 I/O 操作await Task.Delay(100);Console.WriteLine(item);}}static void ProcessMemoryOnThread(ReadOnlyMemory<int> readOnlyMemory){// 使用 ReadOnlySpan 来处理内存区域ReadOnlySpan<int> span = readOnlyMemory.Span;// 跨线程操作,这里只是简单地遍历并打印值foreach (var item in span){Console.WriteLine(item);}}
}

在这个示例中,Memory<T>ReadOnlyMemory<T> 确保了在异步和跨线程操作中能够安全地共享对连续内存区域的引用,而无需复制数据,提高了性能。

总结

Span<T>Memory<T> 类型为 .NET 开发者提供了强大的内存处理能力。Span<T> 适用于需要直接、高效访问内存的场景,而 Memory<T> 则更适合处理复杂的内存所有权和跨线程、异步操作。合理运用这些类型,可以显著提升程序的性能和资源利用率。在实际开发中,我们应根据具体的需求和场景,灵活选择使用这些类型,以达到最佳的编程效果。 ======================================================================
前些天发现了一个比较好玩的人工智能学习网站,通俗易懂,风趣幽默,可以了解了解AI基础知识,人工智能教程,不是一堆数学公式和算法的那种,用各种举例子来学习,读起来比较轻松,有兴趣可以看一下。
人工智能教程

相关文章:

.net Span类型和Memory类型

.NET 中 Span 类型和 Memory 类型的深度剖析 在 .NET 编程的世界里&#xff0c;高效处理内存是提升程序性能的关键。Span<T> 和 Memory<T> 类型的出现&#xff0c;为开发者提供了强大而灵活的工具&#xff0c;用于高效地访问和操作连续内存区域。今天&#xff0c;…...

Dify工具插件开发和智能体开发全流程

想象一下&#xff0c;你正在开发一个 AI 聊天机器人&#xff0c;想让它能实时搜索 Google、生成图像&#xff0c;甚至自动规划任务&#xff0c;但手动集成这些功能耗时又复杂。Dify 来了&#xff01;这个开源的 AI 应用平台让你轻松开发工具插件和智能体策略插件&#xff0c;快…...

ES6——对象扩展之Set对象

在ES6&#xff08;ECMAScript 2015&#xff09;中&#xff0c;Set 对象允许存储任何类型的唯一值&#xff0c;无论是原始值还是对象引用。Set 对象有一些有用的方法&#xff0c;可以操作集合中的数据。以下是一些常用的 Set 对象方法&#xff1a; 方法描述 add 向 Set 对象添加…...

AI书签管理工具开发全记录(十三):TUI基本框架搭建

文章目录 AI书签管理工具开发全记录&#xff08;十三&#xff09;&#xff1a;TUI基本框架搭建前言 &#x1f4dd;1.TUI介绍 &#x1f50d;2. 框架选择 ⚙️3. 功能梳理 &#x1f3af;4. 基础框架搭建⚙️4.1 安装4.2 参数设计4.3 绘制ui4.3.1 设计结构体4.3.2 创建头部4.3.3 创…...

<2>-MySQL库的操作

目录 一&#xff0c;创建数据库 二&#xff0c;查看字符集和校验规则 三&#xff0c;修改数据库 四&#xff0c;删除数据库 五&#xff0c;备份和恢复数据库 六&#xff0c;查看连接 一&#xff0c;创建数据库 创建一个名为bin_db的数据库&#xff0c;并设置字符集为utf8…...

Apache DolphinScheduler 和 Apache Airflow 对比

Apache DolphinScheduler 和 Apache Airflow 都是开源的工作流调度平台&#xff0c;用于管理和编排复杂的数据处理任务和管道。以下是对两者在功能、架构、使用场景等方面的对比&#xff0c;用中文清晰说明&#xff1a; 1. 概述 Apache DolphinScheduler&#xff1a; 一个分布…...

初识结构体,整型提升及操作符的属性

目录 一、结构体成员访问操作符1.1 结构体二、操作符的属性&#xff1a;优先级、结合性2.1 优先级2.2 结合性C 运算符优先级 三、表达式求值3.1 整型提升3.2 算数转化 总结 一、结构体成员访问操作符 1.1 结构体 C语言已经提供了内置类型&#xff0c;如&#xff1a;char,shor…...

检测到 #include 错误。请更新 includePath。已为此翻译单元(D:\软件\vscode\test.c)禁用波形曲线

原文链接&#xff1a;【VScodeMinGw】安装配置教程 下载mingw64 打开可以看到bin文件夹下是多个.exe文件&#xff0c;gcc.exe地址在环境配置中要用到 原文链接&#xff1a;VSCode中出现“#include错误&#xff0c;请更新includePath“问题&#xff0c;解决方法 重新VScode后…...

python --导出数据库表结构(pymysql)

import pymysql from pymysql.cursors import DictCursor from typing import Optional, Dict, List, Anyclass DBSchemaExporter:"""MySQL数据库表结构导出工具&#xff0c;支持提取表和字段注释使用示例:>>> exporter DBSchemaExporter("local…...

如何自动部署GitLab项目

如何自动部署 原理 GitLab有预制的钩子, 在代码提交/合并等事件中,会自动调用WebHoos, 即向该URL发送POST请求在布署服务器上监听该POST, 验证通过后执行相关的布置Shell脚本, 即可完成自动布署 配置环境 安装Python和Pip 2.如果需要, 安装python的requests模块和argparse模…...

在 Windows 系统上运行 Docker 容器中的 Ubuntu 镜像并显示 GUI

在 Windows 上安装一个 X Server&#xff08;如 VcXsrv 或 X410&#xff09;&#xff0c;Ubuntu 容器通过网络将图形界面转发到 Windows。 步骤&#xff1a; 安装 X Server&#xff1a; 推荐使用VcXsrv&#xff0c;免费开源。 安装后运行 XLaunch&#xff0c;选择&#xff1…...

基于 COM 的 XML 解析技术(MSXML) 的总结

✅ 一、COM 与 MSXML 简要说明 &#x1f537; 什么是 COM&#xff1f; COM&#xff08;Component Object Model&#xff09;是一种 Windows 平台下的组件技术&#xff0c;可以实现在不重新编译代码的前提下复用组件。 特点&#xff1a; 用 接口调用方式 解耦依赖&#xff1b…...

多分辨率 LCD 的 GUI 架构设计与实现

1.1多分辨率显示系统的挑战与解决方案 1.1.1 分辨率适配的核心问题 在嵌入式系统中,同时支持不同分辨率的 LCD(如 240160、320480 等)面临以下挑战: 布局适配:同一界面元素在不同分辨率下需要调整大小和位置 字体显示:小分辨率屏幕需要更小的字体,而大分辨率需要更清…...

2025年,百度智能云打响AI落地升维战

如果说从AI到Agent是对于产品落地形态的共识&#xff0c;那么如今百度智能云打响的恰是一个基于Agent进行TO B行业表达的AI生产力升维战。 在这个新的工程体系能力里&#xff0c;除了之前百度Create大会上提出的面向Agent的RAG能力等通用能力模块&#xff0c;对更为专业、个性…...

Seed1.5-VL登顶,国产闭源模型弯道超车丨多模态模型5月最新榜单揭晓

随着图像、文本、语音、视频等多模态信息融合能力的持续增强&#xff0c;多模态大模型在感知理解、逻辑推理和内容生成等任务中的综合表现不断提升&#xff0c;正在展现出愈发接近人类的智能水平。多模态能力也正在从底层的感知理解&#xff0c;迈向具备认知、推理、决策能力的…...

SON.stringify()和JSON.parse()之间的转换

1.JSON.stringify() 作用&#xff1a;将对象、数组转换成字符串 const obj {code: "500",message: "出错了", }; const jsonString JSON.stringify(obj); console.log(jsonString);//"{"code":"Mark Lee","message"…...

【学习笔记】构造函数+重载相关

【学习笔记】构造函数重载相关 一、构造函数 构造函数在创建对象的过程就会执行&#xff0c;带参数与不带参数&#xff0c;带参数的构造函数会默认将成员变量赋值传进去的参数。 class Layer { private:int layer_id; // 层IDstd::string layer_json; // 层的JSON配置…...

JVM——打开JVM后门的钥匙:反射机制

引入 在Java的世界里&#xff0c;反射机制&#xff08;Reflection&#xff09;就像一把万能钥匙&#xff0c;能够打开JVM的“后门”&#xff0c;让开发者在运行时突破静态类型的限制&#xff0c;动态操控类的内部结构。想象一下&#xff0c;传统的Java程序如同按菜单点菜的食客…...

第3章——SSM整合

一、整合持久层框架MyBatis 1.准备数据库表及数据 创建数据库&#xff1a;springboot 使用IDEA工具自带的mysql插件来完成表的创建和数据的准备&#xff1a; 创建表 表创建成功后&#xff0c;为表准备数据&#xff0c;如下&#xff1a; 2.创建SpringBoot项目 使用脚手架创建…...

VTK 显示文字、图片及2D/3D图

1. 基本环境设置 首先确保你已经安装了VTK库&#xff0c;并配置好了C开发环境。 #include <vtkSmartPointer.h> #include <vtkRenderWindow.h> #include <vtkRenderWindowInteractor.h> #include <vtkRenderer.h> 2. 显示文字 2D文字 #include &l…...

小白如何在cursor中使用mcp服务——以使用notion的api为例

1. 首先安装node.js,在这一步的时候不要勾选不要勾选 2. 安装完之后,前往notion页面 我的创作者个人资料 | Notion 前往集成页面&#xff0c;添加新集成&#xff0c;自己输入名字&#xff0c;选择内部 新建完之后&#xff0c;进入选择只读 复制密匙 然后前往cursor页面 新建…...

引领AI安全新时代 Accelerate 2025北亚巡展·北京站成功举办

6月5日&#xff0c;网络安全行业年度盛会——"Accelerate 2025北亚巡展北京站"圆满落幕&#xff01;来自智库、产业界、Fortinet管理层及技术团队的权威专家&#xff0c;与来自各行业的企业客户代表齐聚一堂&#xff0c;围绕"AI智御全球引领安全新时代"主题…...

为什么说数列是特殊的函数

文章目录 前情概要函数特性特殊之处典例剖析前情概要 高三的学生几乎都听老师说过,数列是特殊的函数,那么如何理解这句话呢,无外乎需要关注两点:①函数性,②特殊性,以下举例说明,帮助各位学子理解。 函数特性 既然是按照一定的次序排列而成的一列数字,那么这些数字(…...

解决uniapp开发app map组件最高层级 遮挡自定义解决底部tabbar方法

subNvue&#xff0c;是 vue 页面的原生子窗体&#xff0c;把weex渲染的原生界面当做 vue 页面的子窗体覆盖在页面上。它不是全屏页面&#xff0c;它给App平台vue页面中的层级覆盖和原生界面自定义提供了更强大和灵活的解决方案。它也不是组件&#xff0c;就是一个原生子窗体。 …...

96. 2017年蓝桥杯省赛 - Excel地址(困难)- 进制转换

96. Excel地址&#xff08;进制转换&#xff09; 1. 2017年蓝桥杯省赛 - Excel地址&#xff08;困难&#xff09; 标签&#xff1a;2017 省赛 1.1 题目描述 Excel 单元格的地址表示很有趣&#xff0c;它使用字母来表示列号。 比如&#xff0c; A 表示第 1 列&#xff0c;…...

PPT转图片拼贴工具 v1.0

软件介绍 这个软件的作用就是将单个PPT的每一页转换为单独的图片&#xff0c;然后将图片进行拼接起来。 但是我没有还没有解决一次性处理多个文件。 效果展示如下&#xff1a; 软件安装 软件源码 import os import re import win32com.client from PIL import Imagedef con…...

大模型在脑梗塞后遗症风险预测及治疗方案制定中的应用研究

目录 一、引言 1.1 研究背景与意义 1.2 研究目的与方法 1.3 国内外研究现状 二、脑梗塞概述 2.1 定义与分类 2.2 发病机制与病理生理过程 2.3 临床表现与诊断方法 三、大模型技术原理与应用现状 3.1 基本概念与技术架构 3.2 在医疗领域的应用案例与优势 3.3 适用于…...

Qwen2.5-VL - 模型结构

Qwen2.5-VL - 模型结构 flyfish 配置项 (Configuration)Qwen2.5-VL-3BQwen2.5-VL-7BQwen2.5-VL-72B视觉Transformer (ViT)隐藏层大小 (Hidden Size)128012801280层数 (# Layers)323232头数 (# Num Heads)161616中间层大小 (Intermediate Size)345634563456patch尺寸 (Patch S…...

【QT常用技术讲解】多线程执行后台命令行的两种方式(后台运行和返回打印信息)

前言 QT调用后台命令行&#xff0c;通常有两种场景&#xff1a;执行命令&#xff0c;等待并获取返回结果&#xff1b;执行命令&#xff0c;让程序后台一直执行(孤儿进程)&#xff0c;不需要获取命令返回的结果。以下是分享在国产信创桌面操作系统&#xff08;麒麟kylin、统信UO…...

【行驶证识别成表格】批量OCR行驶证识别与Excel自动化处理系统,行驶证扫描件和照片图片识别后保存为Excel表格,基于QT和华为ocr识别的实现教程

在车辆管理、物流运输、保险理赔等领域&#xff0c;经常需要处理大量的行驶证信息。传统的人工录入方式效率低、易出错&#xff0c;而使用 OCR 技术可以自动识别行驶证图片中的文字信息&#xff0c;极大提高数据处理效率。该系统可以应用于以下场景&#xff1a; 保险公司快速…...