How do I generate a stream from a string?

后端 未结 12 727
春和景丽
春和景丽 2020-11-22 14:29

I need to write a unit test for a method that takes a stream which comes from a text file. I would like to do do something like this:

Stream s = GenerateStre         


        
相关标签:
12条回答
  • 2020-11-22 15:10
    public Stream GenerateStreamFromString(string s)
    {
        return new MemoryStream(Encoding.UTF8.GetBytes(s));
    }
    
    0 讨论(0)
  • 2020-11-22 15:13

    A good combination of String extensions:

    public static byte[] GetBytes(this string str)
    {
        byte[] bytes = new byte[str.Length * sizeof(char)];
        System.Buffer.BlockCopy(str.ToCharArray(), 0, bytes, 0, bytes.Length);
        return bytes;
    }
    
    public static Stream ToStream(this string str)
    {
        Stream StringStream = new MemoryStream();
        StringStream.Read(str.GetBytes(), 0, str.Length);
        return StringStream;
    }
    
    0 讨论(0)
  • 2020-11-22 15:16

    Add this to a static string utility class:

    public static Stream ToStream(this string str)
    {
        MemoryStream stream = new MemoryStream();
        StreamWriter writer = new StreamWriter(stream);
        writer.Write(str);
        writer.Flush();
        stream.Position = 0;
        return stream;
    }
    

    This adds an extension function so you can simply:

    using (var stringStream = "My string".ToStream())
    {
        // use stringStream
    }
    
    0 讨论(0)
  • If you need to change the encoding I vote for @ShaunBowe's solution. But every answer here copies the whole string in memory at least once. The answers with ToCharArray + BlockCopy combo do it twice.

    If that matters here is a simple Stream wrapper for the raw UTF-16 string. If used with a StreamReader select Encoding.Unicode for it:

    public class StringStream : Stream
    {
        private readonly string str;
    
        public override bool CanRead => true;
        public override bool CanSeek => true;
        public override bool CanWrite => false;
        public override long Length => str.Length * 2;
        public override long Position { get; set; } // TODO: bounds check
    
        public StringStream(string s) => str = s ?? throw new ArgumentNullException(nameof(s));
    
        public override long Seek(long offset, SeekOrigin origin)
        {
            switch (origin)
            {
                case SeekOrigin.Begin:
                    Position = offset;
                    break;
                case SeekOrigin.Current:
                    Position += offset;
                    break;
                case SeekOrigin.End:
                    Position = Length - offset;
                    break;
            }
    
            return Position;
        }
    
        private byte this[int i] => (i & 1) == 0 ? (byte)(str[i / 2] & 0xFF) : (byte)(str[i / 2] >> 8);
    
        public override int Read(byte[] buffer, int offset, int count)
        {
            // TODO: bounds check
            var len = Math.Min(count, Length - Position);
            for (int i = 0; i < len; i++)
                buffer[offset++] = this[(int)(Position++)];
            return (int)len;
        }
    
        public override int ReadByte() => Position >= Length ? -1 : this[(int)Position++];
        public override void Flush() { }
        public override void SetLength(long value) => throw new NotSupportedException();
        public override void Write(byte[] buffer, int offset, int count) => throw new NotSupportedException();
        public override string ToString() => str; // ;)     
    }
    

    And here is a more complete solution with necessary bound checks (derived from MemoryStream so it has ToArray and WriteTo methods as well).

    0 讨论(0)
  • 2020-11-22 15:29

    Modernized and slightly modified version of the extension methods for ToStream:

    public static Stream ToStream(this string value) => ToStream(value, Encoding.UTF8);
    
    public static Stream ToStream(this string value, Encoding encoding) 
                              => new MemoryStream(encoding.GetBytes(value ?? string.Empty));
    

    Modification as suggested in @Palec's comment of @Shaun Bowe answer.

    0 讨论(0)
  • 2020-11-22 15:30
    public static Stream GenerateStreamFromString(string s)
    {
        var stream = new MemoryStream();
        var writer = new StreamWriter(stream);
        writer.Write(s);
        writer.Flush();
        stream.Position = 0;
        return stream;
    }
    

    Don't forget to use Using:

    using (var stream = GenerateStreamFromString("a,b \n c,d"))
    {
        // ... Do stuff to stream
    }
    

    About the StreamWriter not being disposed. StreamWriter is just a wrapper around the base stream, and doesn't use any resources that need to be disposed. The Dispose method will close the underlying Stream that StreamWriter is writing to. In this case that is the MemoryStream we want to return.

    In .NET 4.5 there is now an overload for StreamWriter that keeps the underlying stream open after the writer is disposed of, but this code does the same thing and works with other versions of .NET too.

    See Is there any way to close a StreamWriter without closing its BaseStream?

    0 讨论(0)
提交回复
热议问题