Search

2.7 — Forward declarations and definitions

Take a look at this seemingly innocent sample program:

You would expect this program to produce the result:

The sum of 3 and 4 is: 7

But in fact, it doesn’t compile at all! Visual Studio produces the following compile error:

add.cpp(5) : error C3861: 'add': identifier not found

The reason this program doesn’t compile is because the compiler compiles the contents of code files sequentially. When the compiler reaches the function call to add on line 5 of main, it doesn’t know what add is, because we haven’t defined add until line 9! That produces the error, identifier not found.

Older versions of Visual Studio would produce an additional error:

add.cpp(9) : error C2365: 'add' : redefinition; previous definition was 'formerly unknown identifier'

This is somewhat misleading, given that add wasn’t ever defined in the first place. Despite this, it’s useful to generally note that it is fairly common for a single error to produce many redundant or related errors or warnings.

Best practice

When addressing compile errors in your programs, always resolve the first error produced first and then compile again.

To fix this problem, we need to address the fact that the compiler doesn’t know what add is. There are two common ways to address the issue.

Option 1: Reorder the function calls

One way to address the issue is to reorder the function calls so add is defined before main:

That way, by the time main calls add, the compiler will already know what add is. Because this is such a simple program, this change is relatively easy to do. However, in a larger program, it can be tedious trying to figure out which functions call which other functions (and in what order) so they can be declared sequentially.

Furthermore, this option is not always possible. Let’s say we’re writing a program that has two functions A and B. If function A calls function B, and function B calls function A, then there’s no way to order the functions in a way that will make the compiler happy. If you define A first, the compiler will complain it doesn’t know what B is. If you define B first, the compiler will complain that it doesn’t know what A is.

Option 2: Use a forward declaration

We can also fix this by using a forward declaration.

A forward declaration allows us to tell the compiler about the existence of an identifier before actually defining the identifier.

In the case of functions, this allows us to tell the compiler about the existence of a function before we define the function’s body. This way, when the compiler encounters a call to the function, it’ll understand that we’re making a function call, and can check to ensure we’re calling the function correctly, even if it doesn’t yet know how or where the function is defined.

To write a forward declaration for a function, we use a declaration statement called a function prototype. The function prototype consists of the function’s return type, name, parameters, but no function body (the curly braces and everything in between them), terminated with a semicolon.

Here’s a function prototype for the add function:

Now, here’s our original program that didn’t compile, using a function prototype as a forward declaration for function add:

Now when the compiler reaches the call to add in main, it will know what add looks like (a function that takes two integer parameters and returns an integer), and it won’t complain.

It is worth noting that function prototypes do not need to specify the names of the parameters. In the above code, you can also forward declare your function like this:

However, we prefer to name our parameters (using the same names as the actual function), because it allows you to understand what the function parameters are just by looking at the prototype. Otherwise, you’ll have to locate the function definition.

Best practice

When defining function prototypes, keep the parameter names. You can easily create forward declarations by using copy/paste on your function declaration. Don’t forget the semicolon on the end.

Forgetting the function body

New programmers often wonder what happens if they forward declare a function but do not define it.

The answer is: it depends. If a forward declaration is made, but the function is never called, the program will compile and run fine. However, if a forward declaration is made and the function is called, but the program never defines the function, the program will compile okay, but the linker will complain that it can’t resolve the function call.

Consider the following program:

In this program, we forward declare add, and we call add, but we never define add anywhere. When we try and compile this program, Visual Studio produces the following message:

Compiling...
add.cpp
Linking...
add.obj : error LNK2001: unresolved external symbol "int __cdecl add(int,int)" (?add@@YAHHH@Z)
add.exe : fatal error LNK1120: 1 unresolved externals

As you can see, the program compiled okay, but it failed at the link stage because int add(int, int) was never defined.

Other types of forward declarations

Forward declarations are most often used with functions. However, forward declarations can also be used with other identifiers in C++, such as variables and user-defined types. Variables and user-defined types have a different syntax for forward declaration, so we’ll cover these in future lessons.

Declarations vs. definitions

In C++, you’ll often hear the words “declaration” and “definition” used, often interchangeably. What do they mean? You now have enough of a framework to understand the difference between the two.

A definition actually implements (for functions or types) or instantiates (for variables) the identifier. Here are some examples of definitions:

A definition is needed to satisfy the linker. If you use an identifier without providing a definition, the linker will error.

