Skip to content

Commit b5ed696

Browse files
committed
API break: renamed some class_<..> members, added documentation
While documenting the nanobind ``class_<>::def*`` methods, I realized how poorly named they all are. For example, there were function ``.def_readonly_static()`` and ``.def_readwrite_static()`` for read-only and mutable fields, but their property variants were called ``.def_property_readonly_static()`` and ``.def_property_static()``. This commit changes the interface to be more consistent and less verbose. It now looks as follows: - Methods & constructors: ``.def()`` - Fields: ``.def_ro()`` and ``.def_rw()`` - Properties: ``.def_prop_ro()`` and ``.def_prop_rw()`` - Static methods: ``.def_static()`` - Static fields: ``.def_ro_static()`` and ``.def_rw_static()`` - Static properties: ``.def_prop_ro_static()`` and ``.def_prop_rw_static()`` Apologies about the API break. nanobind is still in its experimental phase (version number 0.x.y), which means that occasional API changes can occur if they are considered a long-term improvement.
1 parent 05a34a9 commit b5ed696

22 files changed

+933
-618
lines changed

docs/api_core.rst

+306-57
Large diffs are not rendered by default.

docs/api_extra.rst

+35
Original file line numberDiff line numberDiff line change
@@ -148,6 +148,41 @@ nanobind API and require an additional include directive:
148148
* - ``items(self, arg: Map) -> Map.ItemView``
149149
- Returns an iterable view of the map's items
150150

151+
Unique pointer deleter
152+
----------------------
153+
154+
The following *deleter* should be used to gain maximal flexibility in combination with
155+
``std::unique_ptr<..>``. It requires the following additional include directive:
156+
157+
.. code-block:: cpp
158+
159+
#include <nanobind/stl/unique_ptr.h>
160+
161+
See the two documentation sections on unique pointers for further detail
162+
(:ref:`#1 <unique_ptr>`, :ref:`#2 <unique_ptr_adv>`).
163+
164+
.. cpp:struct:: template <typename T> deleter
165+
166+
.. cpp:function:: deleter() = default
167+
168+
Create a deleter that destroys the object using a ``delete`` expression.
169+
170+
.. cpp:function:: deleter(handle h)
171+
172+
Create a deleter that destroys the object by reducing the Python reference count.
173+
174+
.. cpp:function:: bool owned_by_python() const
175+
176+
Check if the object is owned by Python.
177+
178+
.. cpp:function:: bool owned_by_cpp() const
179+
180+
Check if the object is owned by C++.
181+
182+
.. cpp:function:: void operator()(void * p) noexcept
183+
184+
Destroy the object at address `p`.
185+
151186
.. _iterator_bindings:
152187

153188
Iterator bindings

docs/basics.rst

+45-20
Original file line numberDiff line numberDiff line change
@@ -234,8 +234,8 @@ Docstrings
234234
----------
235235

236236
Let's add one more bit of flourish by assigning a docstring to the extension
237-
module itself. Add the following line anywhere in the body of the ``NB_MODULE()
238-
{...}`` declaration:
237+
module itself. Include the following line anywhere in the body of the
238+
``NB_MODULE() {...}`` declaration:
239239

240240
.. code-block:: cpp
241241
@@ -284,10 +284,10 @@ simple C++ type named ``Dog`` defined as follows:
284284
#include <string>
285285
286286
struct Dog {
287-
std::string m_name;
287+
std::string name;
288288
289289
std::string bark() const {
290-
return m_name + ": woof!";
290+
return name + ": woof!";
291291
}
292292
};
293293
@@ -305,7 +305,7 @@ The ``Dog`` bindings look as follows:
305305
.def(nb::init<>())
306306
.def(nb::init<const std::string &>())
307307
.def("bark", &Dog::bark)
308-
.def_readwrite("name", &Dog::name);
308+
.def_rw("name", &Dog::name);
309309
}
310310
311311
Let's look at selected lines of this example, starting with the added include directive:
@@ -317,8 +317,8 @@ Let's look at selected lines of this example, starting with the added include di
317317
nanobind has a minimal core and initially doesn't know how to deal with STL
318318
types like ``std::string``. This line imports a *type caster* that realizes a
319319
bidirectional conversion (C++ ``std::string`` ↔ Python ``str``) to make the
320-
example usable. An :ref:`upcoming documentation section <type_casters>`
321-
contrasts type casters and other alternatives.
320+
example usable. The next :ref:`documentation section <type_casters>` will
321+
provide more detail on type casters and other alternatives.
322322

