आपको एक मेमोरीस्ट्रीम से स्ट्रिंग कैसे मिलता है?


532

अगर मुझे दिया जाता है MemoryStreamकि मुझे पता है कि मैं एक के साथ आबाद हो गया Stringहूं, तो मैं कैसे Stringवापस आ सकता हूं ?


1
यदि पाठक हमेशा आवश्यक हो तो कभी निश्चित नहीं। मेरे पास अतीत में मुद्दे थे इसलिए एक नियम के रूप में मैं हमेशा सुरक्षित पक्ष पर रहने के लिए करता हूं।
Crusty

जवाबों:


468

यह नमूना दिखाता है कि मेमोरी स्ट्रिंग में स्ट्रिंग को कैसे पढ़ा और लिखा जाए।


Imports System.IO

Module Module1
  Sub Main()
    ' We don't need to dispose any of the MemoryStream 
    ' because it is a managed object. However, just for 
    ' good practice, we'll close the MemoryStream.
    Using ms As New MemoryStream
      Dim sw As New StreamWriter(ms)
      sw.WriteLine("Hello World")
      ' The string is currently stored in the 
      ' StreamWriters buffer. Flushing the stream will 
      ' force the string into the MemoryStream.
      sw.Flush()
      ' If we dispose the StreamWriter now, it will close 
      ' the BaseStream (which is our MemoryStream) which 
      ' will prevent us from reading from our MemoryStream
      'sw.Dispose()

      ' The StreamReader will read from the current 
      ' position of the MemoryStream which is currently 
      ' set at the end of the string we just wrote to it. 
      ' We need to set the position to 0 in order to read 
      ' from the beginning.
      ms.Position = 0
      Dim sr As New StreamReader(ms)
      Dim myStr = sr.ReadToEnd()
      Console.WriteLine(myStr)

      ' We can dispose our StreamWriter and StreamReader 
      ' now, though this isn't necessary (they don't hold 
      ' any resources open on their own).
      sw.Dispose()
      sr.Dispose()
    End Using

    Console.WriteLine("Press any key to continue.")
    Console.ReadKey()
  End Sub
End Module

3
क्या यह फाइटव्यूटर को डिस्पोज करने नहीं जा रहा है, जब फ़ंक्शन वैसे भी दायरे से बाहर हो जाता है?
टिम कीटिंग

14
जब वैरिएबल स्कोप से बाहर चला जाता है तो डिस्पोजल को नहीं कहा जाता है। जब जीसी इसके चारों ओर हो जाता है, तो अंतिम रूप से कॉल किया जाएगा, लेकिन डिस्पोज कुछ ऐसी चीज है जिसे चर के दायरे से बाहर जाने से पहले बुलाया जाना चाहिए। मैं इसे ऊपर नहीं बुलाता क्योंकि मुझे पता है कि StreamWriter और StreamReader के कार्यान्वयन के लिए Dispose को कॉल करने की आवश्यकता नहीं है, यह कॉल को अंतर्निहित स्ट्रीम में पास करता है। हालाँकि, किसी भी चीज़ के लिए डिपोज़ को कॉल करने के लिए एक वैध तर्क दिया जा सकता है, जो कि आईडीसिसोप्लिक को लागू करता है क्योंकि आप भविष्य की रिलीज़ की गारंटी नहीं दे सकते, इसे निपटाने की आवश्यकता नहीं होगी।
ब्रायन

12
@MichaelEakins प्रश्न # VB.Net के रूप में टैग किए जाने पर C # में भी क्यों होना चाहिए?
रोलैंड शॉ

1
मुझे खुशी है कि मैंने "हेल्पर्स" के बारे में सीखा कि डिस्पोजल कॉल को उनकी अंतर्निहित धाराओं में पारित किया गया था, लेकिन यह एक खराब डिजाइन निर्णय जैसा लगता है।
गेरार्ड ओनील

2
बाद में इस निर्णय को कम कर दिया गया: msdn.microsoft.com/en-us/library/…
मार्क सोउल

310

आप भी उपयोग कर सकते हैं

Encoding.ASCII.GetString(ms.ToArray());

मुझे नहीं लगता कि यह कम कुशल है, लेकिन मैं इसकी कसम नहीं खा सकता था। यह आपको एक अलग एन्कोडिंग चुनने की भी सुविधा देता है, जबकि एक स्ट्रीमराइडर का उपयोग करते हुए आपको एक पैरामीटर के रूप में निर्दिष्ट करना होगा।


6
एन्कोडिंग System.Text नामस्थान में है
उत्तरबेन

2
मैं इस के बराबर PowerShell की तलाश कर रहा था और इसका उपयोग करना था। ([System.Text.Encoding] :: ASCII) .GetString (ms.ToArray ())
लुईस

यह समाधान उपयोगी है क्योंकि इसका उपयोग मेमोरीस्ट्रीम के बंद होने के बाद किया जा सकता है।
जैकब हॉर्बुलक

2
एफडब्ल्यूआईडब्ल्यू, मैंने पाया कि यह बहुत बड़े तारों के साथ काम नहीं कर रहा था, मुझे मिल रहा था OutOfMemoryException। का उपयोग करते हुए एक StreamReaderके बजाय समस्या हल हो।
एच। एच। टी।

चूंकि यह एक नुकसान हो सकता है: यह बाइट ऑर्डर मार्क से अवगत नहीं है और 00स्ट्रिंग की शुरुआत में हेक्स को शामिल कर सकता है । 00 3C 3F-> .<?हेक्स संपादक में लेकिन वीएस या नोटपैड ++ में <?:। इसलिए आप अंतर को तब भी नहीं देख सकते हैं जब आप तार को आंख से तुलना करते हैं, केवल एक तुलना उपकरण या हेक्स संपादक अंतर दिखाएगा। यदि आप अभी भी इसका उपयोग करते हैं, तो String.TrimStart के बारे में सोचें। देखें: docs.microsoft.com/en-us/dotnet/api/…
Skalli

99

MemoryStream को String में बदलने के लिए StreamReader का उपयोग करना।

<Extension()> _
Public Function ReadAll(ByVal memStream As MemoryStream) As String
    ' Reset the stream otherwise you will just get an empty string.
    ' Remember the position so we can restore it later.
    Dim pos = memStream.Position
    memStream.Position = 0

    Dim reader As New StreamReader(memStream)
    Dim str = reader.ReadToEnd()

    ' Reset the position so that subsequent writes are correct.
    memStream.Position = pos

    Return str
End Function

3
स्थिति को 0 पर सेट करना विधि के पुन: उपयोग की क्षमता को सीमित करता है - यह कॉलर को इसे प्रबंधित करने के लिए सबसे अच्छा है। क्या होगा यदि धारा में स्ट्रिंग से पहले डेटा होता है, जिसे कॉल करने वाला जानता है कि कैसे संभालना है?
एलेक्स लाइमैन

1
उपयोग कथन यह सुनिश्चित करेगा कि आपका स्ट्रीमराइडर निपटारा हो जाता है, लेकिन दस्तावेज का कहना है कि स्ट्रीमराइडर अंतर्निहित स्ट्रीम को बंद कर देता है जब वह निपटा जाता है। इसलिए, आपका तरीका मेमोरीस्ट्रीम को बंद कर देता है, यह पास हो जाता है, जो कि कॉल करने वालों के लिए वैचारिक रूप से बेकार है, भले ही मुझे मेमोरीस्ट्रीम पर संदेह हो।
२३:२३

तुम सही हो। आमतौर पर स्ट्रीम हेल्पर वर्गों पर डिस्पोज़ विधि का उपयोग करना एक बुरा विचार है, खासकर अगर स्ट्रीम को एक पैरामीटर के रूप में विधि में पारित किया जाता है। मैं इस उत्तर को अपडेट करूंगा। मेरे पास नीचे एक और पूर्ण उत्तर भी है।
ब्रायन

यदि आप उन कक्षाओं को विघटित करते हैं, तो आप देखेंगे कि डिस्पोज़ विधि किसी भी स्ट्रीम में
डिस्पोज़

39

StreamReader का उपयोग करें , फिर आप ReadToEnd पद्धति का उपयोग कर सकते हैं जो एक स्ट्रिंग लौटाता है।


12
मैं केवल उल्लेख करना चाहता हूं, कि Basestreamइसकी स्थिति ०. Like की तरह होनी चाहिए memoryStream.Position = 0;
अय्यूक Ayएविक

26
byte[] array = Encoding.ASCII.GetBytes("MyTest1 - MyTest2");
MemoryStream streamItem = new MemoryStream(array);

// convert to string
StreamReader reader = new StreamReader(streamItem);
string text = reader.ReadToEnd();

22

पिछले समाधान उन मामलों में काम नहीं करेंगे जिनमें एन्कोडिंग शामिल है। यहाँ एक "वास्तविक जीवन" की तरह है - उदाहरण के लिए यह कैसे ठीक से करने के लिए ...

using(var stream = new System.IO.MemoryStream())
{
  var serializer = new DataContractJsonSerializer(typeof(IEnumerable<ExportData>),  new[]{typeof(ExportData)}, Int32.MaxValue, true, null, false);               
  serializer.WriteObject(stream, model);  


  var jsonString = Encoding.Default.GetString((stream.ToArray()));
}

15

इस मामले में, यदि आप वास्तव ReadToEndमें MemoryStreamआसान तरीके से विधि का उपयोग करना चाहते हैं, तो आप इसे प्राप्त करने के लिए इस एक्सटेंशन विधि का उपयोग कर सकते हैं:

public static class SetExtensions
{
    public static string ReadToEnd(this MemoryStream BASE)
    {
        BASE.Position = 0;
        StreamReader R = new StreamReader(BASE);
        return R.ReadToEnd();
    }
}

और आप इस विधि का उपयोग इस प्रकार कर सकते हैं:

using (MemoryStream m = new MemoryStream())
{
    //for example i want to serialize an object into MemoryStream
    //I want to use XmlSeralizer
    XmlSerializer xs = new XmlSerializer(_yourVariable.GetType());
    xs.Serialize(m, _yourVariable);

    //the easy way to use ReadToEnd method in MemoryStream
    MessageBox.Show(m.ReadToEnd());
}

11

यह नमूना दिखाता है कि मेमोरीस्ट्रीम से एक स्ट्रिंग को कैसे पढ़ा जाए, जिसमें मैंने क्रमांकन का उपयोग किया है (DataContractJsonSerializer का उपयोग करके), कुछ सर्वर से क्लाइंट को स्ट्रिंग पास करें, और फिर, पैरामीटर के रूप में पारित स्ट्रिंग से मेमोरीस्ट्रीम को कैसे पुनर्प्राप्त करें, तब , मेमोरीस्ट्रीम को डिसेर्बलाइज करें।

मैंने इस नमूने को करने के लिए विभिन्न पोस्ट के कुछ हिस्सों का उपयोग किया है।

आशा है कि यह मदद करता है।

using System;
using System.Collections.Generic;
using System.IO;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Json;
using System.Threading;

namespace JsonSample
{
    class Program
    {
        static void Main(string[] args)
        {
            var phones = new List<Phone>
            {
                new Phone { Type = PhoneTypes.Home, Number = "28736127" },
                new Phone { Type = PhoneTypes.Movil, Number = "842736487" }
            };
            var p = new Person { Id = 1, Name = "Person 1", BirthDate = DateTime.Now, Phones = phones };

            Console.WriteLine("New object 'Person' in the server side:");
            Console.WriteLine(string.Format("Id: {0}, Name: {1}, Birthday: {2}.", p.Id, p.Name, p.BirthDate.ToShortDateString()));
            Console.WriteLine(string.Format("Phone: {0} {1}", p.Phones[0].Type.ToString(), p.Phones[0].Number));
            Console.WriteLine(string.Format("Phone: {0} {1}", p.Phones[1].Type.ToString(), p.Phones[1].Number));

            Console.Write(Environment.NewLine);
            Thread.Sleep(2000);

            var stream1 = new MemoryStream();
            var ser = new DataContractJsonSerializer(typeof(Person));

            ser.WriteObject(stream1, p);

            stream1.Position = 0;
            StreamReader sr = new StreamReader(stream1);
            Console.Write("JSON form of Person object: ");
            Console.WriteLine(sr.ReadToEnd());

            Console.Write(Environment.NewLine);
            Thread.Sleep(2000);

            var f = GetStringFromMemoryStream(stream1);

            Console.Write(Environment.NewLine);
            Thread.Sleep(2000);

            Console.WriteLine("Passing string parameter from server to client...");

            Console.Write(Environment.NewLine);
            Thread.Sleep(2000);

            var g = GetMemoryStreamFromString(f);
            g.Position = 0;
            var ser2 = new DataContractJsonSerializer(typeof(Person));
            var p2 = (Person)ser2.ReadObject(g);

            Console.Write(Environment.NewLine);
            Thread.Sleep(2000);

            Console.WriteLine("New object 'Person' arrived to the client:");
            Console.WriteLine(string.Format("Id: {0}, Name: {1}, Birthday: {2}.", p2.Id, p2.Name, p2.BirthDate.ToShortDateString()));
            Console.WriteLine(string.Format("Phone: {0} {1}", p2.Phones[0].Type.ToString(), p2.Phones[0].Number));
            Console.WriteLine(string.Format("Phone: {0} {1}", p2.Phones[1].Type.ToString(), p2.Phones[1].Number));

            Console.Read();
        }

        private static MemoryStream GetMemoryStreamFromString(string s)
        {
            var stream = new MemoryStream();
            var sw = new StreamWriter(stream);
            sw.Write(s);
            sw.Flush();
            stream.Position = 0;
            return stream;
        }

        private static string GetStringFromMemoryStream(MemoryStream ms)
        {
            ms.Position = 0;
            using (StreamReader sr = new StreamReader(ms))
            {
                return sr.ReadToEnd();
            }
        }
    }

    [DataContract]
    internal class Person
    {
        [DataMember]
        public int Id { get; set; }
        [DataMember]
        public string Name { get; set; }
        [DataMember]
        public DateTime BirthDate { get; set; }
        [DataMember]
        public List<Phone> Phones { get; set; }
    }

    [DataContract]
    internal class Phone
    {
        [DataMember]
        public PhoneTypes Type { get; set; }
        [DataMember]
        public string Number { get; set; }
    }

    internal enum PhoneTypes
    {
        Home = 1,
        Movil = 2
    }
}

5

ब्रायन के उत्तर का थोड़ा संशोधित संस्करण रीड स्टार्ट के वैकल्पिक प्रबंधन की अनुमति देता है, यह सबसे आसान तरीका लगता है। शायद सबसे कुशल नहीं है, लेकिन समझने और उपयोग करने में आसान है।

Public Function ReadAll(ByVal memStream As MemoryStream, Optional ByVal startPos As Integer = 0) As String
    ' reset the stream or we'll get an empty string returned
    ' remember the position so we can restore it later
    Dim Pos = memStream.Position
    memStream.Position = startPos

    Dim reader As New StreamReader(memStream)
    Dim str = reader.ReadToEnd()

    ' reset the position so that subsequent writes are correct
    memStream.Position = Pos

    Return str
End Function

3
यह वास्तव में ब्रायन के जवाब में कुछ भी नया नहीं जोड़ता है
लुइस फ़िलिप

5

क्यों नहीं MemoryStream प्रकार पर एक अच्छा विस्तार विधि?

public static class MemoryStreamExtensions
{

    static object streamLock = new object();

    public static void WriteLine(this MemoryStream stream, string text, bool flush)
    {
        byte[] bytes = Encoding.UTF8.GetBytes(text + Environment.NewLine);
        lock (streamLock)
        {
            stream.Write(bytes, 0, bytes.Length);
            if (flush)
            {
                stream.Flush();
            }
        }
    }

    public static void WriteLine(this MemoryStream stream, string formatString, bool flush, params string[] strings)
    {
        byte[] bytes = Encoding.UTF8.GetBytes(String.Format(formatString, strings) + Environment.NewLine);
        lock (streamLock)
        {
            stream.Write(bytes, 0, bytes.Length);
            if (flush)
            {
                stream.Flush();
            }
        }
    }

    public static void WriteToConsole(this MemoryStream stream)
    {
        lock (streamLock)
        {
            long temporary = stream.Position;
            stream.Position = 0;
            using (StreamReader reader = new StreamReader(stream, Encoding.UTF8, false, 0x1000, true))
            {
                string text = reader.ReadToEnd();
                if (!String.IsNullOrEmpty(text))
                {
                    Console.WriteLine(text);
                }
            }
            stream.Position = temporary;
        }
    }
}

बेशक, मानक लोगों के साथ संयोजन में इन विधियों का उपयोग करते समय सावधान रहें। :) ... यदि आप समवर्ती के लिए करते हैं, तो आपको उस आसान स्ट्रीम का उपयोग करने की आवश्यकता होगी।


0

मुझे एक वर्ग के साथ एकीकृत करने की आवश्यकता है जिसे उस पर लिखने के लिए स्ट्रीम की आवश्यकता है:

XmlSchema schema;
// ... Use "schema" ...

var ret = "";

using (var ms = new MemoryStream())
{
    schema.Write(ms);
    ret = Encoding.ASCII.GetString(ms.ToArray());
}
//here you can use "ret"
// 6 Lines of code

मैं एक साधारण वर्ग बनाता हूं जो गुणन के उपयोग के लिए कोड की लाइनों को कम करने में मदद कर सकता है:

public static class MemoryStreamStringWrapper
{
    public static string Write(Action<MemoryStream> action)
    {
        var ret = "";
        using (var ms = new MemoryStream())
        {
            action(ms);
            ret = Encoding.ASCII.GetString(ms.ToArray());
        }

        return ret;
    }
}

फिर आप कोड की एक पंक्ति के साथ नमूने को बदल सकते हैं

var ret = MemoryStreamStringWrapper.Write(schema.Write);
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.