2010-05-09 99 views
3

我想用C++編寫高階函數filter。我想出了到目前爲止的代碼如下:C++中的高階函數«filter»

#include <iostream> 
#include <string> 
#include <functional> 
#include <algorithm> 
#include <vector> 
#include <list> 
#include <iterator> 

using namespace std; 

bool isOdd(int const i) { 
    return i % 2 != 0; 
} 

template < 
    template <class, class> class Container, 
    class Predicate, 
    class Allocator, 
    class A 
> 
Container<A, Allocator> filter(Container<A, Allocator> const & container, Predicate const & pred) { 
    Container<A, Allocator> filtered(container); 
    container.erase(remove_if(filtered.begin(), filtered.end(), pred), filtered.end()); 
    return filtered; 
} 

int main() { 
    int const a[] = {23, 12, 78, 21, 97, 64}; 
    vector<int const> const v(a, a + 6); 
    vector<int const> const filtered = filter(v, isOdd); 
    copy(filtered.begin(), filtered.end(), ostream_iterator<int const>(cout, " ")); 
} 

但是在編譯的代碼,我得到的是我無法理解,因此以下錯誤消息擺脫:

/usr/include/c++/4.3/ext/new_allocator.h: In instantiation of ‘__gnu_cxx::new_allocator<const int>’: 
/usr/include/c++/4.3/bits/allocator.h:84: instantiated from ‘std::allocator<const int>’ 
/usr/include/c++/4.3/bits/stl_vector.h:75: instantiated from ‘std::_Vector_base<const int, std::allocator<const int> >’ 
/usr/include/c++/4.3/bits/stl_vector.h:176: instantiated from ‘std::vector<const int, std::allocator<const int> >’ 
Filter.cpp:29: instantiated from here 
/usr/include/c++/4.3/ext/new_allocator.h:82: error: ‘const _Tp* __gnu_cxx::new_allocator<_Tp>::address(const _Tp&) const [with _Tp = const int]’ cannot be overloaded 
/usr/include/c++/4.3/ext/new_allocator.h:79: error: with ‘_Tp* __gnu_cxx::new_allocator<_Tp>::address(_Tp&) const [with _Tp = const int]’ 
Filter.cpp: In function ‘Container<A, Allocator> filter(const Container<A, Allocator>&, const Predicate&) [with Container = std::vector, Predicate = bool()(int), Allocator = std::allocator<const int>, A = const int]’: 
Filter.cpp:30: instantiated from here 
Filter.cpp:23: error: passing ‘const std::vector<const int, std::allocator<const int> >’ as ‘this’ argument of ‘__gnu_cxx::__normal_iterator<typename std::_Vector_base<_Tp, _Alloc>::_Tp_alloc_type::pointer, std::vector<_Tp, _Alloc> > std::vector<_Tp, _Alloc>::erase(__gnu_cxx::__normal_iterator<typename std::_Vector_base<_Tp, _Alloc>::_Tp_alloc_type::pointer, std::vector<_Tp, _Alloc> >, __gnu_cxx::__normal_iterator<typename std::_Vector_base<_Tp, _Alloc>::_Tp_alloc_type::pointer, std::vector<_Tp, _Alloc> >) [with _Tp = const int, _Alloc = std::allocator<const int>]’ discards qualifiers 
/usr/include/c++/4.3/bits/stl_algo.h: In function ‘_FIter std::remove_if(_FIter, _FIter, _Predicate) [with _FIter = __gnu_cxx::__normal_iterator<const int*, std::vector<const int, std::allocator<const int> > >, _Predicate = bool (*)(int)]’: 
Filter.cpp:23: instantiated from ‘Container<A, Allocator> filter(const Container<A, Allocator>&, const Predicate&) [with Container = std::vector, Predicate = bool()(int), Allocator = std::allocator<const int>, A = const int]’ 
Filter.cpp:30: instantiated from here 
/usr/include/c++/4.3/bits/stl_algo.h:821: error: assignment of read-only location ‘__result.__gnu_cxx::__normal_iterator<_Iterator, _Container>::operator* [with _Iterator = const int*, _Container = std::vector<const int, std::allocator<const int> >]()’ 
/usr/include/c++/4.3/ext/new_allocator.h: In member function ‘void __gnu_cxx::new_allocator<_Tp>::deallocate(_Tp*, size_t) [with _Tp = const int]’: 
/usr/include/c++/4.3/bits/stl_vector.h:150: instantiated from ‘void std::_Vector_base<_Tp, _Alloc>::_M_deallocate(_Tp*, size_t) [with _Tp = const int, _Alloc = std::allocator<const int>]’ 
/usr/include/c++/4.3/bits/stl_vector.h:136: instantiated from ‘std::_Vector_base<_Tp, _Alloc>::~_Vector_base() [with _Tp = const int, _Alloc = std::allocator<const int>]’ 
/usr/include/c++/4.3/bits/stl_vector.h:286: instantiated from ‘std::vector<_Tp, _Alloc>::vector(_InputIterator, _InputIterator, const _Alloc&) [with _InputIterator = const int*, _Tp = const int, _Alloc = std::allocator<const int>]’ 
Filter.cpp:29: instantiated from here 
/usr/include/c++/4.3/ext/new_allocator.h:98: error: invalid conversion from ‘const void*’ to ‘void*’ 
/usr/include/c++/4.3/ext/new_allocator.h:98: error: initializing argument 1 of ‘void operator delete(void*)’ 
/usr/include/c++/4.3/bits/stl_algobase.h: In function ‘_OI std::__copy_move_a(_II, _II, _OI) [with bool _IsMove = false, _II = const int*, _OI = const int*]’: 
/usr/include/c++/4.3/bits/stl_algobase.h:435: instantiated from ‘_OI std::__copy_move_a2(_II, _II, _OI) [with bool _IsMove = false, _II = __gnu_cxx::__normal_iterator<const int*, std::vector<const int, std::allocator<const int> > >, _OI = __gnu_cxx::__normal_iterator<const int*, std::vector<const int, std::allocator<const int> > >]’ 
/usr/include/c++/4.3/bits/stl_algobase.h:466: instantiated from ‘_OI std::copy(_II, _II, _OI) [with _II = __gnu_cxx::__normal_iterator<const int*, std::vector<const int, std::allocator<const int> > >, _OI = __gnu_cxx::__normal_iterator<const int*, std::vector<const int, std::allocator<const int> > >]’ 
/usr/include/c++/4.3/bits/vector.tcc:136: instantiated from ‘__gnu_cxx::__normal_iterator<typename std::_Vector_base<_Tp, _Alloc>::_Tp_alloc_type::pointer, std::vector<_Tp, _Alloc> > std::vector<_Tp, _Alloc>::erase(__gnu_cxx::__normal_iterator<typename std::_Vector_base<_Tp, _Alloc>::_Tp_alloc_type::pointer, std::vector<_Tp, _Alloc> >, __gnu_cxx::__normal_iterator<typename std::_Vector_base<_Tp, _Alloc>::_Tp_alloc_type::pointer, std::vector<_Tp, _Alloc> >) [with _Tp = const int, _Alloc = std::allocator<const int>]’ 
Filter.cpp:23: instantiated from ‘Container<A, Allocator> filter(const Container<A, Allocator>&, const Predicate&) [with Container = std::vector, Predicate = bool()(int), Allocator = std::allocator<const int>, A = const int]’ 
Filter.cpp:30: instantiated from here 
/usr/include/c++/4.3/bits/stl_algobase.h:396: error: no matching function for call to ‘std::__copy_move<false, true, std::random_access_iterator_tag>::__copy_m(const int*&, const int*&, const int*&)’ 

