-
Notifications
You must be signed in to change notification settings - Fork 23
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Replace some
std::set<int>
with a sorted std::vector
`Transducer::determinize()` doesn't do very much inserting into sets, but does iterate over them and use them as map keys, both operations which are substantially sped up by ensuring that the elements of the set are contiguous. Thus this commit adds `VecSet` which is a wrapper around `std::vector` that ensures that inserted elements are sorted and unique. The results are significant improvements in both runtime and memory usage for transducer minimization. Effect on `lt-comp`: | | eng | oci | |----------|---------------|---------------| | orig | 136 MB 23.9 s | 980 MB 219 s | | pre-comp | 160 MB 16.8 s | 1120 MB 163 s | | VecSet | 99 MB 12.7 s | 800 MB 116 s | | net diff | -27% -47% | -18% -47% |
- Loading branch information
1 parent
9111665
commit 5bcdbfa
Showing
3 changed files
with
286 additions
and
9 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,277 @@ | ||
/* | ||
* Copyright (C) 2022 Apertium | ||
* | ||
* This program is free software; you can redistribute it and/or | ||
* modify it under the terms of the GNU General Public License as | ||
* published by the Free Software Foundation; either version 2 of the | ||
* License, or (at your option) any later version. | ||
* | ||
* This program is distributed in the hope that it will be useful, but | ||
* WITHOUT ANY WARRANTY; without even the implied warranty of | ||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
* General Public License for more details. | ||
* | ||
* You should have received a copy of the GNU General Public License | ||
* along with this program; if not, see <https://www.gnu.org/licenses/>. | ||
*/ | ||
|
||
#pragma once | ||
#ifndef c6d28b7452ec699b_SORTED_VECTOR_HPP | ||
#define c6d28b7452ec699b_SORTED_VECTOR_HPP | ||
#include <set> | ||
#include <vector> | ||
#include <algorithm> | ||
#include <functional> | ||
|
||
namespace detail { | ||
template<typename ForwardIt, typename Comp> | ||
bool is_sorted(ForwardIt first, ForwardIt last, Comp comp) { | ||
if (first != last) { | ||
ForwardIt next = first; | ||
while (++next != last) { | ||
if (comp(*next, *first)) { | ||
return false; | ||
} | ||
first = next; | ||
} | ||
} | ||
return true; | ||
} | ||
} | ||
|
||
template<typename T, typename Comp = std::less<T>> | ||
class sorted_vector { | ||
public: | ||
typedef typename std::vector<T> container; | ||
typedef typename container::iterator iterator; | ||
typedef typename container::const_iterator const_iterator; | ||
typedef typename container::const_reverse_iterator const_reverse_iterator; | ||
typedef typename container::size_type size_type; | ||
typedef T value_type; | ||
typedef T key_type; | ||
|
||
sorted_vector() {} | ||
|
||
sorted_vector(const std::set<T>& o) { | ||
insert(o.begin(), o.end()); | ||
} | ||
|
||
std::pair<iterator, bool> insert(T t) { | ||
if (elements.empty()) { | ||
elements.push_back(t); | ||
return std::make_pair(elements.begin(), true); | ||
} | ||
iterator it = std::lower_bound(elements.begin(), elements.end(), t, comp); | ||
size_t at = std::distance(elements.begin(), it); | ||
if (it == elements.end() || comp(*it, t) || comp(t, *it)) { | ||
elements.insert(it, t); | ||
return std::make_pair(elements.begin() + at, true); | ||
} | ||
return std::make_pair(elements.begin() + at, false); | ||
} | ||
|
||
template<typename It> | ||
void insert(It b, It e) { | ||
size_t d = std::distance(b, e); | ||
if (d == 1) { | ||
insert(*b); | ||
return; | ||
} | ||
|
||
static thread_local container merged; | ||
merged.resize(0); | ||
merged.reserve(elements.size() + d); | ||
|
||
if (detail::is_sorted(b, e, comp)) { | ||
std::merge(elements.begin(), elements.end(), b, e, std::back_inserter(merged), comp); | ||
} | ||
else { | ||
static thread_local container sorted; | ||
sorted.assign(b, e); | ||
std::sort(sorted.begin(), sorted.end(), comp); | ||
std::merge(elements.begin(), elements.end(), sorted.begin(), sorted.end(), std::back_inserter(merged), comp); | ||
} | ||
|
||
merged.swap(elements); | ||
auto it = std::unique(elements.begin(), elements.end()); | ||
elements.erase(it, elements.end()); | ||
} | ||
|
||
void push_back(T t) { | ||
insert(t); | ||
} | ||
|
||
bool erase(T t) { | ||
if (elements.empty()) { | ||
return false; | ||
} | ||
if (comp(elements.back(), t)) { | ||
return false; | ||
} | ||
if (comp(t, elements.front())) { | ||
return false; | ||
} | ||
auto it = lower_bound(t); | ||
if (it != elements.end() && !comp(*it, t) && !comp(t, *it)) { | ||
elements.erase(it); | ||
return true; | ||
} | ||
return false; | ||
} | ||
|
||
const_iterator erase(const_iterator it) { | ||
size_type o = std::distance<const_iterator>(elements.begin(), it); | ||
return elements.erase(elements.begin() + o); | ||
} | ||
|
||
template<typename It> | ||
void erase(It b, It e) { | ||
for (; b != e; ++b) { | ||
erase(*b); | ||
} | ||
} | ||
|
||
const_iterator find(T t) const { | ||
if (elements.empty()) { | ||
return elements.end(); | ||
} | ||
if (comp(elements.back(), t)) { | ||
return elements.end(); | ||
} | ||
if (comp(t, elements.front())) { | ||
return elements.end(); | ||
} | ||
auto it = lower_bound(t); | ||
if (it != elements.end() && (comp(*it, t) || comp(t, *it))) { | ||
return elements.end(); | ||
} | ||
return it; | ||
} | ||
|
||
size_t count(T t) const { | ||
return (find(t) != end()); | ||
} | ||
|
||
iterator begin() { | ||
return elements.begin(); | ||
} | ||
|
||
iterator end() { | ||
return elements.end(); | ||
} | ||
|
||
const_iterator begin() const { | ||
return elements.begin(); | ||
} | ||
|
||
const_iterator end() const { | ||
return elements.end(); | ||
} | ||
|
||
const_iterator cbegin() const { | ||
return elements.cbegin(); | ||
} | ||
|
||
const_iterator cend() const { | ||
return elements.cend(); | ||
} | ||
|
||
const_reverse_iterator rbegin() const { | ||
return elements.rbegin(); | ||
} | ||
|
||
const_reverse_iterator rend() const { | ||
return elements.rend(); | ||
} | ||
|
||
T front() const { | ||
return elements.front(); | ||
} | ||
|
||
T back() const { | ||
return elements.back(); | ||
} | ||
|
||
iterator lower_bound(T t) { | ||
return std::lower_bound(elements.begin(), elements.end(), t, comp); | ||
} | ||
|
||
const_iterator lower_bound(T t) const { | ||
return std::lower_bound(elements.begin(), elements.end(), t, comp); | ||
} | ||
|
||
const_iterator upper_bound(T t) const { | ||
return std::upper_bound(elements.begin(), elements.end(), t, comp); | ||
} | ||
|
||
bool intersects(const sorted_vector<T>& other) const { | ||
auto ti = begin(); | ||
auto oi = other.begin(); | ||
auto te = end(); | ||
auto oe = other.end(); | ||
while (ti != te && oi != oe) { | ||
if (*ti == *oi) { | ||
return true; | ||
} | ||
else if (comp(*ti, *oi)) { | ||
++ti; | ||
} | ||
else { | ||
++oi; | ||
} | ||
} | ||
return false; | ||
} | ||
|
||
size_type size() const { | ||
return elements.size(); | ||
} | ||
|
||
size_type capacity() const { | ||
return elements.capacity(); | ||
} | ||
|
||
bool empty() const { | ||
return elements.empty(); | ||
} | ||
|
||
template<typename It> | ||
void assign(It b, It e) { | ||
clear(); | ||
insert(b, e); | ||
} | ||
|
||
void assign(const_iterator b, const_iterator e) { | ||
elements.assign(b, e); | ||
} | ||
|
||
void swap(sorted_vector& other) { | ||
elements.swap(other.elements); | ||
} | ||
|
||
void clear() { | ||
elements.clear(); | ||
} | ||
|
||
void sort() { | ||
std::sort(elements.begin(), elements.end(), Comp()); | ||
} | ||
|
||
void pop_back() { | ||
elements.pop_back(); | ||
} | ||
|
||
container& get() { | ||
return elements; | ||
} | ||
|
||
bool operator<(const sorted_vector<T>& o) const { | ||
return elements < o.elements; | ||
} | ||
|
||
private: | ||
container elements; | ||
Comp comp; | ||
}; | ||
|
||
#endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters