Using-declaration
Introduces a name that is defined elsewhere into the declarative region where this using-declaration appears. See using enum and (since C++20)using namespace for other related declarations.
using typename (optional) nested-name-specifier unqualified-id ;
|
(until C++17) | ||||||||
using declarator-list ;
|
(since C++17) | ||||||||
typename
|
- | the keyword typename may be used as necessary to resolve dependent names, when the using-declaration introduces a member type from a base class into a class template |
nested-name-specifier | - | a sequence of names and scope resolution operators :: , ending with a scope resolution operator. A single :: refers to the global namespace.
|
unqualified-id | - | an id-expression |
declarator-list | - | comma-separated list of one or more declarators of the typename (optional) nested-name-specifier unqualified-id. Some or all of the declarators may be followed by an ellipsis ... to indicate pack expansion
|
Explanation
Using-declarations can be used to introduce namespace members into other namespaces and block scopes, or to introduce base class members into derived class definitions, or to introduce enumerators into namespaces, block, and class scopes(since C++20).
A using-declaration with more than one using-declarator is equivalent to a corresponding sequence of using-declarations with one using-declarator. |
(since C++17) |
In namespace and block scope
Using-declarations introduce a member of another namespace into the current namespace or block scope.
#include <iostream> #include <string> using std::string; int main() { string str = "Example"; using std::cout; cout << str; }
See namespace for details.
In class definition
Using-declaration introduces a member of a base class into the derived class definition, such as to expose a protected member of base as public member of derived. In this case, nested-name-specifier must name a base class of the one being defined. If the name is the name of an overloaded member function of the base class, all base class member functions with that name are introduced. If the derived class already has a member with the same name, parameter list, and qualifications, the derived class member hides or overrides (doesn't conflict with) the member that is introduced from the base class.
#include <iostream> struct B { virtual void f(int) { std::cout << "B::f\n"; } void g(char) { std::cout << "B::g\n"; } void h(int) { std::cout << "B::h\n"; } protected: int m; // B::m is protected typedef int value_type; }; struct D : B { using B::m; // D::m is public using B::value_type; // D::value_type is public using B::f; void f(int) override { std::cout << "D::f\n"; } // D::f(int) overrides B::f(int) using B::g; void g(int) { std::cout << "D::g\n"; } // both g(int) and g(char) are visible using B::h; void h(int) { std::cout << "D::h\n"; } // D::h(int) hides B::h(int) }; int main() { D d; B& b = d; // b.m = 2; // Error: B::m is protected d.m = 1; // protected B::m is accessible as public D::m b.f(1); // calls derived f() d.f(1); // calls derived f() std::cout << "----------\n"; d.g(1); // calls derived g(int) d.g('a'); // calls base g(char), exposed via using B::g; std::cout << "----------\n"; b.h(1); // calls base h() d.h(1); // calls derived h() }
Output:
D::f D::f ---------- D::g B::g ---------- B::h D::h
Inheriting constructorsIf the using-declaration refers to a constructor of a direct base of the class being defined (e.g. using Base::Base;), all constructors of that base (ignoring member access) are made visible to overload resolution when initializing the derived class. If overload resolution selects an inherited constructor, it is accessible if it would be accessible when used to construct an object of the corresponding base class: the accessibility of the using-declaration that introduced it is ignored. If overload resolution selects one of the inherited constructors when initializing an object of such derived class, then the struct B1 { B1(int, ...) {} }; struct B2 { B2(double) {} }; int get(); struct D1 : B1 { using B1::B1; // inherits B1(int, ...) int x; int y = get(); }; void test() { D1 d(2, 3, 4); // OK: B1 is initialized by calling B1(2, 3, 4), // then d.x is default-initialized (no initialization is performed), // then d.y is initialized by calling get() D1 e; // Error: D1 has no default constructor } struct D2 : B2 { using B2::B2; // inherits B2(double) B1 b; }; D2 f(1.0); // error: B1 has no default constructor struct W { W(int); }; struct X : virtual W { using W::W; // inherits W(int) X() = delete; }; struct Y : X { using X::X; }; struct Z : Y, virtual W { using Y::Y; }; Z z(0); // OK: initialization of Y does not invoke default constructor of X If the struct V { V() = default; V(int); }; struct Q { Q(); }; struct A : virtual V, Q { using V::V; A() = delete; }; int bar() { return 42; } struct B : A { B() : A(bar()) {} // OK }; struct C : B {}; void foo() { C c; // “bar” is not invoked, because the V subobject // is not initialized as part of B // (the V subobject is initialized as part of C, // because “c” is the most derived object) } If the constructor was inherited from multiple base class subobjects of type struct A { A(int); }; struct B : A { using A::A; }; struct C1 : B { using B::B; }; struct C2 : B { using B::B; }; struct D1 : C1, C2 { using C1::C1; using C2::C2; }; D1 d1(0); // ill-formed: constructor inherited from different B base subobjects struct V1 : virtual B { using B::B; }; struct V2 : virtual B { using B::B; }; struct D2 : V1, V2 { using V1::V1; using V2::V2; }; D2 d2(0); // OK: there is only one B subobject. // This initializes the virtual B base class, // which initializes the A base class // then initializes the V1 and V2 base classes // as if by a defaulted default constructor As with using-declarations for any other non-static member functions, if an inherited constructor matches the signature of one of the constructors of struct B1 { B1(int); }; struct B2 { B2(int); }; struct D2 : B1, B2 { using B1::B1; using B2::B2; D2(int); // OK: D2::D2(int) hides both B1::B1(int) and B2::B2(int) }; D2 d2(0); // calls D2::D2(int) Within a templated class, if a using-declaration refers to a dependent name, it is considered to name a constructor if the nested-name-specifier has a terminal name that is the same as the unqualified-id. template<class T> struct A : T { using T::T; // OK, inherits constructors of T }; template<class T, class U> struct B : T, A<U> { using A<U>::A; // OK, inherits constructors of A<U> using T::A; // does not inherit constructor of T // even though T may be a specialization of A<> }; |
(since C++11) |
Introducing scoped enumeratorsIn addition to members of another namespace and members of base classes, using-declaration can also introduce enumerators of enumerations into namespace, block, and class scopes. A using-declaration can also be used with unscoped enumerators. enum class button { up, down }; struct S { using button::up; button b = up; // OK }; using button::down; constexpr button non_up = down; // OK constexpr auto get_button(bool is_up) { using button::up, button::down; return is_up ? up : down; // OK } enum unscoped { val }; using unscoped::val; // OK, though needless |
(since C++20) |
Notes
Only the name explicitly mentioned in the using-declaration is transferred into the declarative scope: in particular, enumerators are not transferred when the enumeration type name is using-declared.
A using-declaration cannot refer to a namespace, to a scoped enumerator(until C++20), to a destructor of a base class or to a specialization of a member template for a user-defined conversion function.
A using-declaration cannot name a member template specialization (template-id is not permitted by the grammar):
struct B { template<class T> void f(); }; struct D : B { using B::f; // OK: names a template // using B::f<int>; // Error: names a template specialization void g() { f<int>(); } };
A using-declaration also can't be used to introduce the name of a dependent member template as a template-name (the template
disambiguator for dependent names is not permitted).
template<class X> struct B { template<class T> void f(T); }; template<class Y> struct D : B<Y> { // using B<Y>::template f; // Error: disambiguator not allowed using B<Y>::f; // compiles, but f is not a template-name void g() { // f<int>(0); // Error: f is not known to be a template name, // so < does not start a template argument list f(0); // OK } };
If a using-declaration brings the base class assignment operator into derived class, whose signature happens to match the derived class's copy-assignment or move-assignment operator, that operator is hidden by the implicitly-declared copy/move assignment operator of the derived class. Same applies to a using-declaration that inherits a base class constructor that happens to match the derived class copy/move constructor(since C++11).
The semantics of inheriting constructors were retroactively changed by a defect report against C++11. Previously, an inheriting constructor declaration caused a set of synthesized constructor declarations to be injected into the derived class, which caused redundant argument copies/moves, had problematic interactions with some forms of SFINAE, and in some cases can be unimplementable on major ABIs. Older compilers may still implement the previous semantics.
|
(since C++11) |
Pack expansions in using-declarations make it possible to form a class that exposes overloaded members of variadic bases without recursion: template<typename... Ts> struct Overloader : Ts... { using Ts::operator()...; // exposes operator() from every base }; template<typename... T> Overloader(T...) -> Overloader<T...>; // C++17 deduction guide, not needed in C++20 int main() { auto o = Overloader{ [] (auto const& a) {std::cout << a;}, [] (float f) {std::cout << std::setprecision(3) << f;} }; } |
(since C++17) |
Feature-test macro | Value | Std | Feature |
---|---|---|---|
__cpp_inheriting_constructors |
200802L | (C++11) | Inheriting constructors |
201511L | (C++11) (DR) |
Rewording inheriting constructors | |
__cpp_variadic_using |
201611L | (C++17) | Pack expansions in using -declarations
|
Keywords
Defect reports
The following behavior-changing defect reports were applied retroactively to previously published C++ standards.
DR | Applied to | Behavior as published | Correct behavior |
---|---|---|---|
CWG 258 | C++98 | a non-const member function of a derived class can override and/or hide a const member function of its base |
overriding and hiding also require cv-qualifications to be the same |
CWG 1738 | C++11 | it was not clear whether it is permitted to explicitly instantiate or explicitly specialize specializations of inheriting constructor templates |
prohibited |
CWG 2504 | C++11 | the behavior of inheriting constructors from virtual base classes was unclear |
made clear |
P0136R1 | C++11 | inheriting constructor declaration injects additional constructors in the derived class |
causes base class constructors to be found by name lookup |
- References
References
- C++23 standard (ISO/IEC 14882:2024):
- 9.9 The
using
declaration [namespace.udecl]
- 9.9 The
- C++20 standard (ISO/IEC 14882:2020):
- 9.9 The
using
declaration [namespace.udecl]
- 9.9 The
- C++17 standard (ISO/IEC 14882:2017):
- 10.3.3 The
using
declaration [namespace.udecl]
- 10.3.3 The
- C++14 standard (ISO/IEC 14882:2014):
- 7.3.3 The
using
declaration [namespace.udecl]
- 7.3.3 The
- C++11 standard (ISO/IEC 14882:2011):
- 7.3.3 The
using
declaration [namespace.udecl]
- 7.3.3 The
- C++03 standard (ISO/IEC 14882:2003):
- 7.3.3 The
using
declaration [namespace.udecl]
- 7.3.3 The
- C++98 standard (ISO/IEC 14882:1998):
- 7.3.3 The
using
declaration [namespace.udecl]
- 7.3.3 The