GitXplorerGitXplorer
f

recycler

public
33 stars
2 forks
2 issues

Commits

List of commits on branch master.
Unverified
849079f15579b0f820ce1f72acf5bf72f0efb04c

Merge pull request #2 from gereeter/canonical-recyclers

ffrankmcsherry committed 10 years ago
Unverified
30aad5336766d6a7addf0d4df2bdaafd781508bd

Add a way of associating canonical recyclers with types, improving ergonomics for the common case.

ggereeter committed 10 years ago
Unverified
89957ca4ddd9dcfe1b0d222323e1988e9c1723fd

option recycler

ffrankmcsherry committed 10 years ago
Unverified
43998fac40fad585b48a612de1733385c9b14cb4

merge

ffrankmcsherry committed 10 years ago
Unverified
f6632568632f024619b231dd07bde7043d3d682b

Merge branch 'master' of https://github.com/frankmcsherry/recycler

ffrankmcsherry committed 10 years ago
Unverified
dc7cb3ab2dab36fad4dedb8a5826581b33797d63

crates.io

ffrankmcsherry committed 10 years ago

README

The README file for this repository.

recycler

A small Rust library for recycling types with owned memory

Recycler provides the Recycler trait and several implementations. Each Recycler object is capable of "recycling" items of its associated type Item, and using recycled items to "recreate" owned copies of referenced items.

pub trait Recycler : Default {
    type Item;
    fn recycle(&mut self, item: Self::Item);
    fn recreate(&mut self, other: &Self::Item) -> Self::Item;
}

The default TrashRecycler just drops arguments to recycle and clones arguments to recreate. However, smarter recyclers for types with owned memory can deconstruct the item and stash any of its owned memory, and then use the stashed memory to recreate items.

For example, VecRecycler<R> does just this for vectors of recycleable types:

// A recycler for vectors and their contents
pub struct VecRecycler<R: Recycler> {
    pub recycler: R,
    stash: Vec<Vec<R::Item>>,
}

impl<R: Recycler> Recycler for VecRecycler<R> {
    type Item = Vec<R::Item>;
    // recycles vec contents and then stashes the vec
    fn recycle(&mut self, mut vec: Vec<R::Item>) {
        while let Some(x) = vec.pop() {
            self.recycler.recycle(x)
        }
        self.stash.push(vec);
    }
    // pops a stashed vector and then recreates each element
    fn recreate(&mut self, other: &Vec<R::Item>) -> Vec<R::Item> {
        let mut vec = self.stash.pop().unwrap_or(Vec::new());
        for elem in other.iter() {
            vec.push(self.recycler.recreate(elem));
        }
        vec
    }
}

While recycling might sound great just because of civic duty, the real purpose is that these recyclers are able to return the owned memory to you, using a pattern not unlike standard allocation. Where you might write something like

#[bench]
fn allocate_vec_vec_str(bencher: &mut Bencher) {
    bencher.iter(|| {
        let mut v1 = Vec::with_capacity(10);
        for _ in 0..10 {
            let mut v2 = Vec::with_capacity(10);
            for _ in 0..10 {
                v2.push(("test!").to_owned());
            }
            v1.push(v2);
        }
        v1
    });
}

you can now instead write something pretty similar (no, not the same):

#[bench]
fn recycler_vec_vec_str(bencher: &mut Bencher) {
    let mut r1 = make_recycler::<Vec<Vec<String>>>();
    bencher.iter(|| {
        let v = { // scope the borrow of r1
            let (mut v1, r2) = r1.new();
            for _ in 0..10 {
                let (mut v2, r3) = r2.new();
                for _ in 0..10 {
                    v2.push(r3.new_from("test!"));
                }
                v1.push(v2);
            }
            v1
        };
        r1.recycle(v);
    });
}

The reason you do this is because if you run those benchmarks up there, you see numbers like:

test allocate_vec_vec_str ... bench:       3,494 ns/iter (+/- 1,128)
test recycler_vec_vec_str ... bench:       1,709 ns/iter (+/- 643)

If you do less formatting stuff and just put some u64 data in the vectors, you see similar distinction:

test allocate_vec_vec_u64 ... bench:         267 ns/iter (+/- 49)
test recycler_vec_vec_u64 ... bench:         145 ns/iter (+/- 26)

The main down side is that you may get vectors that may have more memory than you need, and memory may also live for quite a while in the recycler. I almost added a clear method, but if you want to do that just make a new recycler and clobber the old one.

Note: a previous version of these numbers looked worse for the allocate variants because they used Vec::new() rather than Vec::with_capacity(10), which correctly sizes the allocation and avoids copies.

recreate

If for some reason you find you are often given references to objects and need a quick clone (for example, using decode or verify in Abomonation), the recreate method is meant to be painless. The above benchmark becomes:

#[bench]
fn recreate_vec_vec_str(bencher: &mut Bencher) {
    let mut recycler = make_recycler::<Vec<Vec<String>>>();
    let data = vec![vec!["test!".to_owned(); 10]; 10];
    bencher.iter(|| {
        let record = recycler.recreate(&data);
        recycler.recycle(record);
    });
}

If you compare using recreate with just using clone, you see numbers like:

test clone_vec_vec_str    ... bench:       2,906 ns/iter (+/- 774)
test recreate_vec_vec_str ... bench:       1,773 ns/iter (+/- 625)

test clone_vec_vec_u64    ... bench:         344 ns/iter (+/- 134)
test recreate_vec_vec_u64 ... bench:         157 ns/iter (+/- 42)

thanks!

If anyone has any hot tips or recommendations, especially about a macro or syntax extension that would let structs and such automatically derive recyclers, I'd be all ears. Any other friendly comments or contributions are also welcome.