나는이 질문에 도달하여 열린 끝의 객체 목록을 스트리밍하는 방법을 찾고 System.IO.Stream
보내기 전에 전체 목록을 버퍼링하지 않고 하여 다른 쪽 끝에서 읽을 수 . (특히 MongoDB에서 웹 API를 통해 지속 객체를 스트리밍하고 있습니다.)
@Paul Tyng과 @Rivers는 원래 질문에 대답하는 데 훌륭한 역할을했으며 내 대답에 대한 개념 증명을 작성하기 위해 답을 사용했습니다. 다른 사람이 같은 문제에 직면 한 경우 여기에 테스트 콘솔 앱을 게시하기로 결정했습니다.
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);
}
}
}
}
AnonymousPipeServerStream
이 처리 될 때 예외가 발생할 수 있습니다 . 현재 문제와 관련이 없으므로 무시했습니다.