// ************************************************************************** // * This file is part of the FreeFileSync project. It is distributed under * // * GNU General Public License: http://www.gnu.org/licenses/gpl-3.0 * // * Copyright (C) Zenju (zenju AT gmx DOT de) - All Rights Reserved * // ************************************************************************** #ifndef STL_TOOLS_HEADER_84567184321434 #define STL_TOOLS_HEADER_84567184321434 #include #include #include //enhancements for namespace zen { //idomatic remove selected elements from container template void vector_remove_if(V& vec, Predicate p); template void vector_append(V& vec, const W& vec2); template void set_append(V& s, const W& s2); template void set_remove_if(S& set, Predicate p); template void map_remove_if(M& map, Predicate p); template V& map_add_or_update(M& map, const K& key, const V& value); //efficient add or update without "default-constructible" requirement (Effective STL, item 24) //binary search returning an iterator template ForwardIterator binary_search(ForwardIterator first, ForwardIterator last, const T& value, CompLess less); template BidirectionalIterator find_last(BidirectionalIterator first, BidirectionalIterator last, const T& value); //replacement for std::find_end taking advantage of bidirectional iterators (and giving the algorithm a reasonable name) template BidirectionalIterator1 search_last(BidirectionalIterator1 first1, BidirectionalIterator1 last1, BidirectionalIterator2 first2, BidirectionalIterator2 last2); template bool equal(InputIterator1 first1, InputIterator1 last1, InputIterator2 first2, InputIterator2 last2); //until std::make_unique is available in GCC: template inline std::unique_ptr make_unique(Args&& ... args) { return std::unique_ptr(new T(std::forward(args)...)); } //######################## implementation ######################## template inline void vector_remove_if(V& vec, Predicate p) { static_assert(IsSameType::iterator_category, std::random_access_iterator_tag>::value, "poor man's check for vector"); vec.erase(std::remove_if(vec.begin(), vec.end(), p), vec.end()); } template inline void vector_append(V& vec, const W& vec2) { vec.insert(vec.end(), vec2.begin(), vec2.end()); } template inline void set_append(V& s, const W& s2) { s.insert(s2.begin(), s2.end()); } template inline void set_remove_if(S& set, Predicate p) { //function compiles and fails (if we're lucky) not before runtime for std::vector!!! static_assert(!IsSameType::iterator_category, std::random_access_iterator_tag>::value, "poor man's check for non-vector"); for (auto iter = set.begin(); iter != set.end();) if (p(*iter)) set.erase(iter++); else ++iter; } template inline void map_remove_if(M& map, Predicate p) { set_remove_if(map, p); } template inline V& map_add_or_update(M& map, const K& key, const V& value) //efficient add or update without "default-constructible" requirement (Effective STL, item 24) { auto iter = map.lower_bound(key); if (iter != map.end() && !(map.key_comp()(key, iter->first))) { iter->second = value; return iter->second; } else return map.insert(iter, typename M::value_type(key, value))->second; } template inline ForwardIterator binary_search(ForwardIterator first, ForwardIterator last, const T& value, CompLess less) { first = std::lower_bound(first, last, value, less); if (first != last && !less(value, *first)) return first; else return last; } template inline BidirectionalIterator find_last(const BidirectionalIterator first, const BidirectionalIterator last, const T& value) { for (BidirectionalIterator iter = last; iter != first;) //reverse iteration: 1. check 2. decrement 3. evaluate { --iter; // if (*iter == value) return iter; } return last; } template inline BidirectionalIterator1 search_last(const BidirectionalIterator1 first1, BidirectionalIterator1 last1, const BidirectionalIterator2 first2, const BidirectionalIterator2 last2) { const BidirectionalIterator1 iterNotFound = last1; //reverse iteration: 1. check 2. decrement 3. evaluate for (;;) { BidirectionalIterator1 it1 = last1; BidirectionalIterator2 it2 = last2; for (;;) { if (it2 == first2) return it1; if (it1 == first1) return iterNotFound; --it1; --it2; if (*it1 != *it2) break; } --last1; } } template inline bool equal(InputIterator1 first1, InputIterator1 last1, InputIterator2 first2, InputIterator2 last2) { return last1 - first1 == last2 - first2 && std::equal(first1, last1, first2); } #if defined _MSC_VER && _MSC_VER <= 1600 //VS2010 performance bug in std::unordered_set<>: http://drdobbs.com/blogs/cpp/232200410 -> should be fixed in VS11 static_assert(false, ""); #endif } #endif //STL_TOOLS_HEADER_84567184321434