323323
The class binding declaration :class:`nb::class_\<T\>() <class_>` supports both
324324
``class`` and ``struct``-style data structures.
@@ -333,15 +333,15 @@ and installs it in the :cpp:class:`nb::module_ <module_>` ``m``.
333333
Initially, this type is completely empty---it has no members and cannot be
334334
instantiated. The subsequent chain of binding declarations binds two
335335
constructor overloads (via :cpp:class:`nb::init\<...\>() <init>`), a method,
336-
and the ``name`` field (via :cpp:func:`.def_readwrite(..)
337-
<class_::def_readwrite>`).
336+
and the mutable ``name`` field (via :cpp:func:`.def_rw(..) <class_::def_rw>`,
337+
where ``rw`` stands for read/write access).
338338

339339
.. code-block:: cpp
340340
341341
.def(nb::init<>())
342342
.def(nb::init<const std::string &>())
343343
.def("bark", &Dog::bark)
344-
.def_readwrite("name", &Dog::name);
344+
.def_rw("name", &Dog::name);
345345
346346
An interactive Python session demonstrating this example is shown below:
347347

@@ -359,19 +359,43 @@ An interactive Python session demonstrating this example is shown below:
359359
>>> d.bark()
360360
'Charlie: woof!'
361361
362+
The example showed how to bind constructors, methods, and mutable fields. Many
363+
other things can be bound using analogous :cpp:class:`nb::class_\<...\>
364+
<class_>` methods:
365+
366+
.. list-table::
367+
:widths: 40 60
368+
:header-rows: 1
369+
370+
* - Type
371+
- method
372+
* - Methods & constructors
373+
- :cpp:func:`.def() <class_::def>`
374+
* - Fields
375+
- :cpp:func:`.def_ro() <class_::def_ro>`,
376+
:cpp:func:`.def_rw() <class_::def_rw>`
377+
* - Properties
378+
- :cpp:func:`.def_prop_ro() <class_::def_prop_ro>`,
379+
:cpp:func:`.def_prop_rw() <class_::def_prop_rw>`
380+
* - Static methods
381+
- :cpp:func:`.def_static() <class_::def_static>`
382+
* - Static fields
383+
- :cpp:func:`.def_ro_static() <class_::def_ro_static>`,
384+
:cpp:func:`.def_rw_static() <class_::def_rw_static>`
385+
* - Static properties
386+
- :cpp:func:`.def_prop_ro_static() <class_::def_prop_ro_static>`,
387+
:cpp:func:`.def_prop_rw_static() <class_::def_prop_rw_static>`
388+
362389
.. note::
363390

364-
Constructors and methods support :ref:`docstrings <docstrings>`,
391+
All of these binding declarations support :ref:`docstrings <docstrings>`,
365392
:ref:`keyword, and default argument <keyword_and_default_args>` annotations
366393
as before.
367394

368-
Binding fields
369-
--------------
370-
371395
.. _binding_lambdas:
372396

373-
Lambda functions
374-
----------------
397+
Binding lambda functions
398+
------------------------
375399

376400
Note how ``print(d)`` produced a rather useless summary in the example above:
377401

@@ -380,10 +404,11 @@ Note how ``print(d)`` produced a rather useless summary in the example above:
380404
>>> print(d)
381405
<my_ext.Dog object at 0x1044540f0>
382406
383-
To address this, we must add a special method named ``__repr__`` that returns a
384-
human-readable summary. Unfortunately, a function with such functionality does
385-
not exist in the ``Dog`` type, and it would be nice if we did not have to
386-
modify it. To accomplish this goal, we can instead bind a *lambda function*:
407+
To address this, we can add a special Python method named ``__repr__`` that
408+
returns a human-readable summary. Unfortunately, a corresponding function with
409+
such functionality does not currently exist in the C++ type, and it would be
410+
nice if we did not have to modify it. We can bind a *lambda function* to
411+
achieve both goals:
387412

