You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
216 lines
9.7 KiB
HTML
216 lines
9.7 KiB
HTML
4 years ago
|
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
|
||
|
<html>
|
||
|
<!-- Copyright (C) 1988-2018 Free Software Foundation, Inc.
|
||
|
|
||
|
Permission is granted to copy, distribute and/or modify this document
|
||
|
under the terms of the GNU Free Documentation License, Version 1.3 or
|
||
|
any later version published by the Free Software Foundation; with the
|
||
|
Invariant Sections being "Funding Free Software", the Front-Cover
|
||
|
Texts being (a) (see below), and with the Back-Cover Texts being (b)
|
||
|
(see below). A copy of the license is included in the section entitled
|
||
|
"GNU Free Documentation License".
|
||
|
|
||
|
(a) The FSF's Front-Cover Text is:
|
||
|
|
||
|
A GNU Manual
|
||
|
|
||
|
(b) The FSF's Back-Cover Text is:
|
||
|
|
||
|
You have freedom to copy and modify this GNU Manual, like GNU
|
||
|
software. Copies published by the Free Software Foundation raise
|
||
|
funds for GNU development. -->
|
||
|
<!-- Created by GNU Texinfo 6.4, http://www.gnu.org/software/texinfo/ -->
|
||
|
<head>
|
||
|
<title>Name lookup (Using the GNU Compiler Collection (GCC))</title>
|
||
|
|
||
|
<meta name="description" content="Name lookup (Using the GNU Compiler Collection (GCC))">
|
||
|
<meta name="keywords" content="Name lookup (Using the GNU Compiler Collection (GCC))">
|
||
|
<meta name="resource-type" content="document">
|
||
|
<meta name="distribution" content="global">
|
||
|
<meta name="Generator" content="makeinfo">
|
||
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
|
||
|
<link href="index.html#Top" rel="start" title="Top">
|
||
|
<link href="Option-Index.html#Option-Index" rel="index" title="Option Index">
|
||
|
<link href="index.html#SEC_Contents" rel="contents" title="Table of Contents">
|
||
|
<link href="C_002b_002b-Misunderstandings.html#C_002b_002b-Misunderstandings" rel="up" title="C++ Misunderstandings">
|
||
|
<link href="Temporaries.html#Temporaries" rel="next" title="Temporaries">
|
||
|
<link href="Static-Definitions.html#Static-Definitions" rel="prev" title="Static Definitions">
|
||
|
<style type="text/css">
|
||
|
<!--
|
||
|
a.summary-letter {text-decoration: none}
|
||
|
blockquote.indentedblock {margin-right: 0em}
|
||
|
blockquote.smallindentedblock {margin-right: 0em; font-size: smaller}
|
||
|
blockquote.smallquotation {font-size: smaller}
|
||
|
div.display {margin-left: 3.2em}
|
||
|
div.example {margin-left: 3.2em}
|
||
|
div.lisp {margin-left: 3.2em}
|
||
|
div.smalldisplay {margin-left: 3.2em}
|
||
|
div.smallexample {margin-left: 3.2em}
|
||
|
div.smalllisp {margin-left: 3.2em}
|
||
|
kbd {font-style: oblique}
|
||
|
pre.display {font-family: inherit}
|
||
|
pre.format {font-family: inherit}
|
||
|
pre.menu-comment {font-family: serif}
|
||
|
pre.menu-preformatted {font-family: serif}
|
||
|
pre.smalldisplay {font-family: inherit; font-size: smaller}
|
||
|
pre.smallexample {font-size: smaller}
|
||
|
pre.smallformat {font-family: inherit; font-size: smaller}
|
||
|
pre.smalllisp {font-size: smaller}
|
||
|
span.nolinebreak {white-space: nowrap}
|
||
|
span.roman {font-family: initial; font-weight: normal}
|
||
|
span.sansserif {font-family: sans-serif; font-weight: normal}
|
||
|
ul.no-bullet {list-style: none}
|
||
|
-->
|
||
|
</style>
|
||
|
|
||
|
|
||
|
</head>
|
||
|
|
||
|
<body lang="en">
|
||
|
<a name="Name-lookup"></a>
|
||
|
<div class="header">
|
||
|
<p>
|
||
|
Next: <a href="Temporaries.html#Temporaries" accesskey="n" rel="next">Temporaries</a>, Previous: <a href="Static-Definitions.html#Static-Definitions" accesskey="p" rel="prev">Static Definitions</a>, Up: <a href="C_002b_002b-Misunderstandings.html#C_002b_002b-Misunderstandings" accesskey="u" rel="up">C++ Misunderstandings</a> [<a href="index.html#SEC_Contents" title="Table of contents" rel="contents">Contents</a>][<a href="Option-Index.html#Option-Index" title="Index" rel="index">Index</a>]</p>
|
||
|
</div>
|
||
|
<hr>
|
||
|
<a name="Name-Lookup_002c-Templates_002c-and-Accessing-Members-of-Base-Classes"></a>
|
||
|
<h4 class="subsection">13.7.2 Name Lookup, Templates, and Accessing Members of Base Classes</h4>
|
||
|
|
||
|
<a name="index-base-class-members"></a>
|
||
|
<a name="index-two_002dstage-name-lookup"></a>
|
||
|
<a name="index-dependent-name-lookup"></a>
|
||
|
|
||
|
<p>The C++ standard prescribes that all names that are not dependent on
|
||
|
template parameters are bound to their present definitions when parsing
|
||
|
a template function or class.<a name="DOCF5" href="#FOOT5"><sup>5</sup></a> Only names that are dependent are looked up at the point
|
||
|
of instantiation. For example, consider
|
||
|
</p>
|
||
|
<div class="smallexample">
|
||
|
<pre class="smallexample"> void foo(double);
|
||
|
|
||
|
struct A {
|
||
|
template <typename T>
|
||
|
void f () {
|
||
|
foo (1); // <span class="roman">1</span>
|
||
|
int i = N; // <span class="roman">2</span>
|
||
|
T t;
|
||
|
t.bar(); // <span class="roman">3</span>
|
||
|
foo (t); // <span class="roman">4</span>
|
||
|
}
|
||
|
|
||
|
static const int N;
|
||
|
};
|
||
|
</pre></div>
|
||
|
|
||
|
<p>Here, the names <code>foo</code> and <code>N</code> appear in a context that does
|
||
|
not depend on the type of <code>T</code>. The compiler will thus require that
|
||
|
they are defined in the context of use in the template, not only before
|
||
|
the point of instantiation, and will here use <code>::foo(double)</code> and
|
||
|
<code>A::N</code>, respectively. In particular, it will convert the integer
|
||
|
value to a <code>double</code> when passing it to <code>::foo(double)</code>.
|
||
|
</p>
|
||
|
<p>Conversely, <code>bar</code> and the call to <code>foo</code> in the fourth marked
|
||
|
line are used in contexts that do depend on the type of <code>T</code>, so
|
||
|
they are only looked up at the point of instantiation, and you can
|
||
|
provide declarations for them after declaring the template, but before
|
||
|
instantiating it. In particular, if you instantiate <code>A::f<int></code>,
|
||
|
the last line will call an overloaded <code>::foo(int)</code> if one was
|
||
|
provided, even if after the declaration of <code>struct A</code>.
|
||
|
</p>
|
||
|
<p>This distinction between lookup of dependent and non-dependent names is
|
||
|
called two-stage (or dependent) name lookup. G++ implements it
|
||
|
since version 3.4.
|
||
|
</p>
|
||
|
<p>Two-stage name lookup sometimes leads to situations with behavior
|
||
|
different from non-template codes. The most common is probably this:
|
||
|
</p>
|
||
|
<div class="smallexample">
|
||
|
<pre class="smallexample"> template <typename T> struct Base {
|
||
|
int i;
|
||
|
};
|
||
|
|
||
|
template <typename T> struct Derived : public Base<T> {
|
||
|
int get_i() { return i; }
|
||
|
};
|
||
|
</pre></div>
|
||
|
|
||
|
<p>In <code>get_i()</code>, <code>i</code> is not used in a dependent context, so the
|
||
|
compiler will look for a name declared at the enclosing namespace scope
|
||
|
(which is the global scope here). It will not look into the base class,
|
||
|
since that is dependent and you may declare specializations of
|
||
|
<code>Base</code> even after declaring <code>Derived</code>, so the compiler cannot
|
||
|
really know what <code>i</code> would refer to. If there is no global
|
||
|
variable <code>i</code>, then you will get an error message.
|
||
|
</p>
|
||
|
<p>In order to make it clear that you want the member of the base class,
|
||
|
you need to defer lookup until instantiation time, at which the base
|
||
|
class is known. For this, you need to access <code>i</code> in a dependent
|
||
|
context, by either using <code>this->i</code> (remember that <code>this</code> is of
|
||
|
type <code>Derived<T>*</code>, so is obviously dependent), or using
|
||
|
<code>Base<T>::i</code>. Alternatively, <code>Base<T>::i</code> might be brought
|
||
|
into scope by a <code>using</code>-declaration.
|
||
|
</p>
|
||
|
<p>Another, similar example involves calling member functions of a base
|
||
|
class:
|
||
|
</p>
|
||
|
<div class="smallexample">
|
||
|
<pre class="smallexample"> template <typename T> struct Base {
|
||
|
int f();
|
||
|
};
|
||
|
|
||
|
template <typename T> struct Derived : Base<T> {
|
||
|
int g() { return f(); };
|
||
|
};
|
||
|
</pre></div>
|
||
|
|
||
|
<p>Again, the call to <code>f()</code> is not dependent on template arguments
|
||
|
(there are no arguments that depend on the type <code>T</code>, and it is also
|
||
|
not otherwise specified that the call should be in a dependent context).
|
||
|
Thus a global declaration of such a function must be available, since
|
||
|
the one in the base class is not visible until instantiation time. The
|
||
|
compiler will consequently produce the following error message:
|
||
|
</p>
|
||
|
<div class="smallexample">
|
||
|
<pre class="smallexample"> x.cc: In member function `int Derived<T>::g()':
|
||
|
x.cc:6: error: there are no arguments to `f' that depend on a template
|
||
|
parameter, so a declaration of `f' must be available
|
||
|
x.cc:6: error: (if you use `-fpermissive', G++ will accept your code, but
|
||
|
allowing the use of an undeclared name is deprecated)
|
||
|
</pre></div>
|
||
|
|
||
|
<p>To make the code valid either use <code>this->f()</code>, or
|
||
|
<code>Base<T>::f()</code>. Using the <samp>-fpermissive</samp> flag will also let
|
||
|
the compiler accept the code, by marking all function calls for which no
|
||
|
declaration is visible at the time of definition of the template for
|
||
|
later lookup at instantiation time, as if it were a dependent call.
|
||
|
We do not recommend using <samp>-fpermissive</samp> to work around invalid
|
||
|
code, and it will also only catch cases where functions in base classes
|
||
|
are called, not where variables in base classes are used (as in the
|
||
|
example above).
|
||
|
</p>
|
||
|
<p>Note that some compilers (including G++ versions prior to 3.4) get these
|
||
|
examples wrong and accept above code without an error. Those compilers
|
||
|
do not implement two-stage name lookup correctly.
|
||
|
</p>
|
||
|
|
||
|
<div class="footnote">
|
||
|
<hr>
|
||
|
<h4 class="footnotes-heading">Footnotes</h4>
|
||
|
|
||
|
<h3><a name="FOOT5" href="#DOCF5">(5)</a></h3>
|
||
|
<p>The C++ standard just uses the
|
||
|
term “dependent” for names that depend on the type or value of
|
||
|
template parameters. This shorter term will also be used in the rest of
|
||
|
this section.</p>
|
||
|
</div>
|
||
|
<hr>
|
||
|
<div class="header">
|
||
|
<p>
|
||
|
Next: <a href="Temporaries.html#Temporaries" accesskey="n" rel="next">Temporaries</a>, Previous: <a href="Static-Definitions.html#Static-Definitions" accesskey="p" rel="prev">Static Definitions</a>, Up: <a href="C_002b_002b-Misunderstandings.html#C_002b_002b-Misunderstandings" accesskey="u" rel="up">C++ Misunderstandings</a> [<a href="index.html#SEC_Contents" title="Table of contents" rel="contents">Contents</a>][<a href="Option-Index.html#Option-Index" title="Index" rel="index">Index</a>]</p>
|
||
|
</div>
|
||
|
|
||
|
|
||
|
|
||
|
</body>
|
||
|
</html>
|