Skip to content

Commit 9338878

Browse files
committed
perf(headers): use a Vec instead of HashMap internally
1 parent 8017dac commit 9338878

File tree

3 files changed

+99
-7
lines changed

3 files changed

+99
-7
lines changed

src/header/internals/mod.rs

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
11
pub use self::item::Item;
2+
pub use self::vec_map::{VecMap, Entry};
23

34
mod cell;
45
mod item;
6+
mod vec_map;

src/header/internals/vec_map.rs

Lines changed: 89 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
#[derive(Clone)]
2+
pub struct VecMap<K, V> {
3+
vec: Vec<(K, V)>,
4+
}
5+
6+
impl<K: PartialEq, V> VecMap<K, V> {
7+
pub fn new() -> VecMap<K, V> {
8+
VecMap {
9+
vec: Vec::new()
10+
}
11+
}
12+
13+
pub fn insert(&mut self, key: K, value: V) {
14+
match self.find(&key) {
15+
Some(pos) => self.vec[pos] = (key, value),
16+
None => self.vec.push((key, value))
17+
}
18+
}
19+
20+
pub fn entry(&mut self, key: K) -> Entry<K, V> {
21+
match self.find(&key) {
22+
Some(pos) => Entry::Occupied(OccupiedEntry {
23+
vec: self,
24+
pos: pos,
25+
}),
26+
None => Entry::Vacant(VacantEntry {
27+
vec: self,
28+
key: key,
29+
})
30+
}
31+
}
32+
33+
pub fn get(&self, key: &K) -> Option<&V> {
34+
self.find(key).map(move |pos| &self.vec[pos].1)
35+
}
36+
37+
pub fn get_mut(&mut self, key: &K) -> Option<&mut V> {
38+
self.find(key).map(move |pos| &mut self.vec[pos].1)
39+
}
40+
41+
pub fn contains_key(&self, key: &K) -> bool {
42+
self.find(key).is_some()
43+
}
44+
45+
pub fn len(&self) -> usize { self.vec.len() }
46+
pub fn iter(&self) -> ::std::slice::Iter<(K, V)> {
47+
self.vec.iter()
48+
}
49+
pub fn remove(&mut self, key: &K) -> Option<V> {
50+
self.find(key).map(|pos| self.vec.remove(pos)).map(|(_, v)| v)
51+
}
52+
pub fn clear(&mut self) {
53+
self.vec.clear();
54+
}
55+
56+
fn find(&self, key: &K) -> Option<usize> {
57+
self.vec.iter().position(|entry| key == &entry.0)
58+
}
59+
}
60+
61+
pub enum Entry<'a, K: 'a, V: 'a> {
62+
Vacant(VacantEntry<'a, K, V>),
63+
Occupied(OccupiedEntry<'a, K, V>)
64+
}
65+
66+
pub struct VacantEntry<'a, K: 'a, V: 'a> {
67+
vec: &'a mut VecMap<K, V>,
68+
key: K,
69+
}
70+
71+
impl<'a, K, V> VacantEntry<'a, K, V> {
72+
pub fn insert(self, val: V) -> &'a mut V {
73+
let mut vec = self.vec;
74+
vec.vec.push((self.key, val));
75+
let pos = vec.vec.len() - 1;
76+
&mut vec.vec[pos].1
77+
}
78+
}
79+
80+
pub struct OccupiedEntry<'a, K: 'a, V: 'a> {
81+
vec: &'a mut VecMap<K, V>,
82+
pos: usize,
83+
}
84+
85+
impl<'a, K, V> OccupiedEntry<'a, K, V> {
86+
pub fn into_mut(self) -> &'a mut V {
87+
&mut self.vec.vec[self.pos].1
88+
}
89+
}

src/header/mod.rs

Lines changed: 8 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -77,8 +77,8 @@
7777
//! ```
7878
use std::any::Any;
7979
use std::borrow::{Cow, ToOwned};
80-
use std::collections::HashMap;
81-
use std::collections::hash_map::{Iter, Entry};
80+
//use std::collections::HashMap;
81+
//use std::collections::hash_map::{Iter, Entry};
8282
use std::iter::{FromIterator, IntoIterator};
8383
use std::ops::{Deref, DerefMut};
8484
use std::{mem, fmt};
@@ -87,7 +87,7 @@ use {httparse, traitobject};
8787
use typeable::Typeable;
8888
use unicase::UniCase;
8989

90-
use self::internals::Item;
90+
use self::internals::{Item, VecMap, Entry};
9191

9292
#[cfg(feature = "serde-serialization")]
9393
use serde::{Deserialize, Deserializer, Serialize, Serializer};
@@ -169,7 +169,8 @@ fn header_name<T: Header>() -> &'static str {
169169
/// A map of header fields on requests and responses.
170170
#[derive(Clone)]
171171
pub struct Headers {
172-
data: HashMap<HeaderName, Item>
172+
//data: HashMap<HeaderName, Item>
173+
data: VecMap<HeaderName, Item>,
173174
}
174175

175176
impl Default for Headers {
@@ -183,7 +184,7 @@ impl Headers {
183184
/// Creates a new, empty headers map.
184185
pub fn new() -> Headers {
185186
Headers {
186-
data: HashMap::new()
187+
data: VecMap::new()
187188
}
188189
}
189190

@@ -377,14 +378,14 @@ impl Deserialize for Headers {
377378
/// An `Iterator` over the fields in a `Headers` map.
378379
#[allow(missing_debug_implementations)]
379380
pub struct HeadersItems<'a> {
380-
inner: Iter<'a, HeaderName, Item>
381+
inner: ::std::slice::Iter<'a, (HeaderName, Item)>
381382
}
382383

383384
impl<'a> Iterator for HeadersItems<'a> {
384385
type Item = HeaderView<'a>;
385386

386387
fn next(&mut self) -> Option<HeaderView<'a>> {
387-
self.inner.next().map(|(k, v)| HeaderView(k, v))
388+
self.inner.next().map(|&(ref k, ref v)| HeaderView(k, v))
388389
}
389390
}
390391

0 commit comments

Comments
 (0)