C#怎么使用Span和Memory C#如何用Span优化内存操作减少GC压力提升性能【进阶】
Span 不能跨 await 边界使用,因其为栈分配的 ref 类型,而 await 会将局部变量提升至堆上状态机中,导致编译错误 CS8346;正确做法是异步用 Memory,同步处理时再转 Span。Span 为什么不能跨 await 边界使用因为 Span<t></t> 是栈分配的 ref 类型,编译器禁止它逃逸到堆上——而 await 会触发状态机生成,可能把局部变量提升到堆上的状态机结构体里。一旦你写 async Task<span>> ReadAsync()</span>,编译器直接报错:CS8346: Cannot use a variable of type 'Span<byte>' in an async method</byte>。常见错误现象:在异步读取网络流或文件后想直接返回 Span<byte></byte>,结果编译不过,或强行用 Memory<t></t> 却没意识到后续切片仍受限。正确做法:异步操作用 Memory<t></t> 接收(如 await stream.ReadAsync(buffer) 中的 buffer 是 Memory<byte></byte>)同步处理阶段再转成 Span<byte></byte>:比如 var span = memory.Span,只在方法内做解析、查找、拷贝等短生命周期操作别试图把 Span<t></t> 存进字段、属性或返回给调用方——它不是设计来“传递”的,而是用来“临时操作”的Memory 和 Span 的参数传递差异二者语义不同:Span<t></t> 表示「当前栈上下文里一块连续内存」,Memory<t></t> 表示「一段可安全跨上下文访问的内存视图」。这个区别直接决定你该在哪儿用哪个。使用场景举例:写一个通用字符串解析函数,输入可能是数组、堆内存、本机内存、甚至 unmanaged 指针。如果函数只做纯内存扫描(比如找第一个逗号),用 ReadOnlySpan<char></char> 参数最轻量、零分配、无 GC 压力如果函数需要把结果缓存起来供后续多次访问,或要传给另一个异步方法,必须用 ReadOnlyMemory<char></char> —— 它能封装 ArraySegment<t></t>、T[]、string 等多种来源注意:从 Memory<t></t> 转 Span<t></t> 是廉价的(只是字段复制),但反过来不行;Span<t></t> 无法隐式转成 Memory<t></t>用 AsSpan() 切片时最容易漏掉的边界检查AsSpan() 本身不校验索引合法性,越界不会抛异常,而是引发未定义行为(尤其在 Release 模式下可能静默读错数据或崩溃)。 稿定AI 拥有线稿上色优化、图片重绘、人物姿势检测、涂鸦完善等功能