請告訴我我在這裏做錯了什麼,以及實現我想要的那種更高階多態性的正確方法是什麼。

謝謝。

編輯:

謝謝大家。下面是新的代碼,我將你的建議後,得到了(和現在的工作,耶!)

#include <iostream> 
#include <string> 
#include <functional> 
#include <algorithm> 
#include <vector> 
#include <list> 
#include <iterator> 

using namespace std; 

bool isOdd(int const i) { 
    return i % 2 != 0; 
} 

template < 
    template <typename, typename> class Container, 
    typename Predicate, 
    typename Allocator, 
    typename A 
> 
Container<A, Allocator> filter(Container<A, Allocator> const & container, Predicate const & pred) { 
    Container<A, Allocator> filtered(container); 
    filtered.erase(remove_if(filtered.begin(), filtered.end(), pred), filtered.end()); 
    return filtered; 
} 

int main() { 
    int a[] = {23, 12, 78, 21, 97, 64}; 
    vector<int> v(a, a + 6); 
    vector<int> filtered = filter(v, isOdd); 
    copy(filtered.begin(), filtered.end(), ostream_iterator<int>(cout, " ")); 
} 
+0

這個功能不是很貴嗎?您正在複製容器,過濾副本,然後按值返回已過濾的副本。爲什麼不直接在原始容器上使用'remove_if()'?或者,創建一個別名函數filter(),它接受迭代器然後調用remove_if()'? – wilhelmtell 2010-05-09 09:28:43

回答

5

爲什麼你Container參數化呢?

template <typename C, typename P> 
C filter(C const & container, P pred) { 
    C filtered(container); 
    filtered.erase(remove_if(filtered.begin(), filtered.end(), pred), filtered.end()); 
    return filtered; 
} 

同樣適用。請注意,我通過值爲P而不是通過const引用,正如Meyers在Effective C++中所建議的(迭代器和函子應該按值傳遞)。

+0

如果我想訪問容器的類型參數該怎麼辦? – 2010-05-09 09:28:41

+3

@紅Hyena:然後你'typename C :: value_type' – UncleBens 2010-05-09 09:30:20

+0

還有一個問題......爲什麼不通過引用傳遞迭代器和函數? – 2010-05-09 09:42:10

3

的錯誤不是在filter,但在:

int main() { 
    int const a[] = {23, 12, 78, 21, 97, 64}; 
    vector<int const> const v(a, a + 6); 
} 

You can't have a vector of const stuff。拆下內常量:(當然還有,filtered.erase,不container.erase

int main() { 
    int const a[] = {23, 12, 78, 21, 97, 64}; 
    vector<int> const v(a, a + 6); 
} 

4

怎麼樣remove_copy_if呢? (用isEven())。它已經爲你準備好了。

+0

或者實現'copy_if'。在C++中,與迭代器相比,它更容易使用容器。 – 2010-05-09 10:42:09

0

container是一個常量引用。你不能打電話給erase()。你可能打電話給

filtered.erase(remove_if(filtered.begin(), filtered.end(), pred), filtered.end());