U8String 0.12.0-alpha

This is a prerelease version of U8String.
There is a newer prerelease version of this package available.
See the version list below for details.
dotnet add package U8String --version 0.12.0-alpha                
NuGet\Install-Package U8String -Version 0.12.0-alpha                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="U8String" Version="0.12.0-alpha" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add U8String --version 0.12.0-alpha                
#r "nuget: U8String, 0.12.0-alpha"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install U8String as a Cake Addin
#addin nuget:?package=U8String&version=0.12.0-alpha&prerelease

// Install U8String as a Cake Tool
#tool nuget:?package=U8String&version=0.12.0-alpha&prerelease                

U8String nuget

[work-in-progress] Highly functional and performant UTF-8 string primitive for C# and .NET.

This library adopts the lessons learned from .NET 5's Utf8String prototype, Rust and Go string implementations to provide first-class UTF-8 string primitive which has been historically missing in .NET.

It is a ground-up reimplementation of the string type with UTF-8 semantics, and is designed to be a drop-in replacement for string in scenarios where UTF-8 is the preferred encoding.

Features

  • Zero-allocation slicing
  • Highly optimized and SIMD-accelerated where applicable
  • byte, char and Rune overload variants for maximum flexibility
  • Performant UTF-8 formatting inspired by CoreLib and Yoshifumi Kawai implementations
  • Convenient .Runes, .Lines, .Chars, SplitFirst/Last(...) and .Split(...) projections
  • Opt-in string interning for repeated conversions from and to UTF-16
  • Easy integration with .NET type system thanks to IUtf8SpanFormattable and IUtf8SpanParsable<T> added in .NET 8

Target Scenarios

  • Zero-copy and/or zero-allocation parsing
  • Interop with native libraries that use UTF-8
  • Directly consuming UTF-8 byte sequences
  • Canonical representation of text primitives for serialization and storage which use UTF-8 (e.g. DB drivers)
  • Storing large amounts of ASCII-like text on the heap which is twice as compact as UTF-16

Quick Start

dotnet add package U8String --prerelease

Walkthrough

// From u8 string literal
var greeting = (U8String)"Hello, World!"u8;

// From UTF-16 string
var converted = (U8String)"Hello, World!";

// From a primitive
var num = 42.ToU8String();

// From file
using var file = File.OpenHandle("file.txt");
var text = U8String.Read(file);

// From HttpClient
using var http = new HttpClient();
var example = await http.GetU8StringAsync("http://example.org/");

// Shorthand syntax
// See U8String.Optimization to fold these at compile time
using static U8.Extensions.Syntax;

var literal = u8("Lorem ipsum dolor");
var number = u8(42);

// From an immutable byte array
var array = ImmutableArray.Create("Привіт, Всесвіт!"u8);
// Does not allocate relying on ImmutableArray semantics
var cyrillic = (U8String)array;

// Other forms: U8String.Create(...), U8String.Create(T, format), new U8String(...)

// Slice (substring)
// Prints "World", does not allocate
var slice = greeting[7..^1];

// Equality (works with ROS<byte>, U8String and byte[])
if (hello == "Hello"u8)
{
    // ...
}

// Split on first occurrence
var (hello, world) = greeting.SplitFirst(", "u8);

// Get an n-th element from a split, prints "1E" and does not allocate
var element = joined.Split(':').ElementAt(3);

// Iterate over lines
foreach (var line in text.Lines)
{
    // ...
}

// Concatenate two strings
var greeting1 = hello + ", "u8;
// Either of these works
var greeting2 = U8String.Concat(greeting1, world);

// Concatenate multiple strings
var concatenated = U8String.Concat([hello, world, greeting2]);

// Join multiple values (for IUtf8SpanFormattable types)
// Prints "00:0A:14:1E"
var joined = U8String.Join(':', [0, 10, 20, 30], "X2");

// From an interpolated string, Roslyn unrolls this into a special builder pattern
// which writes the data directly to UTF-8 buffer
var formatted = new U8String($"Today is {DateTime.Now:yyyy-MM-dd}.");

Evaluation

As this project is still in development, it is not recommended for deployment in production.
It is, however, in sufficiently advanced state to be used for evaluation and testing purposes.
If you are interested in using this library in your project or have any questions or suggestions, please feel free to reach out by opening an issue or contacting me directly.

Performance

Simple

TBD

Advanced

See https://github.com/neon-sunset/warpskimmer and Twitch IRC parsing comparison

The implementation demonstrates how simple it is to achieve almost maximum hardware utilization with this library. Keep in mind that performance takes a hit due to the use of WSL2 and differences in Linux ABI which influences register allocation, on Windows, it takes about 150-160ns per message, taking the first place.

Historically, a lot of Golang implementations in various benchmarks used to have an advantage due to non-copying slicing and the fact that Go's strings do not validate whether the slices are correct. With U8String (even though it ensures slice correctness), the significant advantage of .NET's compiler and standard library over Golang's can be observed.

In many scenarios, this will even outperform Rust's &str and String types due to much more conservative vectorization of Rust's standard library and shortcomings of its select internal string abstractions (which require additional manual work to achieve comparable performance).

Product Compatible and additional computed target framework versions.
.NET net8.0 is compatible.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
0.16.0-alpha 173 4/2/2024
0.15.5-alpha 63 3/31/2024
0.15.0-alpha 57 3/31/2024
0.14.0-alpha.1 62 3/26/2024
0.13.0-alpha 58 3/14/2024
0.12.0-alpha 71 3/3/2024
0.11.10-alpha 69 1/30/2024
0.11.5-alpha 57 1/24/2024
0.11.0-alpha 56 1/22/2024
0.10.0-alpha 83 12/27/2023
0.9.0-alpha 92 12/10/2023
0.8.5-alpha 115 10/20/2023
0.8.0-alpha.1 75 10/18/2023
0.6.0-alpha 77 9/28/2023
0.5.0-alpha 86 9/7/2023
0.2.1-alpha 109 7/12/2023
0.2.0-alpha 97 7/11/2023