mirror of
https://github.com/pybind/pybind11.git
synced 2024-11-11 08:03:55 +00:00
fix various iterator issues (fixes #181)
This commit is contained in:
parent
a01977ec96
commit
6ca6e82f7c
@ -57,4 +57,9 @@ void init_issues(py::module &m) {
|
|||||||
v.push_back(p4);
|
v.push_back(p4);
|
||||||
return v;
|
return v;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// #181: iterator passthrough did not compile
|
||||||
|
m2.def("iterator_passthrough", [](py::iterator s) -> py::iterator {
|
||||||
|
return py::make_iterator(std::begin(s), std::end(s));
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@ sys.path.append('.')
|
|||||||
from example.issues import print_cchar, print_char
|
from example.issues import print_cchar, print_char
|
||||||
from example.issues import DispatchIssue, dispatch_issue_go
|
from example.issues import DispatchIssue, dispatch_issue_go
|
||||||
from example.issues import Placeholder ,return_vec_of_reference_wrapper
|
from example.issues import Placeholder ,return_vec_of_reference_wrapper
|
||||||
|
from example.issues import iterator_passthrough
|
||||||
|
|
||||||
print_cchar("const char *")
|
print_cchar("const char *")
|
||||||
print_char('c')
|
print_char('c')
|
||||||
@ -29,3 +30,5 @@ b = PyClass2()
|
|||||||
dispatch_issue_go(b)
|
dispatch_issue_go(b)
|
||||||
|
|
||||||
print(return_vec_of_reference_wrapper(Placeholder(4)))
|
print(return_vec_of_reference_wrapper(Placeholder(4)))
|
||||||
|
|
||||||
|
print(list(iterator_passthrough(iter([3, 5, 7, 9, 11, 13, 15]))))
|
||||||
|
@ -3,3 +3,4 @@ c
|
|||||||
Failed as expected: Tried to call pure virtual function "dispatch"
|
Failed as expected: Tried to call pure virtual function "dispatch"
|
||||||
Yay..
|
Yay..
|
||||||
[Placeholder[1], Placeholder[2], Placeholder[3], Placeholder[4]]
|
[Placeholder[1], Placeholder[2], Placeholder[3], Placeholder[4]]
|
||||||
|
[3, 5, 7, 9, 11, 13, 15]
|
||||||
|
@ -68,7 +68,7 @@ public:
|
|||||||
return handle(tmp);
|
return handle(tmp);
|
||||||
}
|
}
|
||||||
|
|
||||||
object& operator=(object &other) {
|
object& operator=(const object &other) {
|
||||||
other.inc_ref();
|
other.inc_ref();
|
||||||
dec_ref();
|
dec_ref();
|
||||||
m_ptr = other.m_ptr;
|
m_ptr = other.m_ptr;
|
||||||
@ -231,8 +231,8 @@ NAMESPACE_END(detail)
|
|||||||
Name(const handle &h, bool borrowed) : Parent(h, borrowed) { CvtStmt; } \
|
Name(const handle &h, bool borrowed) : Parent(h, borrowed) { CvtStmt; } \
|
||||||
Name(const object& o): Parent(o) { CvtStmt; } \
|
Name(const object& o): Parent(o) { CvtStmt; } \
|
||||||
Name(object&& o) noexcept : Parent(std::move(o)) { CvtStmt; } \
|
Name(object&& o) noexcept : Parent(std::move(o)) { CvtStmt; } \
|
||||||
Name& operator=(object&& o) noexcept { (void) static_cast<Name&>(object::operator=(std::move(o))); CvtStmt; return *this; } \
|
Name& operator=(object&& o) noexcept { (void) object::operator=(std::move(o)); CvtStmt; return *this; } \
|
||||||
Name& operator=(object& o) { return static_cast<Name&>(object::operator=(o)); CvtStmt; } \
|
Name& operator=(const object& o) { return static_cast<Name&>(object::operator=(o)); CvtStmt; } \
|
||||||
bool check() const { return m_ptr != nullptr && (bool) CheckFun(m_ptr); }
|
bool check() const { return m_ptr != nullptr && (bool) CheckFun(m_ptr); }
|
||||||
|
|
||||||
#define PYBIND11_OBJECT(Name, Parent, CheckFun) \
|
#define PYBIND11_OBJECT(Name, Parent, CheckFun) \
|
||||||
@ -244,21 +244,62 @@ NAMESPACE_END(detail)
|
|||||||
|
|
||||||
class iterator : public object {
|
class iterator : public object {
|
||||||
public:
|
public:
|
||||||
PYBIND11_OBJECT_DEFAULT(iterator, object, PyIter_Check)
|
PYBIND11_OBJECT_CVT(iterator, object, PyIter_Check, value = object(); ready = false)
|
||||||
iterator& operator++() {
|
iterator() : object(), value(object()), ready(false) { }
|
||||||
if (ptr())
|
iterator(const iterator& it) : object(it), value(it.value), ready(it.ready) { }
|
||||||
value = object(PyIter_Next(m_ptr), false);
|
iterator(iterator&& it) : object(std::move(it)), value(std::move(it.value)), ready(it.ready) { }
|
||||||
|
|
||||||
|
/** Caveat: this copy constructor does not (and cannot) clone the internal
|
||||||
|
state of the Python iterable */
|
||||||
|
iterator &operator=(const iterator &it) {
|
||||||
|
(void) object::operator=(it);
|
||||||
|
value = it.value;
|
||||||
|
ready = it.ready;
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
iterator &operator=(iterator &&it) noexcept {
|
||||||
|
(void) object::operator=(std::move(it));
|
||||||
|
value = std::move(it.value);
|
||||||
|
ready = it.ready;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
iterator& operator++() {
|
||||||
|
if (m_ptr)
|
||||||
|
advance();
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Caveat: this postincrement operator does not (and cannot) clone the
|
||||||
|
internal state of the Python iterable. It should only be used to
|
||||||
|
retrieve the current iterate using <tt>operator*()</tt> */
|
||||||
|
iterator operator++(int) {
|
||||||
|
iterator rv(*this);
|
||||||
|
rv.value = value;
|
||||||
|
if (m_ptr)
|
||||||
|
advance();
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
bool operator==(const iterator &it) const { return *it == **this; }
|
bool operator==(const iterator &it) const { return *it == **this; }
|
||||||
bool operator!=(const iterator &it) const { return *it != **this; }
|
bool operator!=(const iterator &it) const { return *it != **this; }
|
||||||
|
|
||||||
const handle &operator*() const {
|
const handle &operator*() const {
|
||||||
if (m_ptr && !value)
|
if (!ready && m_ptr) {
|
||||||
value = object(PyIter_Next(m_ptr), false);
|
auto& self = const_cast<iterator &>(*this);
|
||||||
|
self.advance();
|
||||||
|
self.ready = true;
|
||||||
|
}
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
mutable object value;
|
void advance() { value = object(PyIter_Next(m_ptr), false); }
|
||||||
|
|
||||||
|
private:
|
||||||
|
object value;
|
||||||
|
bool ready;
|
||||||
};
|
};
|
||||||
|
|
||||||
class iterable : public object {
|
class iterable : public object {
|
||||||
|
Loading…
Reference in New Issue
Block a user