Skip to content

Commit

Permalink
refactor/file-extensions-writer (faster & less allocation) (#98)
Browse files Browse the repository at this point in the history
  • Loading branch information
leandromoh authored Dec 23, 2023
1 parent 351517e commit 8213cc2
Show file tree
Hide file tree
Showing 2 changed files with 46 additions and 49 deletions.
24 changes: 24 additions & 0 deletions RecordParser/Extensions/EnumerableExtensions.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
using System.Collections.Generic;

namespace RecordParser.Extensions
{
internal static class EnumerableExtensions
{
public static IEnumerable<IEnumerable<T>> Batch<T>(this IEnumerable<T> source, int batchSize)
{
using var e = source.GetEnumerator();
var hasMore = true;
bool MoveNext() => hasMore && (hasMore = e.MoveNext());

while (MoveNext())
yield return Chunk(batchSize);

IEnumerable<T> Chunk(int countdown)
{
do
yield return e.Current;
while (--countdown > 0 && MoveNext());
}
}
}
}
71 changes: 22 additions & 49 deletions RecordParser/Extensions/FileWriter/WriterExtensions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -55,63 +55,36 @@ public static void WriteRecords<T>(this TextWriter textWriter, IEnumerable<T> it
}
}

private class BufferContext
{
public int pow;
public char[] buffer;
public object lockObj;
}

private static void WriteParallel<T>(TextWriter textWriter, IEnumerable<T> items, TryFormat<T> tryFormat, ParallelismOptions options)
{
var initialPool = 20;
var pool = new Stack<char[]>(initialPool);
var poolSize = 10_000;
var pool = new char[poolSize][];

for (var index = 0; index < initialPool; index++)
pool.Push(ArrayPool<char>.Shared.Rent((int)Math.Pow(2, initialPow)));
for (var index = 0; index < poolSize; index++)
pool[index] = new char[(int)Math.Pow(2, initialPow)];

var xs = items.AsParallel(options).Select((item, i) =>
foreach (var xx in items.Batch(poolSize))
{
var buffer = Pop() ?? ArrayPool<char>.Shared.Rent((int)Math.Pow(2, initialPow));
retry:

if (tryFormat(item, buffer, out var charsWritten))
var xs = xx.AsParallel(options).Select((item, i) =>
{
return (buffer, charsWritten);
}
else
{
ArrayPool<char>.Shared.Return(buffer);
buffer = ArrayPool<char>.Shared.Rent(buffer.Length * 2);
goto retry;
}
});

foreach (var x in xs)
{
textWriter.WriteLine(x.buffer, 0, x.charsWritten);
Push(x.buffer);
}

foreach (var x in pool)
{
ArrayPool<char>.Shared.Return(x);
}

pool.Clear();
var buffer = pool[i];
retry:

char[] Pop()
{
char[] x;
lock (pool)
pool.TryPop(out x);
return x;
}
if (tryFormat(item, buffer, out var charsWritten))
{
return (buffer, charsWritten);
}
else
{
buffer = pool[i] = new char[buffer.Length * 2];
goto retry;
}
});

void Push(char[] item)
{
lock (pool)
pool.Push(item);
foreach (var x in xs)
{
textWriter.WriteLine(x.buffer, 0, x.charsWritten);
}
}
}

Expand Down

0 comments on commit 8213cc2

Please sign in to comment.