using System.Collections.Generic;
using System.Xml.Serialization;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Formatters.Binary;
using System.IO.Compression;
using System.Threading.Tasks;
public sealed class CharMemoryReader : TextReader
private ReadOnlyMemory<char> chars;
public CharMemoryReader(ReadOnlyMemory<char> chars)
throw new ObjectDisposedException(null, string.Format("{0} is closed.", ToString()));
public override void Close() => Dispose(true);
protected override void Dispose(bool disposing)
chars = ReadOnlyMemory<char>.Empty;
public override int Peek()
return position >= chars.Length ? -1 : chars.Span[position];
public override int Read()
return position >= chars.Length ? -1 : chars.Span[position++];
public override int Read(char[] buffer, int index, int count)
throw new ArgumentNullException(nameof(buffer));
throw new ArgumentOutOfRangeException(nameof(index));
throw new ArgumentOutOfRangeException(nameof(count));
if (buffer.Length - index < count)
throw new ArgumentException("buffer.Length - index < count");
return Read(buffer.AsSpan().Slice(index, count));
public override int Read(Span<char> buffer)
var nRead = chars.Length - position;
if (nRead > buffer.Length)
chars.Span.Slice(position, nRead).CopyTo(buffer);
public override string ReadToEnd()
var s = position == 0 ? chars.ToString() : chars.Slice(position, chars.Length - position).ToString();
public override string ReadLine()
for( ; i < span.Length; i++)
if (ch == '\r' || ch == '\n')
var result = span.Slice(position, i - position).ToString();
if (ch == '\r' && position < span.Length && span[position] == '\n')
var result = span.Slice(position, i - position).ToString();
public override int ReadBlock(char[] buffer, int index, int count) => Read(buffer, index, count);
public override int ReadBlock(Span<char> buffer) => Read(buffer);
public override Task<String> ReadLineAsync() => Task.FromResult(ReadLine());
public override Task<String> ReadToEndAsync() => Task.FromResult(ReadToEnd());
public override Task<int> ReadBlockAsync(char[] buffer, int index, int count) => Task.FromResult(ReadBlock(buffer, index, count));
public override Task<int> ReadAsync(char[] buffer, int index, int count) => Task.FromResult(Read(buffer, index, count));
public override ValueTask<int> ReadBlockAsync(Memory<char> buffer, CancellationToken cancellationToken = default) =>
cancellationToken.IsCancellationRequested ? new ValueTask<int>(Task.FromCanceled<int>(cancellationToken)) : new ValueTask<int>(ReadBlock(buffer.Span));
public override ValueTask<int> ReadAsync(Memory<char> buffer, CancellationToken cancellationToken = default) =>
cancellationToken.IsCancellationRequested ? new ValueTask<int>(Task.FromCanceled<int>(cancellationToken)) : new ValueTask<int>(Read(buffer.Span));
public List<string> Strings { get; } = new List<string>();
public static void Test()
Test("ぁあぃいぅうぇえぉおかがきぎく", "\n\nぁあぃいぅうぇえぉおかがきぎく\U00029E3D\u0302");
BasicTests("\n\r\n\n\r\nぁあぃいぅうぇえぉおかがきぎく\U00029E3D\u0302");
BasicTests("ぁあぃいぅうぇえぉおかがきぎく\U00029E3D\u0302");
public static void Test(params string [] strings)
var root = new MyEntity();
root.Strings.AddRange(strings);
var _xmlSerializer = new XmlSerializer(typeof(MyEntity));
var xml = root.GetXml(_xmlSerializer);
var buffer = xml.ToArray();
int contentLength = buffer.Length;
var result = buffer.LoadFromXml<MyEntity>(contentLength, _xmlSerializer);
Console.WriteLine("Re-serialized {0}:", result);
Console.WriteLine(result.GetXml());
Assert.IsTrue(result.Strings.SequenceEqual(root.Strings));
static void BasicTests(string s)
using (var reader = new CharMemoryReader(array.AsMemory()))
Assert.AreEqual(s, reader.ReadToEnd(), "s");
using (var reader = new CharMemoryReader(s.AsMemory()))
Assert.IsTrue(object.ReferenceEquals(s, reader.ReadToEnd()));
using (var reader = new CharMemoryReader(array.AsMemory()))
using (var stringReader = new StringReader(s))
Assert.IsTrue(reader.ReadAllLines().SequenceEqual(stringReader.ReadAllLines()));
using (var reader = new CharMemoryReader(array.AsMemory()))
using (var stringReader = new StringReader(s))
Assert.IsTrue(reader.ReadAllChars().SequenceEqual(stringReader.ReadAllChars()));
using (var reader = new CharMemoryReader(array.AsMemory()))
var buffer = new Span<char>(new char [1000]);
while ((n = reader.Read(buffer)) > 0)
Assert.IsTrue(s.AsSpan(i, n).Equals(buffer.Slice(0, n), StringComparison.Ordinal));
static class TextReaderExtensions
public static IEnumerable<string> ReadAllLines(this TextReader reader)
while ((l = reader.ReadLine()) != null)
public static IEnumerable<char> ReadAllChars(this TextReader reader)
while ((ch = reader.Read()) >= 0)
public static partial class XmlSerializationHelper
public static T LoadFromXml<T>(this char [] xml, int contentLength, XmlSerializer serial = null) =>
new ReadOnlyMemory<char>(xml, 0, contentLength).LoadFromXml<T>(serial);
public static T LoadFromXml<T>(this ReadOnlyMemory<char> xml, XmlSerializer serial = null)
serial = serial ?? new XmlSerializer(typeof(T));
using (var reader = new CharMemoryReader(xml))
return (T)serial.Deserialize(reader);
public static partial class XmlSerializationHelper
public static T LoadFromXml<T>(this string xmlString, XmlSerializer serial = null)
serial = serial ?? new XmlSerializer(typeof(T));
using (var reader = new StringReader(xmlString))
return (T)serial.Deserialize(reader);
public static string GetXml<T>(this T obj, XmlSerializer serializer = null, bool omitStandardNamespaces = false)
XmlSerializerNamespaces ns = null;
if (omitStandardNamespaces)
ns = new XmlSerializerNamespaces();
using (var textWriter = new StringWriter())
var settings = new XmlWriterSettings() { Indent = true };
using (var xmlWriter = XmlWriter.Create(textWriter, settings))
(serializer ?? new XmlSerializer(obj.GetType())).Serialize(xmlWriter, obj, ns);
return textWriter.ToString();
public static void Main()
Console.WriteLine("Environment version: {0} ({1})", System.Runtime.InteropServices.RuntimeInformation.FrameworkDescription , GetNetCoreVersion());
Console.WriteLine("Failed with unhandled exception: ");
public static string GetNetCoreVersion()
var assembly = typeof(System.Runtime.GCSettings).GetTypeInfo().Assembly;
var assemblyPath = assembly.CodeBase.Split(new[] { '/', '\\' }, StringSplitOptions.RemoveEmptyEntries);
int netCoreAppIndex = Array.IndexOf(assemblyPath, "Microsoft.NETCore.App");
if (netCoreAppIndex > 0 && netCoreAppIndex < assemblyPath.Length - 2)
return assemblyPath[netCoreAppIndex + 1];