.. ot-topic:: cxx11.new_language_features.concepts.writing_concepts :dependencies: cxx11.new_language_features.concepts.intro .. include:: Writing Your Own Concepts ========================= .. contents:: :local: Starting Point: Good Old Function --------------------------------- .. toctree:: :hidden: writing-concepts/example-1-good-old-func .. sidebar:: **Source** * :doc:`writing-concepts/example-1-good-old-func` * N-dimensional hypotenuse * ``std::vector`` * Using index based iteration, only too keep requirements to a minimum * Requirements * ``.size()`` * ``.operator[]()`` .. code-block:: c++ double hypotenuse(const std::vector& v) { double sumsq = 0; for (size_t i=0; i concept has_size = requires(V v) { v.size(); // <--- compiles }; * Concept usage: good old full explicit template .. code-block:: c++ template requires has_size double hypotenuse(const V& v) { /*...*/ } * Concept usage: good old full explicit template (after declaration) .. code-block:: c++ template double hypotenuse(const V& v) requires has_size { /*...*/ } * Concept usage: abbreviated function templates .. code-block:: c++ double hypotenuse(const has_size auto& v) { /*...*/ } Concept: ``index_returns_double`` --------------------------------- .. toctree:: :hidden: writing-concepts/example-5-concept-index-ret-double .. sidebar:: **Source** * :doc:`writing-concepts/example-5-concept-index-ret-double` * Hmm ... what if elements are not ``double``? * |longrightarrow| Somebody could use ``hypotenuse()`` on something that has ``int`` coordinates * Could we check this? *Ruin the whole thing ...* * Modify object initialization to take real double values, e.g. ``{3.5L, 4.5L}`` * |longrightarrow| Result not straight 5 anymore * Modify ``point2d::operator[]()`` to return ``int`` * |longrightarrow| Result straight 5 again *Concept* ``index_returns_double`` .. sidebar:: **Documentation** * `std::same_as `__ * `std::commone_reference_with `__ * : `Standard library header `__ * `std::vector `__ * First use ``std::same_as`` * |longrightarrow| Constraint check fails: ``std::vector::operator[]()`` is *not* same as ``double`` (rather, it returns ``double&``) * Solution: ``std::commone_reference_with`` .. code-block:: c++ template concept index_returns_double = requires(V v) { { v[0] } -> std::common_reference_with; }; * Argh: checking multiple constraints is not possible with abbreviated function templates * |longrightarrow| fall back to ordinary template syntax, and its ``requires`` clause