C++ Syntax Cheat Sheet
C++ Syntax Cheat Sheet
C++ Syntax Cheat Sheet
Preface
Since the C++ language varies so heavily between versions (e.g. C++0x, C++11, C++17, etc.), I will
preface this cheat sheet by saying that the majority of the examples here target C++0x or c++11, as
those are the versions that I am most familiar with. I come from the aerospace industry (embedded
flight software) in which we purposefully don't use the latest technologies for safety reasons, so most
of the code I write is in C++0x and sometimes C++11. Nevertheless, the basic concepts of C++ and
object oriented programming still generally apply to both past and future versions of the language.
Table of Contents
C++ Syntax Cheat Sheet
Table of Contents
1.0 C++ Classes
1.1 Class Syntax
1.1.1 Class Declaration ( .h file)
1.1.2 Class Definition ( .cpp file)
1.1.3 Class Utilization (Another .cpp file)
1.1.4 Getters and Setters
1.2 Inheritance
1.2.1 Rectangle Declaration ( .h file)
1.2.2 Rectangle Definition ( .cpp file)
1.2.3 Rectangle Utilization (Another .cpp file)
1.3 Class Polymorphism
1.3.1 Motivation
1.3.2 Virtual Methods
1.4 Special Methods (Constructor, Destructor, ...)
1.4.1 Constructor and Destructor
1.4.1.1 Use of explicit in Constructors
1.4.1.2 Member Initializer List
1.4.2. new and delete
1.4.3. Copy Constructor and Copy Assignment
1.4.4. Move Constructor and Move Assignment
1.5 Operator Overloading
1.6 Templates
2.0 General C++ Syntax
2.1 Namespaces
2.2 References/Pointers
2.3 Keywords
2.3.1 General keywords
2.3.2 Storage class specifiers
2.3.3 const and dynamic Cast Conversion
2.4 Preprocessor Tokens
2.5 Strings
2.6 Iterators
2.7 Exceptions
2.8 Lambdas
Here's a simple class representing a polygon, a shape with any number of sides.
The class declaration typically goes in the header file, which has the extension .h (or, less commonly,
.hpp to distinguish from C headers). The declaration gives the class name, any classes it may extend,
declares the members and methods, and declares which members/methods are public, private, or
protected. You can think of the declaration as sort of saying: "there will be a thing and here's how it
will look like". The declaration is used to inform the compiler about the future essence and use of a
particular symbol.
// File: polygon.h
#include <string>
class Polygon {
// Private members and methods are only accessible via methods in the class definition
private:
int num_sides; // Number of sides
// Protected members and methods are only accessible in the class definition or by classes who
protected:
std::string name; // Name of the polygon
// Public members and methods are accessible to anyone who creates an instance of the class
public:
// Constructors
Polygon(const int num_sides, const std::string & name); // <--- This constructor takes the
// Getters and Setters
int GetNumSides(void) const;
void SetNumSides(const int num_sides);
The class definition typically goes in the .cpp file. The definition extends the declaration by providing
an actual implementation of whatever it is that you're building. Continuing the example from the
declaration, the definition can be thought of as saying: "Right, that thing I told you briefly about
earlier? Here's how it actually functions". The definition thus provides the compileable
implementation.
// File: polygon.cpp
// Constructor
// You must scope the method definitions with the class name (Polygon::)
// Also, see the section on the 'explicit' keyword for a warning about constructors with exact
Polygon::Polygon(const int num_sides, const std::string & name) {
this->num_sides = num_sides; // 'this' is a pointer to the instance of the class. Membe
this->name = name; // In this case you need to use 'this->...' to avoid shadowing
}
The getters and setters here don't do much, but you could imagine limiting the number of sides such
that it must have at least 3 sides to be a useful polygon, in which case you could enforce that in
Polygon::SetNumSides() . Of course, you'd also need to modify the constructor, which could then call
SetNumSides() instead of setting the variable directly.
NOTE: Regarding the use of this-> in a class definition, there are places where it's strictly
necessary for readability, e.g. when your method parameter shares the exact same name as a
member variable, you use this-> to avoid what's called shadowing. However, some prefer to
always use this-> explicitly regardless of whether it's necessary.
// File: main.cpp
#include <string>
#include <iostream>
A shortcut often used for Getters/Setters is to define them in the class declaration ( .h ) file as follows:
// File: car.h
#include <string>
class Car {
private:
int year;
std::string make;
public:
int GetYear(void) const { return this->year; }
void SetYear(const int year) { this->year = year; }
std::string & GetMake(void) const { return this->make; }
void SetMake(const std::string & make) { this->make = make; }
};
This is often used for very basic getters and setters, and also for basic constructors. In contrast, you'll
nearly always find more complex methods defined in the .cpp file. One exception to this is with class
templates, in which the entire templated class declaration and definition must reside in the header
file.
Another important consideration: If you have getters and setters for all of your members, you may
want to reconsider the design of your class. Sometimes having getters and setters for every member
is indicative of poor planning of the class design and interface. In particular, setters should be used
more thoughtfully. Could a variable be set once in the constructor and left constant thereafter? Does
it need to be modified at all? Is it set somewhere else in another method, perhaps even indirectly?
1.2 Inheritance
A class can extend another class, meaning that the new class inherits all of the data from the other
class, and can also override its methods, add new members, etc. Inheritance is the key feature
required for polymorphism.
It is important to note that this feature is often overused by beginners and sometimes unnecessary
hierarchies are created, adding to the overally complexity. There are some good alternatives such as
composition and aggregation, although, of course, sometimes inheritance is exactly what is needed.
Example: the class Rectangle can inherit from the class Polygon . You would then say that a
Rectangle extends from a Polygon , or that class Rectangle is a sub-class of Polygon . In plain
English, this means that a Rectangle is a more specialized version of a Polygon . Thus, all rectangles
are polygons, but not all polygons are rectangles.
// File: rectangle.h
#include <string> // <--- Explicitly include the string header, even though polygon.h al
#include "polygon.h" // <--- You must include the declaration in order to extend the class
// We extend from Polygon by using the colon (:) and specifying which type of inheritance
// will be used (public inheritance, in this case)
// <--- NOTE: The member variables 'num_sides' and 'name' are already inherited from Polyg
// it's as if we sort of get them for free, since we are a sub-class
public:
// Constructors
explicit Rectangle(const std::string &name);
Rectangle(const std::string &name, const int length, const int width);
// <--- NOTE: Again, the getters/setters for 'num_sides' and 'name' are already inherited
// Other Methods
const int Area(void) const;
};
NOTE: The inheritance access specifier ( public , protected , or private ) is used to determine
the type of inheritance. If this is omitted then private inheritance is used by default. Public
inheritance is by far the most common type of inheritance.
// File: rectangle.cpp
#include "rectangle.h" // <--- Only need to include 'Rectangle', since 'Polygon' is included
// This constructor calls the superclass (Polygon) constructor and sets the name and number of
Rectangle::Rectangle(const std::string &name, const int length, const int width) : Polygon(4,
this->length = length;
this->width = width;
}
// This constructor calls the superclass (Polygon) constructor, but sets the length and width
// The explicit keyword is used to restrict the use of the constructor. See section below for
explicit Rectangle::Rectangle(const std::string &name) : Polygon(4, name) {
this->length = 1;
this->width = 1;
}
// File: main.cpp
#include <iostream>
#include "Rectangle.h"
make certain guarantees about which methods/variables are available for use. By adhering to this
common interface, one can use a pointer to an object of the base interface type to call the methods
of any number of extending classes. Using polymorphism one can say "I don't care what type this
really is; I know it implements Foo() and Bar() because it inherits from this interface", which is a
pretty nifty feature.
The virtual keyword is used to ensure runtime polymorphism for class methods. Additionally, an
overriding method can be forced by the compiler by not providing a default implementation in the
interface, which is done by setting the method to = 0 , as will be shown later.
1.3.1 Motivation
Let's consider a similar class hierarchy using shapes as previously discussed. Considering a shape to
be any 3 or more sided polygon from which we can compute certain attributes (like the shape's area),
let's extend from it to create a rectangle class from which we can set the length/width and a circle
class in which you can set the radius. In both cases, we want to be able to compute the area of the
shape. This is a key observation that we will expand upon later.
// File: shape.h
#include <cmath> // needed for M_PI constant
class Shape {
// We'll leave Shape empty for now... not very interesting yet
};
public:
// Constructor using a member initializer list instead of assignment in the method body
Rectangle(const double w, const double l) : width(w), length(l) {}
public:
explicit Circle(double r) : radius(r) {}
NOTE: As shown here, you can put multiple classes in a single header, although in practice unless
you have a good reason for doing so it's probably best to use a separate header file per class.
NOTE: I'm not using default value initialization for member variables (i.e. double length = 0; )
and I'm using parentheses () instead of braces {} for the initializer list since older compilers
(pre-C++11) may not support the new syntax.
So, we have our two classes, Rectangle and Circle , but in this case inheriting from Shape isn't
really buying us anything. To make use of polymorphism we need to pull the common Area()
method into the base class as follows, by using virtual methods.
Imagine you want to have a pointer to a shape with which you want to compute the area of that
shape. For example, maybe you want to hold shapes in some sort of data structure, but you don't
want to limit yourself to just rectangles or just circles; you want to support all objects that call
themselves a 'Shape'. Something like:
The way to achieve this is to use the virtual keyword on the base class methods, which specifies
that when a pointer to a base class invokes the method of an object that it points to, it should
determine, at runtime, the correct method to invoke. That is, when unknown_shape points to a
Rectangle it invokes Rectangle::Area() and if unknown_shape points to a Circle it invokes
Circle::Area() .
#include <cmath>
class Shape {
public:
// Virtual destructor (VERY IMPORTANT, SEE NOTE BELOW)
virtual ~Shape() {}
public:
Rectangle(double w, double l) : width(w), length(l) {}
public:
explicit Circle(double t) : radius(r) {}
NOTE: It is very important that a default virtual destructor was included after adding the virtual
Area() method to the base class. Whenever a base class includes even a single virtual method,
it must include a virtual destructor so that the correct destructor(s) are called in the correct order
when the object is eventually deleted.
This is called runtime polymorphism because the decision of which implementation of the Area()
method to use is determined during program execution based on the type that the base is pointing
at. It is implemented using the virtual table mechanism. In a nutshell: it is a little more expensive to
use but it can be immensely useful. There is also compile-time polymorphism. Here is more on the
differences between them.
In the example above, if a class extends from Shape but does not include an override of Area() then
calling the Area() method will invoke the base class method which (in the implementation above)
returns 0.0 .
In some cases, you may want to enforce that sub-classes implement this method. This is done by not
providing a default implementation, thus making it what is called a pure virtual method.
class Shape {
public:
virtual ~Shape() {}
virtual double Area() const = 0;
};
In general a class with only pure virtual methods and a virtual destructor is called an abstract class or
interface and is typically named as such (e.g. ButtonInterface , or similar). An interface class
guarantees that all extending classes implement a specific method with a specific method signature.
1.4 Special Methods
All classes have at least one constructor and a destructor, even if they are not explicitly defined. The
constructor and destructor assist in managing the lifetime of the object. The constructor is invoked
when an object is created and the destructor is invoked when an object is destroyed (either by going
out of scope or explicitly using delete ).
The constructor establishes a class invariant, a set of assertions guaranteed to be true during the
lifetime of the object, which is then removed when the destructor is called.
The explicit keyword should be used in single-argument constructors to avoid a situation in which
the constructor is implicitly invoked when a single argument is given in place of an object. Consider
the following Array class:
class Array {
private:
int size;
public:
// Constructor
Array(int size) {
this->size = size;
}
// Destructor
~Array() {}
Uh-oh. That's now legal, compileable code, but what does it mean? It is extremely unclear to the user.
class Array {
int size;
public:
explicit Array(int size) {
this->size = size;
}
// ...
};
array.Print(Array(12345));
Member initializer lists allow you to initialize member variables in the definition of a method. This
turns out to provide some performance benefits for class-type member variables, since a call to the
default constructor is avoided. For POD (plain old data) like ints and floats, though, it is the same as
initializing them in the body of the method.
class Car {
private:
int year;
int miles;
std::string make;
public:
Car(const int year, const int miles, const std::string & make) : year(year), miles(miles),
};
Using the initializer list is basically the same as the following more verbose constructor
implementation, notwithstanding the note above regarding performance:
Car(const int year, const int miles, const std::string & make) {
this->year = year;
this->miles = miles;
this->make = make;
}
Since C++11 initializer lists have some added functionality and curly braces {} can be used instead
of parentheses () in the initializer list, but to maintain compatibility with older compilers you may
want to use parentheses. The same applies in general to initialization syntax when creating objects.
Many people prefer braces, and in some cases it's necessary (e.g. vector containing [100, 1] or a
vector of one hundred 1s?), but to support older compilers you may consider using parentheses.
The new and delete operators (and their array counterparts, new[] and delete[] ) are operators
used to dynamically allocate memory for objects, much like C's malloc() and free() .
When manually allocating memory dynamically, it is the responsibility of the programmer to manage
the memory and properly delete objects that have been allocated.
Copy constructors and copy assigment operators allow one object to be constructed or assigned a
copy of another object directly:
Foo a(10);
Foo b(a); // (1): Copy via constructor
Foo c ;
c = a; // (2): Copy via assignment operator
This is accomplished by supplying a copy constructor and an assigment operator overload, both of
which have a special syntax where they accept a const reference to an object of their same type.
class Foo {
private:
int data;
public:
// Default (no argument) constructor
Foo() : data(0) {}
// Copy constructor
Foo(const Foo & f) : data(f.data) {}
// Copy assignment operator
Foo & operator=(const Foo & f) {
data = f.data;
return *this;
}
};
Note that the compiler will always provide a default constructor, a default copy constructor, and a
default copy assignment operator, so for simple cases (like this trivial example) you will not have to
implement them yourself. More info on this can be found here.
Sometimes instead of performing a copy you instead wish to completely move data from one object
to another. This requires the use of a move constructor and move assignement operator.
class Movable {
private:
Foo * data_ptr;
public:
Movable(Foo data) : data_ptr(new Foo(data)) {}
// Move constructor
Movable(Movable && m) {
// Point to the other object's data
data_ptr = m.data_ptr;
~Movable() {
delete data_ptr;
}
};
Movable Bar() {
// ...
}
int main() {
Movable a(Bar()); // Using the move constructor
Movable b = Bar(); // Using the move assignment operator
}
Since Bar() creates an object that won't be used elsewhere and is deleted after the call, we can use
the move constructor or move assignment operator to move the data to our object.
A programming idiom called 'copy and swap' makes use of the move constructor and can be a useful
idiom.
There are two main ways to do operator overloading. The first is using normal member functions. The
second uses the friend keyword and non-member methods that have access to the private member
variables of the class.
Using normal member functions (requires a getter method for the member variables):
// File: complex.h
class Complex {
private:
double r = 0.0; // Real part, defaults to 0.0
double i = 0.0; // Imaginary part, defaults to 0.0
public:
Complex(const double r, const double i) : r(r), i(i) {}
// Accessor methods
double GetReal(void) const { return r; }
double GetImaginary(void) const { return i; }
// + Operator
Complex operator+(const Complex & a, const Complex & b) {
return Complex(a.GetReal() + b.GetReal(), a.GetImaginary() + b.GetImaginary());
}
// - Operator
Complex operator-(const Complex& a, const Complex& b) {
return Complex(a.GetReal() - b.GetReal(), a.GetImaginary() - b.GetImaginary());
}
};
// File: complex.h
class Complex {
private:
double r = 0.0; // Real part, defaults to 0.0
double i = 0.0; // Imaginary part, defaults to 0.0
public:
Complex(const double r, const double i) : r(r), i(i) {}
int main() {
Complex a(1, 2); // 1 + 2i
Complex b(5, 3); // 5 + 3i
Complex c = a + b; // 6 + 5i
Complex d = a - b; // -4 - 1i
}
It's also often useful to overload the output stream operator to provide a custom output string
displaying the object's internal state in a human-readable format. This is done by overloading the <<
operator and requires using the <iostream> functionality.
#include <iostream>
class Complex {
private:
// ...
public:
// ...
friend std::ostream & operator<<(std::ostream & os, const Complex & c);
};
// Definition
// Again, this is NOT a member function!
std::ostream & operator<<(std::ostream & os, const Complex & c) {
os << c.r << " + " << c.i << "i";
return os;
}
int main() {
Complex a {1, 2};
Complex b {5, 3};
You can also similiarly overload the input stream operator ( >> ), and can read more about the various
operators here.
1.6 Templates
Templates are a very powerful abstraction allowing you to generate compile-time
methods/classes/etc. for any number of types while writing only one implementation.
Say you have a method that adds two floating point number together, and another to add two
integers together:
That's great, but since both floating point numbers and integers implement the + operator you can
use a template to instead write one generic implementation of a method that can operate on
doubles, ints, floats, and (in this case) any other type that implements the + operator.
A simple templatized version of Add would look something like this:
// Usages
int main() {
Add<int>(3, 5); // int version
Add<double>(3.2, 5.8); // double
Add(3.45f, 5.0f); // implicit float version: we leave off the <float> here, since it
In this simple example the compiler would generate four different methods, one for each type.
Templating allows you to write more concise and modular code at the expense of generating a larger
executable (code bloat).
Templates are especially useful to create class templates. Class templates must be completely defined
in a single header file.
// File: storage.h
// Usage
int main() {
Container<int> a(1);
Container<float> b(10.0f);
Container<Container<int>> c(a);
}
For a brief namespace example, suppose that you have two data structures, both of which implement
a Node class. In the following code, namespaces are used to allow the compiler (and the
programmer) to distinguish between the two types.
// File: list.h
namespace list {
}; // namespace
// File: bst.h
namespace bst {
}; // namespace
// File: main.cpp
#include "list.h"
#include "bst.h"
int main() {
list::Node<int> a;
bst::Node<int> b;
};
The standard C++ library uses the namespace std , e.g. std::cout , std::string , std::endl , etc.
While you can use a using namespace foo; directive to address symbols directly in the foo
namespace without prefixing the foo:: qualifier, this is generally considered bad practice as it
pollutes the global namespace and sort of undermines the point of using namespaces in the first
place.
#include <iostream>
using namespace std;
cout << "Hello, World" << endl; // <--- BAD: pollutes the global namespace
#include <iostream>
std::cout << "Hello, World" << std::endl; // <--- GOOD: It's clear that you're using symbols
Raw pointers in C++ behave exactly the same way as they do in C: a pointer variable stores the
address of whatever it is pointing to. You can think of pointers as essentially storing a link to another
piece of data. You can access the data that the pointer points to with the -> operator, or dereference
it with the * operator.
References are more akin to an alias. References cannot be NULL or nullptr , and references cannot
be reassigned to reference something else after they have been created. Additionally, references do
not take up extra memory; they share the same address as whatever they reference to. References
cannot have multiple levels of indirection (pointers can), and there is no reference arithmetic like
there is for pointers. You can access the underlying data of a reference directly by using the reference
itself: that is, if it's a reference to an integer it can be used as an integer. If it's a reference to a class
you can access the class members directly with the . operator.
Although pointers are incredibly powerful, references are generally much safer, especially when
passing objects to methods using pass-by-reference. It is very common in C++ code to pass an object
as a const reference (if the data should be unmutable within the method) or a non-const reference
rather than a raw pointer as is required in C.
// Pointers
int a = 10; // Ends up at memory address '0x2A000084', for example
int b = 20; // Ends up at memory address '0x2A000088'
ptr = &a; // The value of ptr is now the address of the variable 'a'
std::cout << ptr << std::endl; // Prints: 0x2a000084
std::cout << *ptr << std::endl; // Prints: 10
ptr = &b; // The value of ptr is now the address of the variable 'b'
std::cout << ptr << std::endl; // Prints: 0x2a000088
std::cout << *ptr << std::endl; // Prints: 20
// References
int a = 10; // Ends up at memory address '0x2A000084', for example
int b = 20; // Ends up at memory address '0x2A000088'
Perhaps the most widely used aspect of references is to pass objects by reference (sometimes
constant reference) to a method. To avoid hammering the stack with large objects when you pass
them by value it is nearly always preferrable to pass by reference, which is the term used when using
either a reference or a pointer. Using a reference allows you to pass any size object by reference,
while still allowing you to access the object directly.
if (bar.SomeMethod()) {
// ...
}
if (bar.SomeMethod()) {
// ...
}
Don't need to check for NULL or nullptr since references cannot be null
Can access the referenced object's data directly instead of using the -> operator or
dereferencing a pointer
Make it clearer which parameters are meant to be input parameters and which are meant to be
output parameters by using const to denote strictly input parameters
Gain the benefits of both passing by value and passing by reference since you don't need to use
a lot of memory on the stack for your object
Thus, passing by reference using a const reference is essentially the same as passing by value, but
you avoid copying the object onto the stack. Passing by reference using a non-const reference is
essentially the same as passing by reference using a pointer, but you are guaranteed that it is not null
and it's as if the pointer is effectively dereferenced.
2.3 Keywords
Reference
asm auto const constexpr (since C++11) explicit export (until C++11) extern (language
linkage) friend inline mutable noexcept (operator) noexcept (function specifier) nullptr
override static (class member specifier) template this virtual (function specifier) virtual
(base class specifier) volatile
static
extern
const_cast
dynamic_cast
#define : Defines a text macro. See here for full explanation, including macro functions and
predefined macros.
#undef : Un-defines a text macro
#line : Changes the current file name and line number in the preprocessor
#pragma : Non-standard, can be used instead of header guards ( #ifndef HEADER_H ...)
2.7 Exceptions
Reference
2.8 Lambdas
Reference