flexstr

Crate Docs

A flexible, simple to use, immutable, clone-efficient String replacement for Rust. It unifies literals, inlined, and heap allocated strings in a single type.

Overview

Rust is great, but it's String type is optimized as a mutable string buffer, not for typical string use cases. Most string use cases don't modify their string contents, often need to copy strings around as if they were cheap like integers, typically concatenate instead of modify, and often end up being cloned with identical contents. Additionally, String isn't able to wrap a string literal without additional allocation and copying. Rust needs a new string type to unify usage of both literals and allocated strings in typical use cases. This crate creates a new string type that is optimized for those use cases, while retaining the usage simplicity of String.

This type is not inherently "better" than String, but different. It is a higher level type, that can at times mean higher overhead. It really depends on the use case.

Examples

toml [dependencies] flexstr = "0.4"

```rust use flexstr::{flex_fmt, FlexStr, IntoFlexStr, ToCase, ToFlexStr};

fn main() { // Use an into function to simply wrap a string literal, no allocation or copying let staticstr = "This will not allocate or copy".intoflexstr(); assert!(staticstr.is_static());

// Strings up to 22 bytes (on 64-bit) will be inlined automatically (demo only, use into for literals as above) let inlinestr = "inlined".toflexstr(); assert!(inlinestr.is_inlined());

// When a string can't be wrapped or inlined, it wil fall back to heap allocation let rcstr = "This is too long to be inlined. It will be wrapped in Rc".toflexstr(); assert!(rcstr.is_heap());

// You can efficiently create new FlexStrs without ever creating a String // This is equivalent to stdlib format! macro let inlinestr2 = flexfmt!("in{}", "lined"); assert!(inlinestr2.isinlined()); asserteq!(inlinestr, inline_str2);

// We can even upper/lower strings without ever using String - the below doesn't allocate let inlinestr3: FlexStr = "INLINED".toasciilower(); assert!(inlinestr3.isinlined()); asserteq!(inlinestr, inlinestr3);

// Clone is almost free, even when borrowed (at most it is a ref count increment for heap allocated strings) let staticstr2 = (&staticstr).clone(); assert!(staticstr2.isstatic());

// Regardless of storage type, these all operate seamlessly together and choose storage as required let heapstr2 = staticstr2 + &inlinestr; assert!(heapstr2.isheap()); asserteq!(heap_str2, "This will not allocate or copyinlined"); } ```

How Does It Work?

Internally, FlexStr uses an enum with these variants:

The type automatically chooses the best storage and allows you to use them interchangeably as a single string type.

Features

Types

Usage

Hello World

```rust use flexstr::IntoFlexStr;

fn main() { // From literal - no copying or allocation let world = "world!".intoflexstr();

println!("Hello {world}"); } ```

Conversions

```rust use flexstr::{IntoAFlexStr, IntoFlexStr, ToFlexStr};

fn main() { // From literal - no copying or allocation // NOTE: to_flex_str will copy, so use into_flex_str for literals let literal = "literal".intoaflex_str();

// From borrowed string - Copied into inline string let owned = "inlined".tostring(); let strtoinlined = (&owned).toflex_str();

// From borrowed String - copied into str wrapped in Rc let owned = "A bit too long to be inlined!!!".tostring(); let strtowrapped = (&owned).toflex_str();

// From String - copied into inline string (String storage released) let inlined = "inlined".tostring().intoaflexstr();

// From String - str wrapped in Rc (String storage released) let counted = "A bit too long to be inlined!!!".tostring().intoaflexstr();

// * If you want a Send/Sync type you need AFlexStr instead *

// From FlexStr wrapped literal - no copying or allocation let literal = literal.intoaflex_str();

// From FlexStr inlined string - no allocation let inlined = inlined.intoaflex_str();

// From FlexStr Rc wrapped str - copies into str wrapped in Arc let counted = counted.intoaflex_str(); } ```

Passing FlexStr to Conditional Ownership Functions

This has always been a confusing situation in Rust, but it is easy with FlexStr since multi ownership is cheap.

```rust use flexstr::{IntoFlexStr, FlexStr};

struct MyStruct { s: FlexStr }

impl MyStruct { fn toownornottoown(s: &FlexStr) -> Self { let s = if s == "ownme" { // Since a wrapped literal, no copy or allocation s.clone() } else { // Wrapped literal - no copy or allocation "own_me".into() };

Self { s }

} }

fn main() { // Wrapped literals - no copy or allocation let s = "borrow me".intoaflexstr(); let s2 = "own me".intoaflexstr();

let struct1 = MyStruct::toownornottoown(&s); let struct2 = MyStruct::toownornottoown(&s2);

asserteq!(s2, struct1.s); asserteq!(s2, struct2.s); } ```

Performance Characteristics

NOTE: No benchmarking has yet been done

Negatives

There is no free lunch:

Status

This is currently beta quality and still needs testing. The API may very possibly change but semantic versioning will be followed.

License

This project is licensed optionally under either: