mirror of
https://github.com/pybind/pybind11.git
synced 2024-11-11 08:03:55 +00:00
Merge pull request #353 from aldanor/feature/generalized-iterators
Add support for iterators with different begin/end types
This commit is contained in:
commit
89f2db4596
@ -25,7 +25,8 @@ Breaking changes queued for v2.0.0 (Not yet released)
|
|||||||
* ``std::enable_shared_from_this<>`` now also works for ``const`` values
|
* ``std::enable_shared_from_this<>`` now also works for ``const`` values
|
||||||
* A return value policy can now be passed to ``handle::operator()``
|
* A return value policy can now be passed to ``handle::operator()``
|
||||||
* ``make_iterator()`` improvements for better compatibility with various types
|
* ``make_iterator()`` improvements for better compatibility with various types
|
||||||
(now uses prefix increment operator)
|
(now uses prefix increment operator); it now also accepts iterators with
|
||||||
|
different begin/end types as long as they are equality comparable.
|
||||||
* ``arg()`` now accepts a wider range of argument types for default values
|
* ``arg()`` now accepts a wider range of argument types for default values
|
||||||
* Added ``repr()`` method to the ``handle`` class.
|
* Added ``repr()`` method to the ``handle`` class.
|
||||||
* Added support for registering structured dtypes via ``PYBIND11_NUMPY_DTYPE()`` macro.
|
* Added support for registering structured dtypes via ``PYBIND11_NUMPY_DTYPE()`` macro.
|
||||||
|
@ -1119,8 +1119,10 @@ PYBIND11_NOINLINE inline void keep_alive_impl(int Nurse, int Patient, handle arg
|
|||||||
keep_alive_impl(nurse, patient);
|
keep_alive_impl(nurse, patient);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Iterator, bool KeyIterator = false> struct iterator_state {
|
template <typename Iterator, typename Sentinel, bool KeyIterator = false>
|
||||||
Iterator it, end;
|
struct iterator_state {
|
||||||
|
Iterator it;
|
||||||
|
Sentinel end;
|
||||||
bool first;
|
bool first;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -1129,10 +1131,11 @@ NAMESPACE_END(detail)
|
|||||||
template <typename... Args> detail::init<Args...> init() { return detail::init<Args...>(); }
|
template <typename... Args> detail::init<Args...> init() { return detail::init<Args...>(); }
|
||||||
|
|
||||||
template <typename Iterator,
|
template <typename Iterator,
|
||||||
|
typename Sentinel,
|
||||||
typename ValueType = decltype(*std::declval<Iterator>()),
|
typename ValueType = decltype(*std::declval<Iterator>()),
|
||||||
typename... Extra>
|
typename... Extra>
|
||||||
iterator make_iterator(Iterator first, Iterator last, Extra &&... extra) {
|
iterator make_iterator(Iterator first, Sentinel last, Extra &&... extra) {
|
||||||
typedef detail::iterator_state<Iterator> state;
|
typedef detail::iterator_state<Iterator, Sentinel> state;
|
||||||
|
|
||||||
if (!detail::get_type_info(typeid(state))) {
|
if (!detail::get_type_info(typeid(state))) {
|
||||||
class_<state>(handle(), "")
|
class_<state>(handle(), "")
|
||||||
@ -1151,10 +1154,11 @@ iterator make_iterator(Iterator first, Iterator last, Extra &&... extra) {
|
|||||||
return (iterator) cast(state { first, last, true });
|
return (iterator) cast(state { first, last, true });
|
||||||
}
|
}
|
||||||
template <typename Iterator,
|
template <typename Iterator,
|
||||||
typename KeyType = decltype(std::declval<Iterator>()->first),
|
typename Sentinel,
|
||||||
|
typename KeyType = decltype((*std::declval<Iterator>()).first),
|
||||||
typename... Extra>
|
typename... Extra>
|
||||||
iterator make_key_iterator(Iterator first, Iterator last, Extra &&... extra) {
|
iterator make_key_iterator(Iterator first, Sentinel last, Extra &&... extra) {
|
||||||
typedef detail::iterator_state<Iterator, true> state;
|
typedef detail::iterator_state<Iterator, Sentinel, true> state;
|
||||||
|
|
||||||
if (!detail::get_type_info(typeid(state))) {
|
if (!detail::get_type_info(typeid(state))) {
|
||||||
class_<state>(handle(), "")
|
class_<state>(handle(), "")
|
||||||
@ -1166,7 +1170,7 @@ iterator make_key_iterator(Iterator first, Iterator last, Extra &&... extra) {
|
|||||||
s.first = false;
|
s.first = false;
|
||||||
if (s.it == s.end)
|
if (s.it == s.end)
|
||||||
throw stop_iteration();
|
throw stop_iteration();
|
||||||
return s.it->first;
|
return (*s.it).first;
|
||||||
}, return_value_policy::reference_internal, std::forward<Extra>(extra)...);
|
}, return_value_policy::reference_internal, std::forward<Extra>(extra)...);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -245,7 +245,7 @@ pybind11::class_<std::vector<T, Allocator>, holder_type> bind_vector(pybind11::m
|
|||||||
|
|
||||||
cl.def("__iter__",
|
cl.def("__iter__",
|
||||||
[](Vector &v) {
|
[](Vector &v) {
|
||||||
return pybind11::make_iterator<ItType, T>(v.begin(), v.end());
|
return pybind11::make_iterator<ItType, ItType, T>(v.begin(), v.end());
|
||||||
},
|
},
|
||||||
pybind11::keep_alive<0, 1>() /* Essential: keep list alive while iterator exists */
|
pybind11::keep_alive<0, 1>() /* Essential: keep list alive while iterator exists */
|
||||||
);
|
);
|
||||||
|
@ -116,6 +116,15 @@ private:
|
|||||||
float *m_data;
|
float *m_data;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class IntPairs {
|
||||||
|
public:
|
||||||
|
IntPairs(std::vector<std::pair<int, int>> data) : data_(std::move(data)) {}
|
||||||
|
const std::pair<int, int>* begin() const { return data_.data(); }
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<std::pair<int, int>> data_;
|
||||||
|
};
|
||||||
|
|
||||||
// Interface of a map-like object that isn't (directly) an unordered_map, but provides some basic
|
// Interface of a map-like object that isn't (directly) an unordered_map, but provides some basic
|
||||||
// map-like functionality.
|
// map-like functionality.
|
||||||
class StringMap {
|
class StringMap {
|
||||||
@ -143,8 +152,24 @@ public:
|
|||||||
decltype(map.cend()) end() const { return map.cend(); }
|
decltype(map.cend()) end() const { return map.cend(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
class NonZeroIterator {
|
||||||
|
const T* ptr_;
|
||||||
|
public:
|
||||||
|
NonZeroIterator(const T* ptr) : ptr_(ptr) {}
|
||||||
|
const T& operator*() const { return *ptr_; }
|
||||||
|
NonZeroIterator& operator++() { ++ptr_; return *this; }
|
||||||
|
};
|
||||||
|
|
||||||
|
class NonZeroSentinel {};
|
||||||
|
|
||||||
|
template<typename A, typename B>
|
||||||
|
bool operator==(const NonZeroIterator<std::pair<A, B>>& it, const NonZeroSentinel&) {
|
||||||
|
return !(*it).first || !(*it).second;
|
||||||
|
}
|
||||||
|
|
||||||
void init_ex_sequences_and_iterators(py::module &m) {
|
void init_ex_sequences_and_iterators(py::module &m) {
|
||||||
|
|
||||||
py::class_<Sequence> seq(m, "Sequence");
|
py::class_<Sequence> seq(m, "Sequence");
|
||||||
|
|
||||||
seq.def(py::init<size_t>())
|
seq.def(py::init<size_t>())
|
||||||
@ -210,6 +235,15 @@ void init_ex_sequences_and_iterators(py::module &m) {
|
|||||||
py::keep_alive<0, 1>())
|
py::keep_alive<0, 1>())
|
||||||
;
|
;
|
||||||
|
|
||||||
|
py::class_<IntPairs>(m, "IntPairs")
|
||||||
|
.def(py::init<std::vector<std::pair<int, int>>>())
|
||||||
|
.def("nonzero", [](const IntPairs& s) {
|
||||||
|
return py::make_iterator(NonZeroIterator<std::pair<int, int>>(s.begin()), NonZeroSentinel());
|
||||||
|
}, py::keep_alive<0, 1>())
|
||||||
|
.def("nonzero_keys", [](const IntPairs& s) {
|
||||||
|
return py::make_key_iterator(NonZeroIterator<std::pair<int, int>>(s.begin()), NonZeroSentinel());
|
||||||
|
}, py::keep_alive<0, 1>());
|
||||||
|
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
// Obsolete: special data structure for exposing custom iterator types to python
|
// Obsolete: special data structure for exposing custom iterator types to python
|
||||||
|
@ -10,6 +10,18 @@ def allclose(a_list, b_list, rel_tol=1e-05, abs_tol=0.0):
|
|||||||
return all(isclose(a, b, rel_tol=rel_tol, abs_tol=abs_tol) for a, b in zip(a_list, b_list))
|
return all(isclose(a, b, rel_tol=rel_tol, abs_tol=abs_tol) for a, b in zip(a_list, b_list))
|
||||||
|
|
||||||
|
|
||||||
|
def test_generalized_iterators():
|
||||||
|
from pybind11_tests import IntPairs
|
||||||
|
|
||||||
|
assert list(IntPairs([(1, 2), (3, 4), (0, 5)]).nonzero()) == [(1, 2), (3, 4)]
|
||||||
|
assert list(IntPairs([(1, 2), (2, 0), (0, 3), (4, 5)]).nonzero()) == [(1, 2)]
|
||||||
|
assert list(IntPairs([(0, 3), (1, 2), (3, 4)]).nonzero()) == []
|
||||||
|
|
||||||
|
assert list(IntPairs([(1, 2), (3, 4), (0, 5)]).nonzero_keys()) == [1, 3]
|
||||||
|
assert list(IntPairs([(1, 2), (2, 0), (0, 3), (4, 5)]).nonzero_keys()) == [1]
|
||||||
|
assert list(IntPairs([(0, 3), (1, 2), (3, 4)]).nonzero_keys()) == []
|
||||||
|
|
||||||
|
|
||||||
def test_sequence():
|
def test_sequence():
|
||||||
from pybind11_tests import Sequence, ConstructorStats
|
from pybind11_tests import Sequence, ConstructorStats
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user