Je suis arrivé à cette question vous cherchez un moyen de diffuser une fin ouverte de la liste des objets sur un System.IO.Stream
et de les lire à l'autre extrémité, sans mise en mémoire tampon de l'ensemble de la liste avant de l'envoyer. (Plus précisément, je streaming persisté objets de MongoDB sur les Web API.)
@Paul Tyng et @Rivières ont fait un excellent travail de répondre à la question d'origine, et j'ai utilisé leurs réponses à construire une preuve de concept pour mon problème. J'ai décidé de poster mon test de la console app ici au cas où quelqu'un d'autre est face à la même question.
using System;
using System.Diagnostics;
using System.IO;
using System.IO.Pipes;
using System.Threading;
using System.Threading.Tasks;
using Newtonsoft.Json;
namespace TestJsonStream {
class Program {
static void Main(string[] args) {
using(var writeStream = new AnonymousPipeServerStream(PipeDirection.Out, HandleInheritability.None)) {
string pipeHandle = writeStream.GetClientHandleAsString();
var writeTask = Task.Run(() => {
using(var sw = new StreamWriter(writeStream))
using(var writer = new JsonTextWriter(sw)) {
var ser = new JsonSerializer();
writer.WriteStartArray();
for(int i = 0; i < 25; i++) {
ser.Serialize(writer, new DataItem { Item = i });
writer.Flush();
Thread.Sleep(500);
}
writer.WriteEnd();
writer.Flush();
}
});
var readTask = Task.Run(() => {
var sw = new Stopwatch();
sw.Start();
using(var readStream = new AnonymousPipeClientStream(pipeHandle))
using(var sr = new StreamReader(readStream))
using(var reader = new JsonTextReader(sr)) {
var ser = new JsonSerializer();
if(!reader.Read() || reader.TokenType != JsonToken.StartArray) {
throw new Exception("Expected start of array");
}
while(reader.Read()) {
if(reader.TokenType == JsonToken.EndArray) break;
var item = ser.Deserialize<DataItem>(reader);
Console.WriteLine("[{0}] Received item: {1}", sw.Elapsed, item);
}
}
});
Task.WaitAll(writeTask, readTask);
writeStream.DisposeLocalCopyOfClientHandle();
}
}
class DataItem {
public int Item { get; set; }
public override string ToString() {
return string.Format("{{ Item = {0} }}", Item);
}
}
}
}
Notez que vous pouvez recevoir une exception lorsque l' AnonymousPipeServerStream
est éliminé, j'ai ignoré ce qu'il n'est pas pertinent pour le problème à portée de main.