astl
Version:
AssemblyScript-STL (Standard Template Library, migrated from the C++)
200 lines (169 loc) • 5.89 kB
text/typescript
import { MapElementList } from "../internal/container/associative/MapElementList";
import { MultiIteratorTree } from "../internal/tree/MultiIteratorTree";
import { IForwardIterator } from "../iterator/IForwardIterator";
import { IPair } from "../utility/IPair";
import { Pair } from "../utility/Pair";
import { Entry } from "../utility/Entry";
import { Comparator } from "../internal/functional/Comparator";
import { less } from "../functional/comparators";
export class TreeMultiMap<Key, T>
{
private data_: MapElementList<Key, T, false, TreeMultiMap<Key, T>> = new MapElementList(<TreeMultiMap<Key, T>>this);
private tree_: MultiIteratorTree<Key, TreeMultiMap.Iterator<Key, T>>;
public constructor(comp: Comparator<Key> = (x, y) => less(x, y))
{
this.tree_ = new MultiIteratorTree<Key, TreeMultiMap.Iterator<Key, T>>
(
it => it.first,
comp,
(x, y) => MapElementList.Iterator._Compare_uid<Key, T, false, TreeMultiMap<Key, T>>(x, y)
);
}
public assign<InputIterator extends IForwardIterator<IPair<Key, T>, InputIterator>>
(first: InputIterator, last: InputIterator): void
{
if (this.empty() === false)
this.clear();
this.insert_range<InputIterator>(first, last);
}
public clear(): void
{
this.data_.clear();
this.tree_.clear();
}
public swap(obj: TreeMultiMap<Key, T>): void
{
// SWAP ELEMENTS
this.data_.swap(obj.data_);
const data: MapElementList<Key, T, true, TreeMultiMap<Key, T>> = this.data_;
this.data_ = obj.data_;
obj.data_ = data;
// SWAP TREE
const tree: MultiIteratorTree<Key, TreeMultiMap.Iterator<Key, T>> = this.tree_;
this.tree_ = obj.tree_;
obj.tree_ = tree;
}
/* ---------------------------------------------------------
ACCESSORS
--------------------------------------------------------- */
public size(): usize
{
return this.data_.size();
}
public empty(): boolean
{
return this.data_.empty();
}
public begin(): TreeMultiMap.Iterator<Key, T>
{
return this.data_.begin();
}
public end(): TreeMultiMap.Iterator<Key, T>
{
return this.data_.end();
}
public rbegin(): TreeMultiMap.ReverseIterator<Key, T>
{
return this.data_.rbegin();
}
public rend(): TreeMultiMap.ReverseIterator<Key, T>
{
return this.data_.rend();
}
public find(key: Key): TreeMultiMap.Iterator<Key, T>
{
const it: TreeMultiMap.Iterator<Key, T> = this.lower_bound(key);
return (it != this.end() && this.key_comp()(key, it.first) === false)
? it
: this.end();
}
public has(key: Key): boolean
{
return this.find(key) != this.end();
}
public count(key: Key): usize
{
let ret: usize = 0;
for (let it = this.find(key); it != this.end() && this.key_comp()(key, it.first) === false; it = it.next())
++ret;
return ret;
}
public key_comp(): Comparator<Key>
{
return this.tree_.key_comp();
}
public lower_bound(key: Key): TreeMultiMap.Iterator<Key, T>
{
return this.tree_.lower_bound(this.end(), key);
}
public upper_bound(key: Key): TreeMultiMap.Iterator<Key, T>
{
return this.tree_.upper_bound(this.end(), key);
}
public equal_range(key: Key): Pair<TreeMultiMap.Iterator<Key, T>, TreeMultiMap.Iterator<Key, T>>
{
return this.tree_.equal_range(this.end(), key);
}
/* ---------------------------------------------------------
ELEMENTS I/O
--------------------------------------------------------- */
public emplace(key: Key, value: T): TreeMultiMap.Iterator<Key, T>
{
const upper: TreeMultiMap.Iterator<Key, T> = this.upper_bound(key);
const it: TreeMultiMap.Iterator<Key, T> = this.data_.insert(upper, new Entry(key, value));
this.tree_.insert(it);
return it;
}
public emplace_hint(hint: TreeMultiMap.Iterator<Key, T>, key: Key, value: T): TreeMultiMap.Iterator<Key, T>
{
return this.emplace(key, value);
}
public insert_range<InputIterator extends IForwardIterator<IPair<Key, T>, InputIterator>>
(first: InputIterator, last: InputIterator): void
{
for (; first != last; first = first.next())
this.emplace(first.value.first, first.value.second);
}
public erase(first: TreeMultiMap.Iterator<Key, T>, last: TreeMultiMap.Iterator<Key, T> = first.next()): TreeMultiMap.Iterator<Key, T>
{
const ret = this.data_.erase(first, last);
for (; first != last; first = first.next())
this.tree_.erase(first);
return ret;
}
public erase_by_key(key: Key): usize
{
let it: TreeMultiMap.Iterator<Key, T> = this.find(key);
if (it == this.end())
return 0;
let count: usize = 0;
while (it != this.end() && this.key_comp()(key, it.first) === false)
{
it = this.erase(it);
++count;
}
return count;
}
}
export namespace TreeMultiMap
{
export type Iterator<Key, T> = MapElementList.Iterator<Key, T, false, TreeMultiMap<Key, T>>;
export type ReverseIterator<Key, T> = MapElementList.ReverseIterator<Key, T, false, TreeMultiMap<Key, T>>;
}