The one definition rule (or ODR for short) is a well-known rule in C++. The ODR has three parts:

  1. Within a given file, a function, object, type, or template can only have one definition.
  2. Within a given program, an object or normal function can only have one definition. This distinction is made because programs can have more than one file (we’ll cover this in the next lesson).
  3. Within a given program, types, template functions, and inline functions can have multiple definitions so long as they are identical. We haven’t covered what most of these things are yet, so don’t worry about this for now -- we’ll bring it back up when it’s relevant.

Violating part 1 of the ODR will cause a compile to issue a redefinition error. Violating ODR parts 2 or 3 will cause the linker to issue a redefinition error. Here’s an example of a violation of part 1:

Because the above program violates ODR part 1, this causes the Visual Studio compiler to issue the following compile errors:

project3.cpp(9): error C2084: function 'int add(int,int)' already has a body
project3.cpp(3): note: see previous definition of 'add'
project3.cpp(16): error C2086: 'int x': redefinition
project3.cpp(15): note: see declaration of 'x'

A declaration is a statement that tells the compiler about the existence of an identifier and its type information. Here are some examples of declarations:

A declaration is all that is needed to satisfy the compiler. This is why we can use a forward declaration to tell the compiler about an identifier that isn’t actually defined until later.

In C++, all definitions also serve as declarations. This is why int x appears in our examples for both definitions and declarations. Since int x is a definition, it’s a declaration too. In most cases, a definition serves our purposes, as it satisfies both the compiler and linker. We only need to provide an explicit declaration when we want to use an identifier before it has been defined.

While it is true that all definitions are declarations, the converse is not true: all declarations are not definitions. An example of this is the function prototype -- it satisfies the compiler, but not the linker. These declarations that aren’t definitions are called pure declarations. Other types of pure declarations include forward declarations for variables and type declarations (you will encounter these in future lessons, no need to worry about them now).

The ODR doesn’t apply to pure declarations (it’s the one definition rule, not the one declaration rule), so you can have as many pure declarations for an identifier as you desire (although having more than one is redundant).

Author's note

In common language, the term “declaration” is typically used to mean “a pure declaration”, and “definition” is used to mean “a definition that also serves as a declaration”. Thus, we’d typically call int x; a definition, even though it both a definition and a declaration.

Quiz time

Question #1

What is a function prototype?

Show Solution

Question #2

What is a forward declaration?

Show Solution

Question #3

How do we declare a forward declaration for functions?

Show Solution

Question #4

Write the function prototype for this function (use the preferred form with names):

Show Solution

Question #5

For each of the following programs, state whether they fail to compile, fail to link, or compile and link. If you are not sure, try compiling them!

a)

Show Solution

b)

Show Solution

c)

Show Solution

d)

Show Solution


2.8 -- Programs with multiple code files
Index
2.6 -- Whitespace and basic formatting

236 comments to 2.7 — Forward declarations and definitions

  • Levi

    "New programmers often wonder what happens if forward declare a function but do not define it." idk if this is complete english haha

  • Ethan Smith

    What is the standard for placing in Forward declarations, should they be outside of any functions and at the start of the code outside of functions entirely?

  • Derick

    I find the section the one definition rule a bit confusing.

    The first part states: "Within a given file, an identifier can only have one definition.", but I can have two variables with the same name in the same file, as long as they're in different scopes.

    I guess the question is, what constitutes an identifier?

    • Alex

      I updated the section on the ODR to make it more accurate, as you are correct -- it's not the identifier that must be unique, it's the underlying things themselves (objects, functions, types, etc...) that must be unique.

      An identifier is just a name for something, and as you correctly note, identifiers don't need to be unique so long as they're in different scopes.

  • Ove

    Hello!
    In the section Function prototypes and forward declaration of functions you write this tip:
    Tip: You can easily create function prototypes by using copy/paste on your function declaration. Don’t forget the semicolon on the end.

    My understanding is that you meant to write the following:
    Tip: You can easily create function prototypes by using copy/paste on your function definition. Don’t forget the semicolon on the end.

    Is this correct or have I misunderstood something?
    Thanks for these awesome tutorials!

    • The declaration is the top of the function

      The definition includes the function body (the part inside the curly brackets)

      • Ove

        Aha! Ok, thanks.
        My understanding was that the function prototype and forward declaration was the same thing and that the actual function is the definition.

        Have I understood it rightly that there are two kinds of declarations?
        Forward declaration where you have the function prototype, and function declaration where you have the definition.

        • > My understanding was that the function prototype and forward declaration was the same thing and that the actual function is the definition.
          That's correct.

          > Have I understood it rightly that there are two kinds of declarations?
          You can either have the declaration alone (forward declaration/prototype) or the declaration along with the definition. I wouldn't call that two kinds of declaration since it's the same thing. Only what comes after the declaration is different (semicolon or body).

  • Aditya

    Hey Alex!
    Thanks for the correction.
    Now, I want this code to calculate in decimals. That is,if i say 35/45,i want  to get the answer in decimals.
    How do i do that?

    • Alex

      If you use type "double" instead of "int", you can use numbers with decimals (e.g. 1.45). Handling fractions (e.g. 35/45) is a lot more complicated and far beyond the scope of these introductory lessons.

      I do show examples of a Fraction class in chapter 8 or 9.

  • aditya

    What is wrong with this code?

  • Diego Sandoval

    You say that a function can have only one definition, yet this code compiles and executes correctly. both functions are called (depending on the arguments provided). I didn't understand the rule correctly? right?

    • nascardriver

      Hi Diego!

      Those are two different functions which share the same name (Function overloading).
      The one definition rule is only valid for the same function header (Same return-type, same name, same parameters).

      References
      * Lesson 7.6 - Function overloading

  • Hi, couldn't all the forward definitions just be put together in a header file such as <<forward.h>> and precompiled with a #include call at the top such as

    ?

    • nascardriver

      Hi Nigel!

      Yes, that's how it's usually done, because it allows usage of the functions from any file. Nothing is being compiled though, only the contents will be copied.

      • thanks, just getting my head around why we were looking at "global" scoped definitions in the program that just make the listing look untidy. Are headers covered later?

        • nascardriver

          Yep, starting with the next lesson.
          The code on learncpp is single-file in most cases, because it's easier to understand that way. In your projects, you should use multiple files.

  • jft

    This is always a definition as a memory location is allocated for x (which is uninitialized).

    This is a declaration as no memory is allocated for x and x here refers to x defined in another translation unit.

  • Zane

    Why does this work?

    I ask, because the definition of add() is after that of the body of main(). If the the program is read/ran sequentially, when would it ever encounter the definition? Shouldn't the definition come before main()? If not, I am missing something. As I understand it, at the return of a value for main() the program stops, never having made it to/read add().

    • nascardriver

      Hi Zane!

      Your compiler compiles all functions, no matter where they are. To call a function, all your compiler needs to know, is the function declaration, not the definition. After compilation the order of functions doesn't matter, because your computer can call functions above and below the calling function.

  • Deepesh Choudhary

    Okay, so a declaration is used just to satisfy the compiler.
    Does that mean that the declaration statement does not get converted to any machine code?
    Does that also mean that if main.o is reversed, there won't be any code/statements indicating such declaration?

    • nascardriver

      Hi Deepesh!

      Yes to both.
      The compiler will generate the function at address Y. Your computer isn't restricted to calling functions that have been declared before the current point of execution. It can just from address X to address Y without worries.

  • Pork and Beans

    Hello Alex!

    I hope you are gearing up for a stellar weekend.

    I also hope that you do, in fact, want me to point out any mistakes I see.

    In the paragraph above "Quiz", there is a mistake.

    "Others types of pure declarations..." should be "Other".

    I thought you would want to know.

    Thank you as always for your incredible work.

    Have a nice day.

  • Farzad.S

    I see, by Forward declarations and definitions, we can call the functions in any order we want. Am I right?

  • Farzad.S

    Hello, I did understand the Forward declarations and prototype. But I am curious why do we need it, we can just use function and keep calling it. I believe it makes the program more complicated.

    By the way, for ages, I was trying to learn C++ but couldn't find the right book or courses to clearly teaching me from scratch. Great job. thanks a lot

    • nascardriver

      Hi Farzad!

      > I believe it makes the program more complicated
      It sure does, but there are situations where it cannot be avoided.

      This program will just run an infinite loop, but there are cases in which a construction like this is required.

    • vbot

      My preferred programing-flow is linear from top to bottom, because I'm used to languages where the order of function definitions and calls doesn't matter:

      so this is where I see the need for forward declarations.

Leave a Comment

Put all code inside code tags: [code]your code here[/code]