Как получить MemoryStream из потока в .NET? - PullRequest
59 голосов
/ 09 июля 2010

У меня есть следующий метод конструктора, который открывает MemoryStream из пути к файлу:

MemoryStream _ms;

public MyClass(string filePath)
{
    byte[] docBytes = File.ReadAllBytes(filePath);
    _ms = new MemoryStream();
    _ms.Write(docBytes, 0, docBytes.Length);
}

Мне нужно изменить это, чтобы принять Stream вместо пути к файлу. Какой самый простой / эффективный способ получить MemoryStream от объекта Stream?

Ответы [ 9 ]

179 голосов
/ 09 июля 2010

В .NET 4 вы можете использовать Stream.CopyTo для копирования потока вместо методов домашнего приготовления, перечисленных в других ответах.

MemoryStream _ms;

public MyClass(Stream sourceStream)

    _ms = new MemoryStream();
    sourceStream.CopyTo(_ms);
}
42 голосов
/ 28 сентября 2015

Используйте это:

var memoryStream = new MemoryStream();
stream.CopyTo(memoryStream);

Это преобразует Stream в MemoryStream.

28 голосов
/ 09 июля 2010

Если вы изменяете свой класс так, чтобы он принимал Stream вместо имени файла, не беспокойтесь о преобразовании в MemoryStream.Пусть основной поток обрабатывает операции:

public class MyClass
{ 
    Stream _s;

    public MyClass(Stream s) { _s = s; }
}

Но если вам действительно нужен MemoryStream для внутренних операций, вам придется скопировать данные из исходного потока в MemoryStream:

public MyClass(Stream stream)
{
    _ms = new MemoryStream();
    CopyStream(stream, _ms);
}

// Merged From linked CopyStream below and Jon Skeet's ReadFully example
public static void CopyStream(Stream input, Stream output)
{
    byte[] buffer = new byte[16*1024];
    int read;
    while((read = input.Read (buffer, 0, buffer.Length)) > 0)
    {
        output.Write (buffer, 0, read);
    }
}
3 голосов
/ 09 июля 2010

Вам нужно будет прочитать все данные из объекта Stream в буфер byte[] и затем передать их в MemoryStream через его конструктор. Может быть лучше быть более точным в отношении типа объекта потока, который вы используете. Stream является очень общим и может не реализовывать атрибут Length, который довольно полезен при чтении данных.

Вот код для вас:

public MyClass(Stream inputStream) {
    byte[] inputBuffer = new byte[inputStream.Length];
    inputStream.Read(inputBuffer, 0, inputBuffer.Length);

    _ms = new MemoryStream(inputBuffer);
}

Если объект Stream не реализует атрибут Length, вам придется реализовать что-то вроде этого:

public MyClass(Stream inputStream) {
    MemoryStream outputStream = new MemoryStream();

    byte[] inputBuffer = new byte[65535];
    int readAmount;
    while((readAmount = inputStream.Read(inputBuffer, 0, inputBuffer.Length)) > 0)
        outputStream.Write(inputBuffer, 0, readAmount);

    _ms = outputStream;
}
2 голосов
/ 09 июля 2010

Я использую эту комбинацию методов расширения:

    public static Stream Copy(this Stream source)
    {
        if (source == null)
            return null;

        long originalPosition = -1;

        if (source.CanSeek)
            originalPosition = source.Position;

        MemoryStream ms = new MemoryStream();

        try
        {
            Copy(source, ms);

            if (originalPosition > -1)
                ms.Seek(originalPosition, SeekOrigin.Begin);
            else
                ms.Seek(0, SeekOrigin.Begin);

            return ms;
        }
        catch
        {
            ms.Dispose();
            throw;
        }
    }

    public static void Copy(this Stream source, Stream target)
    {
        if (source == null)
            throw new ArgumentNullException("source");
        if (target == null)
            throw new ArgumentNullException("target");

        long originalSourcePosition = -1;
        int count = 0;
        byte[] buffer = new byte[0x1000];

        if (source.CanSeek)
        {
            originalSourcePosition = source.Position;
            source.Seek(0, SeekOrigin.Begin);
        }

        while ((count = source.Read(buffer, 0, buffer.Length)) > 0)
            target.Write(buffer, 0, count);

        if (originalSourcePosition > -1)
        {
            source.Seek(originalSourcePosition, SeekOrigin.Begin);
        }
    }
2 голосов
/ 09 июля 2010

Как скопировать содержимое одного потока в другой?

посмотри на это. принять поток и скопировать в память. Вы не должны использовать .Length только для Stream, потому что это не обязательно реализовано в каждом конкретном потоке.

1 голос
/ 09 июля 2010
public static void Do(Stream in)
{
    _ms = new MemoryStream();
    byte[] buffer = new byte[65536];
    while ((int read = input.Read(buffer, 0, buffer.Length))>=0)
        _ms.Write (buffer, 0, read);
}
0 голосов
/ 02 ноября 2018

Вы можете просто сделать:

var ms = new MemoryStream(File.ReadAllBytes(filePath));

Позиция потока равна 0 и готова к использованию.

0 голосов
/ 15 января 2014
byte[] fileData = null;
using (var binaryReader = new BinaryReader(Request.Files[0].InputStream))
{
    fileData = binaryReader.ReadBytes(Request.Files[0].ContentLength);
}
...