在.NET开发中,流(Stream)是一个用于处理输入和输出的抽象类,MemoryStream
是流的一个具体实现,它允许我们在内存中读写数据,就像操作文件一样,而无需涉及磁盘 I/O 操作。尤其适合需要快速读写、转换或传输数据的场景。本文将详细讲解MemoryStream
的使用。
MemoryStream
是 System.IO
命名空间中的一个类,它允许我们在内存中创建可读写的流。与文件流或网络流不同,MemoryStream
的数据存储在内存中,它不需要依赖物理文件,因此读写速度非常快,适合处理临时数据(如网络传输、临时缓存、序列化对象等)。但会占用一定的内存资源。
MemoryStream
是System.IO
命名空间中的一个类,它实现了Stream
抽象类,提供了一系列用于操作数据流的属性和方法。
在数据处理场景中,频繁的磁盘IO操作(如读写文件)会显著降低程序性能,尤其是面对海量数据或高频读写需求时。MemoryStream作为C#中的内存流,将数据存储在内存而非硬盘中,避免了磁盘IO瓶颈,读写速度更快。它适用于网络数据传输、临时缓存、二进制数据处理等场景,是实现高性能代码的利器!
MemoryStream
有多个构造函数,可以根据需要选择合适的构造函数来初始化MemoryStream
。
使用无参构造函数可以创建一个空白的 MemoryStream
对象,其初始容量为 0,随着数据写入自动扩展。
using System.IO;
MemoryStream memoryStream = new MemoryStream();
使用带参构造函数可以根据指定的容量创建 MemoryStream
对象,或者从一个字节数组创建。
MemoryStream memoryStream = new MemoryStream(1024);
创建一个初始容量为1024字节的MemoryStream
。
byte[] buffer = new byte[] { 72, 101, 108, 108, 111, 44, 32, 87, 111, 114, 108, 100, 33 };
MemoryStream ms = new MemoryStream(buffer);
使用现有的字节数组初始化MemoryStream
。
MemoryStream
提供了多种方法来写入数据,最常用的是 Write
方法和 WriteByte
方法。
使用 Write
方法可以将一个字节数组写入 MemoryStream
。
byte[] data = new byte[] { 72, 101, 108, 108, 111 };
memoryStream.Write(data, 0, data.Length);
向 MemoryStream
写入一个字符串,需要将字符串转换为字节数组。
string text = "Hello, World!";
byte[] data = System.Text.Encoding.UTF8.GetBytes(text);
memoryStream.Write(data, 0, data.Length);
WriteByte
方法可以逐字节写入数据。
string text = "Hello, World!";
byte[] data = System.Text.Encoding.UTF8.GetBytes(text);
foreach (byte b in data)
{
memoryStream.WriteByte(b);
}
MemoryStream
提供了多种方法来读取数据,最常用的是 Read
方法和 ReadByte
方法。
从 MemoryStream
读取一定数量的字节到字节数组中。
byte[] buffer = new byte[11];
int bytesRead = memoryStream.Read(buffer, 0, buffer.Length);
从 MemoryStream
读取一定数量的字节,然后将其转换为字符串。
byte[] buffer = new byte[11];
int bytesRead = memoryStream.Read(buffer, 0, buffer.Length);
string text = System.Text.Encoding.UTF8.GetString(buffer, 0, bytesRead);
ReadByte
方法可以逐字节读取数据。
List<byte> byteList = new List<byte>();
while (memoryStream.Position < memoryStream.Length)
{
byteList.Add((byte)memoryStream.ReadByte());
}
string text = System.Text.Encoding.UTF8.GetString(byteList.ToArray());
获取或设置分配给MemoryStream
的字节数
MemoryStream memoryStream = new MemoryStream(1024);
int capacity = memoryStream.Capacity; // 输出:1024
获取MemoryStream
中实际使用的数据长度。
MemoryStream memoryStream = new MemoryStream(1024);
long length = memoryStream.Length; // length = 0
获取或设置MemoryStream
的当前读写位置。
memoryStream.Position = 0; // 定位到流的开头
bool canRead = ms.CanRead;
bool canWrite = ms.CanWrite;
bool canSeek = ms.CanSeek;
表示MemoryStream
是否支持读取、写入和定位操作。对于MemoryStream
,这些属性通常返回true
。
使用 SetLength
方法可以设置 MemoryStream
的长度,如果新长度小于当前长度,数据将被截断;如果新长度大于当前长度,数据将被扩展。
memoryStream.SetLength(50);
ms.Seek(0, SeekOrigin.Begin);
移动MemoryStream
的当前读写位置。
使用 ToArray
方法可以将 MemoryStream
的内容转换为字节数组。
byte[] allBytes = memoryStream.ToArray();
byte[] buffer = memoryStream.GetBuffer();
GetBuffer
方法返回底层缓冲区的完整字节数组(包含未使用的空间),ToArray
方法返回仅包含有效数据的数组(排除未使用的空间)。
关于 ToArray 和 GetBuffer 方法的区别,详见:C# MemoryStream 中 ToArray 和 GetBuffer 的区别
// 创建空的 MemoryStream
using (MemoryStream ms = new MemoryStream())
{
// 写入字符串
byte[] data = System.Text.Encoding.UTF8.GetBytes("Hello, MemoryStream!");
ms.Write(data, 0, data.Length);
// 写入后的位置自动前进
Console.WriteLine($"当前位置:{ms.Position}"); // 输出:20(假设 UTF-8 编码)
}
using (MemoryStream ms = new MemoryStream())
{
// 写入数据后重置位置到开头
ms.Write(data, 0, data.Length);
ms.Position = 0; // 必须重置位置才能读取
// 读取数据
byte[] buffer = new byte[ms.Length];
ms.Read(buffer, 0, (int)ms.Length);
string result = System.Text.Encoding.UTF8.GetString(buffer);
Console.WriteLine(result); // 输出:Hello, MemoryStream!
}
Tips:
除了使用Position
属性重置位置外,读写前还可用Seek()
调整指针位置,如stream.Seek(0, SeekOrigin.Begin)
。
下面是一个完整的示例,演示了如何使用 MemoryStream
:
public class Program
{
public static void Main(string[] args)
{
// 创建 MemoryStream
MemoryStream memoryStream = new MemoryStream();
// 获取当前读写的位置
Console.WriteLine($"MemoryStream Position: {memoryStream.Position}");// 输出:MemoryStream Position: 0
string text = "Hello, World!";
byte[] data = System.Text.Encoding.UTF8.GetBytes(text);
// 写入数据
memoryStream.Write(data, 0, data.Length);
// 转换为字节数组
Console.WriteLine(BitConverter.ToString(memoryStream.ToArray())); //输出:48-65-6C-6C-6F-2C-20-57-6F-72-6C-64-21
// 设置长度
memoryStream.SetLength(50);
// 获取长度
Console.WriteLine($"MemoryStream length: {memoryStream.Length}"); // 输出:MemoryStream length: 50
// 获取当前读写的位置
Console.WriteLine($"MemoryStream Position: {memoryStream.Position}");// 输出:MemoryStream Position: 13
// 读取数据
byte[] buffer = new byte[5];
int bytesRead = memoryStream.Read(buffer, 0, buffer.Length);
// 输出结果
Console.WriteLine(BitConverter.ToString(buffer)); // 输出:00-00-00-00-00
// 定位
memoryStream.Position = 0;
// 再次读取数据
bytesRead = memoryStream.Read(buffer, 0, buffer.Length);
Console.WriteLine(BitConverter.ToString(buffer)); // 输出:48 65 6C 6C 6F
Console.WriteLine(Encoding.UTF8.GetString(buffer)); //输出:Hello
// 清空 MemoryStream
memoryStream.SetLength(0);
memoryStream.Position = 0;
// 检查是否清空
Console.WriteLine("MemoryStream length after clear: " + memoryStream.Length); // 输出:MemoryStream length after clear: 0
}
}
通过这个示例,我们可以看到 MemoryStream
在处理内存中的数据流时是多么灵活和有用。它不仅可以用于临时存储数据,还可以用于实现复杂的数据处理逻辑。
使用 StreamReader
/StreamWriter
public class Program
{
public static void Main(string[] args)
{
using (MemoryStream ms = new MemoryStream())
{
// 创建一个StreamWriter,用于向MemoryStream写入字符串
using (StreamWriter sw = new StreamWriter(ms, Encoding.UTF8, 1024, leaveOpen: true))
{
// leaveOpen: true 的作用:
// 防止 StreamWriter 关闭时连带关闭底层的 MemoryStream,确保后续 StreamReader 可正常操作流
sw.WriteLine("Hello, World!");
sw.WriteLine("This is a test.");
}
// 将MemoryStream的位置重置到开头
ms.Seek(0, SeekOrigin.Begin);
// 创建一个StreamReader,用于从MemoryStream读取字符串
using (StreamReader sr = new StreamReader(ms, Encoding.UTF8))
{
string line;
while ((line = sr.ReadLine()) != null)
{
Console.WriteLine(line);
}
}
}
}
}
在这个例子中,我们首先创建了一个MemoryStream
实例,然后使用StreamWriter
向MemoryStream
写入了两行字符串。写入完成后,我们将MemoryStream
的位置重置到开头,接着使用StreamReader
从MemoryStream
读取字符串并打印到控制台。
使用 BinaryReader
/BinaryWriter
public class Program
{
public static void Main(string[] args)
{
using (MemoryStream ms = new MemoryStream())
{
// 创建一个BinaryWriter,用于向MemoryStream写入二进制数据
using (BinaryWriter writer = new BinaryWriter(ms,Encoding.UTF8,leaveOpen:true))
{
// leaveOpen: true 的作用:
// 防止 StreamWriter 关闭时连带关闭底层的 MemoryStream,确保后续 StreamReader 可正常操作流
writer.Write(32);// 写入整数
writer.Write(12.3f);// 写入单精度浮点数
writer.Write("This is a test.");// 写入字符串
}
// 将MemoryStream的位置重置到开头
ms.Seek(0, SeekOrigin.Begin);
// 创建一个 BinaryReader,用于从MemoryStream读取二进制数据
using (BinaryReader reader = new BinaryReader(ms, Encoding.UTF8))
{
Console.WriteLine(reader.ReadInt32()); //输出:32
Console.WriteLine(reader.ReadSingle()); //输出:12.3
Console.WriteLine(reader.ReadString()); //输出:This is a test.
}
}
}
}
将内存流作为网络传输的缓冲区:
// 服务端接收数据
NetworkStream ns = client.GetStream();
MemoryStream ms = new MemoryStream();
ns.CopyTo(ms); // 将网络流复制到内存流
byte[] buffer = ms.ToArray();
使用using
语句:确保流对象及时释放,避免内存泄漏。
using (MemoryStream stream = new MemoryStream()) { /*...*/ }
预分配容量:若已知数据大小,初始化时指定Capacity
减少动态扩容开销。
频繁写入数据时,指定初始容量可避免内存频繁扩容:
// 预分配 1MB 内存
using (MemoryStream ms = new MemoryStream(1024 * 1024))
{
// 写入大量数据
}
通过 SetLength(0)
和 Seek
方法或 设置Position
重置流:
using (MemoryStream ms = new MemoryStream())
{
ms.Write(data, 0, data.Length);
// 重置流并清空内容
ms.SetLength(0);
ms.Position = 0; //或 ms.Seek(0, SeekOrigin.Begin);
// 重新写入新数据
ms.Write(newData, 0, newData.Length);
}
using System;
using System.Text;
using System.Text.Json;
public class Person
{
public string Name { get; set; }
public int Age { get; set; }
}
class Program
{
static void Main()
{
Person person = new Person { Name = "John Doe", Age = 30 };
// 序列化对象到MemoryStream
using (MemoryStream ms = new MemoryStream())
{
JsonSerializer.Serialize(ms, person);
// 将MemoryStream的位置重置到开头
ms.Seek(0, SeekOrigin.Begin);
// 反序列化对象从MemoryStream
Person deserializedPerson = JsonSerializer.Deserialize<Person>(ms);
Console.WriteLine("Name: " + deserializedPerson.Name); // 输出:Name: John Doe
Console.WriteLine("Age: " + deserializedPerson.Age); // 输出:Age: 30
}
}
}
using System;
using System.IO;
public class MemoryStreamExample
{
public static void Main()
{
// 创建一个MemoryStream作为临时缓冲区
using (MemoryStream ms = new MemoryStream())
{
// 写入一些数据到MemoryStream
byte[] data = new byte[] { 1, 2, 3, 4, 5 };
ms.Write(data, 0, data.Length);
// 将MemoryStream作为参数传递给其他方法
ProcessData(ms);
}
}
public static void ProcessData(MemoryStream ms)
{
// 将MemoryStream的位置重置到开头
ms.Seek(0, SeekOrigin.Begin);
// 读取数据 from MemoryStream
byte[] buffer = new byte[ms.Length];
ms.Read(buffer, 0, buffer.Length);
Console.WriteLine("Received data:");
foreach (byte b in buffer)
{
Console.Write(b + " ");
}
}
}
案例:日志文件关键字筛选
假设需要从多个.log
文件中提取含特定关键字的行,传统方法可能导致内存暴涨。
优化方案:
StreamReader
避免一次性加载大文件。MemoryStream
,减少磁盘IO次数。List<string> matchedLines = new List<string>();
foreach (var file in Directory.GetFiles("logs", "*.log"))
{
using (var reader = new StreamReader(file))
{
while (!reader.EndOfStream)
{
string line = reader.ReadLine();
if (Regex.IsMatch(line, "keyword"))
{
matchedLines.Add(line);
}
}
}
}
// 使用MemoryStream合并数据并写入文件
using (MemoryStream ms = new MemoryStream())
{
byte[] buffer = Encoding.UTF8.GetBytes(string.Join("\n", matchedLines));
ms.Write(buffer, 0, buffer.Length);
File.WriteAllBytes("result.txt", ms.ToArray());
}
实测性能提升显著// 错误示例:未重置位置导致读取失败
using (MemoryStream ms = new MemoryStream())
{
ms.Write(data, 0, data.Length);
byte[] buffer = new byte[ms.Length];
ms.Read(buffer, 0, buffer.Length); // 抛出异常,因为 Position 已在末尾
}
// 正确做法:重置位置
ms.Position = 0;
ms.Read(buffer, 0, buffer.Length);
当数据量超过内存限制时,改用 FileStream
:
// 替代方案:使用文件流
using (FileStream fs = new FileStream("temp.bin", FileMode.Create))
{
// 写入数据到文件流
}
通过 ToArray()
获取字节数组后,可异步处理:
public async Task ProcessAsync()
{
using (MemoryStream ms = new MemoryStream())
{
// 写入数据
byte[] data = ms.ToArray();
// 异步发送到网络
await client.SendAsync(data);
}
}
MemoryStream
的容量可以动态增长,以适应数据量的变化。CanRead
、CanWrite
和CanSeek
属性,便于灵活操作。资源管理
始终使用 using
语句确保流正确释放:
using (MemoryStream ms = new MemoryStream()) { ... }
位置重置
写入后读取前必须重置 Position
到 0
。
性能优化
MemoryStream
时,尽量指定初始容量,以减少动态增长的次数,提高性能。FileStream
替代。)回到目录页:C#/.NET 知识汇总
希望以上内容可以帮助到大家,如文中有不对之处,还请批评指正。
参考资料:
C# MemoryStream流的详解与示例
C# Stream 和 byte[] 之间的转换(文件流的应用)
C# Stream篇(五) – MemoryStream