Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
P
pybind11
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
open
pybind11
Commits
89f2db45
Commit
89f2db45
authored
Aug 25, 2016
by
Wenzel Jakob
Committed by
GitHub
Aug 25, 2016
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #353 from aldanor/feature/generalized-iterators
Add support for iterators with different begin/end types
parents
1ffce742
606160ed
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
61 additions
and
10 deletions
+61
-10
docs/changelog.rst
+2
-1
include/pybind11/pybind11.h
+12
-8
include/pybind11/stl_bind.h
+1
-1
tests/test_sequences_and_iterators.cpp
+34
-0
tests/test_sequences_and_iterators.py
+12
-0
No files found.
docs/changelog.rst
View file @
89f2db45
...
...
@@ -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
* A return value policy can now be passed to ``handle::operator()``
* ``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
* Added ``repr()`` method to the ``handle`` class.
* Added support for registering structured dtypes via ``PYBIND11_NUMPY_DTYPE()`` macro.
...
...
include/pybind11/pybind11.h
View file @
89f2db45
...
...
@@ -1119,8 +1119,10 @@ PYBIND11_NOINLINE inline void keep_alive_impl(int Nurse, int Patient, handle arg
keep_alive_impl
(
nurse
,
patient
);
}
template
<
typename
Iterator
,
bool
KeyIterator
=
false
>
struct
iterator_state
{
Iterator
it
,
end
;
template
<
typename
Iterator
,
typename
Sentinel
,
bool
KeyIterator
=
false
>
struct
iterator_state
{
Iterator
it
;
Sentinel
end
;
bool
first
;
};
...
...
@@ -1129,10 +1131,11 @@ NAMESPACE_END(detail)
template
<
typename
...
Args
>
detail
::
init
<
Args
...
>
init
()
{
return
detail
::
init
<
Args
...
>
();
}
template
<
typename
Iterator
,
typename
Sentinel
,
typename
ValueType
=
decltype
(
*
std
::
declval
<
Iterator
>
()),
typename
...
Extra
>
iterator
make_iterator
(
Iterator
first
,
Iterator
last
,
Extra
&&
...
extra
)
{
typedef
detail
::
iterator_state
<
Iterator
>
state
;
iterator
make_iterator
(
Iterator
first
,
Sentinel
last
,
Extra
&&
...
extra
)
{
typedef
detail
::
iterator_state
<
Iterator
,
Sentinel
>
state
;
if
(
!
detail
::
get_type_info
(
typeid
(
state
)))
{
class_
<
state
>
(
handle
(),
""
)
...
...
@@ -1151,10 +1154,11 @@ iterator make_iterator(Iterator first, Iterator last, Extra &&... extra) {
return
(
iterator
)
cast
(
state
{
first
,
last
,
true
});
}
template
<
typename
Iterator
,
typename
KeyType
=
decltype
(
std
::
declval
<
Iterator
>
()
->
first
),
typename
Sentinel
,
typename
KeyType
=
decltype
((
*
std
::
declval
<
Iterator
>
()).
first
),
typename
...
Extra
>
iterator
make_key_iterator
(
Iterator
first
,
Iterator
last
,
Extra
&&
...
extra
)
{
typedef
detail
::
iterator_state
<
Iterator
,
true
>
state
;
iterator
make_key_iterator
(
Iterator
first
,
Sentinel
last
,
Extra
&&
...
extra
)
{
typedef
detail
::
iterator_state
<
Iterator
,
Sentinel
,
true
>
state
;
if
(
!
detail
::
get_type_info
(
typeid
(
state
)))
{
class_
<
state
>
(
handle
(),
""
)
...
...
@@ -1166,7 +1170,7 @@ iterator make_key_iterator(Iterator first, Iterator last, Extra &&... extra) {
s
.
first
=
false
;
if
(
s
.
it
==
s
.
end
)
throw
stop_iteration
();
return
s
.
it
->
first
;
return
(
*
s
.
it
).
first
;
},
return_value_policy
::
reference_internal
,
std
::
forward
<
Extra
>
(
extra
)...);
}
...
...
include/pybind11/stl_bind.h
View file @
89f2db45
...
...
@@ -245,7 +245,7 @@ pybind11::class_<std::vector<T, Allocator>, holder_type> bind_vector(pybind11::m
cl
.
def
(
"__iter__"
,
[](
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 */
);
...
...
tests/test_sequences_and_iterators.cpp
View file @
89f2db45
...
...
@@ -116,6 +116,15 @@ private:
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
// map-like functionality.
class
StringMap
{
...
...
@@ -143,8 +152,24 @@ public:
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
)
{
py
::
class_
<
Sequence
>
seq
(
m
,
"Sequence"
);
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
::
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
// Obsolete: special data structure for exposing custom iterator types to python
...
...
tests/test_sequences_and_iterators.py
View file @
89f2db45
...
...
@@ -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
))
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
():
from
pybind11_tests
import
Sequence
,
ConstructorStats
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment