3 releases
0.1.2 | Jul 8, 2022 |
---|---|
0.1.1 | Apr 12, 2022 |
0.1.0 | Apr 2, 2022 |
#1295 in Data structures
17KB
307 lines
What
This is a sliding window data structure. Inside it there is a vector that has
at position 0
the newest element and at position capacity
the oldest.
Whenever you push an item, said item is stored at position 0
, everything
shifts, and the last position is forgotten.
For some speed gain, this is not how it internally works. Internally, it has an index that moves, always pointing to the oldest element in the set. This way insertions are O(1).
You can get the information out in three ways, by getting an iterator, by
getting a Vec<T>
, or by directly indexing
Why
The following crates are fairly similar:
Queues has the type CircularBuffer. It has several problems. The first one is that it has O(n) insertions. The way less worrying one is that it only get the newest item back. On the other hand, it expresses the operations of an abstract queue as a trait, which is good. It has no testing. This is most probably not what you were looking for when you searched for a sliding window.
circular_queue is pretty much the same as this crate, except that this crate doesn't provide several possible orders. On the other hand this create provides more ways of creating and pushing items, more comparisons, and a vector return. It doesn't allow indexing, which this crate does, mostly inspired by...
sliding_window has a similar
functionality as circular_queue
, but is no_std
and has some unsafe code. It
allows indexing where 0
is the oldest. This is probably the crate to go for
speed and portability.
However, I have published this crate. My reasons are:
-
It is always filled from the creation, always returning iterators and vectors of the same size. This is specially useful for some mathematical manipulations. This would be a breaking change for the other crates, and is the reason there is a new crate instead of a contribution to those crates.
-
It adds on to
circular_queue
indexing, vector return, and a bunch ofPartialEqs
implementations. -
It is 100% safe.
-
Has no dependencies and is fast to build, (it's a small target, where actual LOC are about 130).
How
There are several examples in the examples
folder. Here is some code to get
a feel for the library.
use sliding_window_alt::SlidingWindow;
fn main() {
// Stores the useful information for a model of a system. In this case the
// latest 5 outputs of the system
let mut sys = SlidingWindow::new(5, 0.0);
// caracteristical polynomial of the system, it's stable
let carac_pol = [0.5, -0.4, 0.2, -0.3, 0.05];
for _ in 0..=100 {
sys.push(
sys.iter()
.zip(carac_pol)
.map(|(item, coef)| coef * *item)
.sum::<f64>() // Multiplies the polinomial
+ 1.0, // the action input, in this case a step
);
println!("{}", sys[0]);
}
}
Declaring
There are three ways of creating a sliding window. One with new, and two froms.
You can create from an array, (not Vec
!!) or a slice.
Converting
Both &SlidingWindow
and SlidingWindow
into_iter
methods allow for use in
for loops, however, both of these methods are O(n).
Pushing
The push
method pushes one item and the push_slice
an array of items, being
the item at position 0
the "youngest" of the items.
Iterating
The methods iter
and iter_mut
provide iterators that start at the newest
item.
Indexing
You can access the contents of your sliding window by indexing, it. 0
is the
newest element and capacity
is the oldest.
Vector
If you need to perform some analysis for your application, you can obtain a
vector from the to_vec
method. This operation is O(n), use it when you
genuinely need a vector.
Benchmarks
There are some benchmarks for the code and a comparison with the alternative crates presented earlier.
Criteria | Wins | Loses |
---|---|---|
Creation | Queues/Sliding Windows (?[^1]) | this crate |
Insertion | this crate | Queues[^2] |
Iteration | this crate/ circular_queue | sliding_window |
Full workflow (initialization not required) | sliding_window | circular_queue |
Full workflow (initialization required) | this crate | sliding_window |
My recommendation is that if you don't require a fixed length, use sliding_window, else, use this one. CircularBuffer from the Queues crate is unlikely to optimally solve your problems.
[^1]: Unknown, because I'm unable to blackbox the amount of items on creation. [^2]: Queues does not offer iterators nor ways to access all the data in the queue and therefore can't compete in the other categories.