字符串转字节流必须指定编码,Encoding.UTF8.GetBytes()是最安全轻量的方式;MemoryStream仅存储字节,不参与编码,需先编码再构造且设writable:false;避免StreamWriter+MemoryStream等反模式。

字符串转字节流必须指定编码,Encoding.UTF8 是最安全的默认选择
直接用 Encoding.UTF8.GetBytes() 得到 byte[],这是最轻量、最可控的方式。别一上来就塞进 MemoryStream——那只是后续操作的容器,不是转换本身。MemoryStream 不参与编码逻辑,它只管存字节。
常见错误:用 new MemoryStream().Write(Encoding.Default.GetBytes(str), 0, ...) 手动写,既绕路又容易漏 Seek(0);或者误以为 MemoryStream 自带编码能力,结果读出来乱码。
-
Encoding.UTF8覆盖绝大多数场景,兼容性好,不会因系统区域设置变化而行为不一致 -
Encoding.Default看似省事,实则危险——在中文 Windows 上是 GB2312,在英文系统上可能是 ISO-8859-1,跨环境必出问题 - 如果协议或 API 明确要求 ANSI 或 UTF-16(比如某些 Windows API 或旧版 COM),才显式换用
Encoding.ASCII或Encoding.Unicode
需要 MemoryStream 的真实场景:传给需要流接口的 API
比如 HttpClient.PostAsync() 的 HttpContent 构造、System.Drawing.Image.FromStream()、或序列化库如 JsonSerializer.SerializeAsync() 的流重载。这时候你不是“为了转而转”,而是“为了喂给某个方法而准备流”。
正确做法是:先编码得字节数组,再用该数组构造 MemoryStream,且**禁用可扩展性**(避免意外扩容):
var bytes = Encoding.UTF8.GetBytes(jsonString); using var stream = new MemoryStream(bytes, writable: false); // writable: false 很关键
- 用
new MemoryStream(byte[])构造器,比先 new 再 Write 更高效,也避免内部缓冲区二次拷贝 - 显式传
writable: false,防止下游代码误调stream.Write()导致不可预期行为 - 别用
new MemoryStream().Write(...)后忘记stream.Position = 0,否则读取时从末尾开始,返回空数据
StreamReader / StreamWriter 配合 MemoryStream 是反模式
有人想“用流的方式处理字符串”,于是写:new StreamWriter(new MemoryStream()) → WriteLine() → Flush() → 取 ToArray()。这纯属绕远路,性能差、代码冗余、还容易漏 Flush() 或 Dispose()。
除非你在拼接大量小字符串且需控制内存分配节奏(极少见),否则没必要。现代 C# 中 string.Concat、StringBuilder、甚至内插字符串都比流式写入更合适。
- 每次
StreamWriter.Write()都触发编码 + 缓冲区管理,开销远高于直接Encoding.GetBytes() -
MemoryStream.ToArray()总是复制一份新数组,而GetBuffer()返回内部引用但含未使用垃圾字节,易引发 bug - 若真要流式生成(如大 JSON 分块写入),应直接用
Utf8JsonWriter写入MemoryStream,而不是套一层StreamWriter
注意 MemoryStream 生命周期和 ToArray() 的陷阱
MemoryStream 本身不持有原始字节数组的独占权。如果你用 ToArray(),它会复制当前已写入内容;用 GetBuffer() 则返回底层缓冲区——但长度往往大于实际数据,末尾可能全是 0。
典型翻车点:把 stream.GetBuffer() 直接当有效字节传给网络或加密函数,结果多了一堆零字节,校验失败或解密报错。
- 安全做法:始终用
stream.ToArray()获取干净字节数组,或手动截取stream.GetBuffer().AsSpan(0, (int)stream.Length) - 如果只是临时传给一个同步 API(如
Convert.ToBase64String()),优先用ToArray(),语义清晰无歧义 - 异步流操作(如
CopyToAsync())要注意MemoryStream是否已Position = 0,否则可能什么也没复制
事情说清了就结束