388413
.. code-block:: cpp
389414

docs/changelog.rst

+28-1
Original file line numberDiff line numberDiff line change
@@ -10,10 +10,37 @@ current version is still in the prototype range (*0.x.y*), there are no (formal)
1010
guarantees of API or ABI stability. That said, I will do my best to minimize
1111
inconvenience whenever possible.
1212

13-
Version 0.1.1 (TBA)
13+
Version 0.2.0 (TBA)
1414
-------------------------------
1515
* Nanobind now features documentation on `readthedocs
1616
<https://nanobind.readthedocs.io>`_.
17+
* While writing the documentation, I realized how lengthy and inconsistently some of
18+
the :cpp:func:`class_\<T\>::def* <class_::def>` members are named. nanobind
19+
will from now on use the following API:
20+
21+
.. list-table::
22+
:widths: 40 60
23+
:header-rows: 1
24+
25+
* - Type
26+
- method
27+
* - Methods & constructors
28+
- :cpp:func:`.def() <class_::def>`
29+
* - Fields
30+
- :cpp:func:`.def_ro() <class_::def_ro>`,
31+
:cpp:func:`.def_rw() <class_::def_rw>`
32+
* - Properties
33+
- :cpp:func:`.def_prop_ro() <class_::def_prop_ro>`,
34+
:cpp:func:`.def_prop_rw() <class_::def_prop_rw>`
35+
* - Static methods
36+
- :cpp:func:`.def_static() <class_::def_static>`
37+
* - Static fields
38+
- :cpp:func:`.def_ro_static() <class_::def_ro_static>`,
39+
:cpp:func:`.def_rw_static() <class_::def_rw_static>`
40+
* - Static properties
41+
- :cpp:func:`.def_prop_ro_static() <class_::def_prop_ro_static>`,
42+
:cpp:func:`.def_prop_rw_static() <class_::def_prop_rw_static>`
43+
1744
* Added casters for dense matrix/array types from the `Eigen library
1845
<https://eigen.tuxfamily.org/index.php?title=Main_Page>`_. (PR `#120
1946
<https://github.com/wjakob/nanobind/pull/120>`_).

docs/exchanging.rst

+11-6
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,8 @@ Exchanging information
77

88
nanobind offers three fundamentally different ways of exchanging information
99
between Python and C++. Depending on the task at hand, one will usually be
10-
preferable over the others, hence it is important to be aware of the pros and
11-
cons of each approach.
10+
preferable over the others, hence it is important to be aware of their
11+
advantages and disadvantages.
1212

1313
.. _type_casters:
1414

@@ -127,12 +127,16 @@ The following table lists the currently available type casters:
127127
- ``#include <nanobind/eigen/dense.h>``
128128

129129

130-
**Con**: Every transition between the Python and C++ side will require a
130+
**Con**: Every transition between the Python and C++ side will generally require a
131131
conversion step (in this case, to re-create all list elements). This can be
132132
wasteful when the other side only needs to access a small part of the data.
133133
Conversely, the overhead should not be a problem when the data is fully
134134
"consumed" following conversion.
135135

136+
Note that some type casters (e.g., those for ``std::unique_ptr<..>``,
137+
``std::shared_ptr<..>``, :cpp:class:`nb::tensor <tensor>`, and for ``Eigen::*``
138+
can perform a type conversion without copying the underlying data.)
139+
136140
.. _type_caster_mutable:
137141

138142
Mutable reference issue
@@ -263,10 +267,11 @@ special cases (vectors, ordered/unordered maps, iterators):
263267
- ``#include <nanobind/make_iterator.h>``
264268
(:ref:`docs <iterator_bindings>`)
265269
* - Other types
266-
- See the :ref:`next section <bindings>`.
270+
- See the previous example on :ref:`binding custom types <binding_types>`.
267271

268-
In general, you will need to write the binding code yourself. The :ref:`next
269-
section <bindings>` explains how to do this for your own types.
272+
In general, you will need to write the binding code yourself. The previous
273+
section on :ref:`binding custom types <binding_types>` showed an example of
274+
such a type binding.
270275

271276
.. _wrappers:
272277

0 commit comments

Comments
 (0)