2018-02-28 22:47:20 +08:00
|
|
|
.. title:: clang-tidy - bugprone-forwarding-reference-overload
|
2017-04-06 17:56:42 +08:00
|
|
|
|
2018-02-28 22:47:20 +08:00
|
|
|
bugprone-forwarding-reference-overload
|
|
|
|
======================================
|
2017-04-06 17:56:42 +08:00
|
|
|
|
|
|
|
The check looks for perfect forwarding constructors that can hide copy or move
|
2017-06-30 06:26:06 +08:00
|
|
|
constructors. If a non const lvalue reference is passed to the constructor, the
|
2017-04-06 17:56:42 +08:00
|
|
|
forwarding reference parameter will be a better match than the const reference
|
|
|
|
parameter of the copy constructor, so the perfect forwarding constructor will be
|
|
|
|
called, which can be confusing.
|
|
|
|
For detailed description of this issue see: Scott Meyers, Effective Modern C++,
|
|
|
|
Item 26.
|
|
|
|
|
|
|
|
Consider the following example:
|
|
|
|
|
2018-11-06 06:21:27 +08:00
|
|
|
.. code-block:: c++
|
2017-06-30 06:26:06 +08:00
|
|
|
|
2017-04-06 17:56:42 +08:00
|
|
|
class Person {
|
|
|
|
public:
|
|
|
|
// C1: perfect forwarding ctor
|
|
|
|
template<typename T>
|
|
|
|
explicit Person(T&& n) {}
|
|
|
|
|
|
|
|
// C2: perfect forwarding ctor with parameter default value
|
|
|
|
template<typename T>
|
|
|
|
explicit Person(T&& n, int x = 1) {}
|
2017-06-30 06:26:06 +08:00
|
|
|
|
2017-04-06 17:56:42 +08:00
|
|
|
// C3: perfect forwarding ctor guarded with enable_if
|
|
|
|
template<typename T, typename X = enable_if_t<is_special<T>,void>>
|
|
|
|
explicit Person(T&& n) {}
|
2017-06-30 06:26:06 +08:00
|
|
|
|
2017-04-06 17:56:42 +08:00
|
|
|
// (possibly compiler generated) copy ctor
|
2017-06-30 06:26:06 +08:00
|
|
|
Person(const Person& rhs);
|
2017-04-06 17:56:42 +08:00
|
|
|
};
|
2017-06-30 06:26:06 +08:00
|
|
|
|
2017-04-06 17:56:42 +08:00
|
|
|
The check warns for constructors C1 and C2, because those can hide copy and move
|
|
|
|
constructors. We suppress warnings if the copy and the move constructors are both
|
|
|
|
disabled (deleted or private), because there is nothing the perfect forwarding
|
|
|
|
constructor could hide in this case. We also suppress warnings for constructors
|
|
|
|
like C3 that are guarded with an enable_if, assuming the programmer was aware of
|
|
|
|
the possible hiding.
|
|
|
|
|
|
|
|
Background
|
|
|
|
----------
|
|
|
|
|
|
|
|
For deciding whether a constructor is guarded with enable_if, we consider the
|
2017-06-30 06:26:06 +08:00
|
|
|
default values of the type parameters and the types of the constructor
|
2017-04-06 17:56:42 +08:00
|
|
|
parameters. If any part of these types is std::enable_if or std::enable_if_t, we
|
|
|
|
assume the constructor is guarded.
|