auto
Keyword in C++In C++, the auto
keyword is a powerful feature introduced in C++11, used to declare variables with an inferred data type.
The compiler deduces the type of the variable based on its initializer, which can make code more readable and easier to maintain.
Consider the following example where auto
is used to declare variables:
auto dragon_population{3}; // dragon_population is deduced as an integer
auto westeros{7.7777}; // westeros is deduced as a double
auto wedding_location{"The Twins"}; // wedding_location is deduced as a const char*, not std::string
In each case, the type of the variable is inferred from the value it is initialized with.
The auto
keyword helps by writing more concise and readable code by reducing the verbosity of explicit types.
const std::vector<std::string> pigeon_pie{"flour", "butter", "pigeon", "salt"};
auto purple_wedding_pie{pigeon_pie};
purple_wedding_pie.emplace_back("the strangler");
In this loop, auto
deduces the type of purple_wedding_pie
as std::vector<std::string>
, avoiding the need to explicitly specify the type again.
The auto
keyword is compatible with various C++ constructs making it a versatile tool in modern C++ programming.
auto& element{array[0]}; // reference to an element
const auto object{otherObject}; // const type version of otherObject's type
auto* ptr{&x}; // pointer to x with the same type as x, but as a pointer.
In later concept we will often see the auto
keyword with lambda expressions, range-based for-loops, and iterators.
Smart pointers are a modern C++ feature designed to provide automatic memory management, helping to prevent memory leaks and dangling pointers commonly associated with raw pointers. They act as wrappers around raw pointers, adding additional functionality such as automatic memory deallocation when the pointer is no longer needed.
Smart pointers are typically implemented as class templates in the C++ standard library.
The two most commonly used smart pointers are std::unique_ptr
and std::shared_ptr
.
std::unique_ptr
is a smart pointer that owns the object exclusively.
It ensures that at any given time, only one std::unique_ptr
object owns the resource.
When the owning std::unique_ptr
is destroyed or reset, it automatically destructs the objects and releases its memory.
#include <memory>
// Declaring and defining a unique pointer
auto rightful_king_of_england = std::make_unique<std::string>("Excalibur");
// Unique pointers cannot be copied or assigned
auto mordred = rightful_king_of_england; // Error: Cannot copy a unique_ptr
std::make_unique()
When creating a std::unique_ptr
, it's preferable to use std::make_unique()
instead of directly using new
to allocate memory.
std::make_unique()
provides several advantages:
std::make_unique()
guarantees exception safety.
If an exception is thrown during the construction of the object, memory will be automatically deallocated, preventing memory leaks.std::make_unique()
makes code clearer and more concise.
It eliminates the need to explicitly specify the type being allocated, as the template arguments are deduced automatically.std::make_unique()
more effectively than manually allocating memory with new
, potentially resulting in improved performance.std::unique_ptr
is constructed manually.
That would lead to undefined behavior, when the std::unique_ptr
tries to delete it at its end of scope.std::shared_ptr
is a smart pointer that allows multiple std::shared_ptr
objects to share ownership of the same resource.
It keeps track of how many shared pointers are referencing the resource, and deallocates the memory only when the last shared pointer owning the resource goes out of scope or is reset.
// Declaring and defining a shared pointer to a dynamically allocated string
auto martian_congressional_republic = std::make_shared<std::string>("protomolecule");
// Creating more shared pointer that shares ownership
auto outer_planets_alliance = martian_congressional_republic;
auto united_nations = martian_congressional_republic;
In C++17 and below, using std::shared_ptr
with arrays via std::make_shared<T[]>
is not directly supported.
While it's possible to allocate arrays with std::make_shared<T[]>
, creating shared pointers directly from them may lead to undefined behavior due to differences in memory management between single objects and arrays.
Instead, consider using std::vector
or custom deletion functions to manage arrays with shared pointers effectively.
Always ensure compatibility with your compiler and standard library implementation when dealing with array allocations and shared pointers in C++17.
std::make_shared()
Similar to std::make_unique()
, std::make_shared()
offers benefits such as improved memory efficiency, exception safety, and readability.
It combines memory allocation for the control block and the managed object into a single operation, enhancing efficiency and reducing the risk of memory leaks.
Additionally, automatic deduction of template arguments simplifies code and enhances readability.
Using std::make_shared()
promotes cleaner, safer, and more efficient code when working with std::shared_ptr
objects in C++.
Use smart pointers by default: std::unique_ptr
for exclusive ownership and std::shared_ptr
for shared ownership.
Reserve raw pointers for non-owning references or when interfacing with legacy code.
In most cases, std::unique_ptr
is sufficient for exclusive ownership, as it offers lightweight memory management without the overhead of reference counting.
std::shared_ptr
should be used sparingly, as it introduces overhead and complexity unless true shared ownership is needed.
This exercise takes you to the world of Troy. The lives of its people are full of wonder and magic. Many humans in Troy possess powers, that are used frequently in their daily lives. Powers are used to re-shape the world or influence Troy's fauna and other people. Magic also manifests in unique artifacts, that are highly sought after by adventurers, artisans and sages.
In this exercise you are going to write code to model the humans of Troy, their possessed artifacts and power interactions.
You have six tasks. The first one is related to creating a human, the other five are about handling powers and artifacts.
For your model of Troy, humans are the most important feature. Your model human should be able to possess a unique artifact. They should also have the ability to manifest a power. These powers might affect other humans, so you also want to model if a human is influenced by some other power.
You are provided with basic implementations of artifact
and power
structs.
Implement a human
struct (or class) that has a smart-pointer to an artifact
member variable named possession
.
Each artifact can only be possessed by a single human at any given time.
A human
must have two additional member variables.
One holds their own_power
and the other is a power they are influenced_by
.
Both own_power
and influenced_by
are smart-pointers to powers
.
Each power
might be owned by a single human, but also influence other humans at the same time.
By default, humans are born without any artifact and neither own any powers nor are they influenced by them.
human mindy_mccready{};
mindy_mccready.possession;
// => nullptr
mindy_mccready.own_power;
// => nullptr
mindy_mccready.influenced_by;
// => nullptr
Your model is boring without the interaction of its parts. You want to create unique artifacts and give them to certain humans.
Define the function give_new_artifact
which returns nothing but takes a reference to a human
and a string
.
With the string
it should define a new artifact
object and set the possession
pointer of the human
accordingly.
The function should not return anything.
human erik_magnus_lehnsherr{};
give_new_artifact(erik_magnus_lehnsherr, "Mind shielding helmet");
erik_magnus_lehnsherr.possession->name;
// "Mind shielding helmet"
The world of Troy is all about interaction. You want people to make trades by exchanging their possessions.
Write a function exchange_artifacts
that returns nothing but takes two artifact smart-pointers to exchange the items.
Remember, that you cannot copy a unique_ptr
.
This includes the usage in function parameters.
Use a reference to the unique_ptr
instead.
human uchiha{};
give_new_artifact(uchiha, "konoha headband");
human uzumaki{};
give_new_artifact(uzumaki, "forehead protector");
exchange_artifacts(uchiha.possession, uzumaki.possession);
uchiha.possession->name;
// "forehead protector"
uzumaki.possession->name;
// "konoha headband"
The most exciting feature of Troy are the special powers that people might wield. Some can smelt iron with their thoughts, while others can heal every wound instantly at nighttime.
Define the function manifest_power
which returns nothing but takes a reference to a human
and a string
.
With the string
it should define a new power
object and set the own_power
pointer of the human
accordingly.
The function should not return anything.
human eleven {};
manifest_power(eleven, "psychokinesis");
eleven.own_power->effect;
// "psychokinesis"
What use are the greatest powers, if you cannot use them. Your model concentrates on humans, so you want to track the influence of powers.
Write a void function use_power
that takes two references to humans.
The first human is the caster and the second represents the target.
The target's influenced_by
pointer should be pointed to the power of the caster.
For simplicity, humans can only be influenced by a single power. This power stays in place even if the caster does not exist any longer.
human pamela_isley{};
manifest_power(pamela_isley, "control pheromones");
human count_vertigo{};
use_power(pamela_isley, count_vertigo);
count_vertigo.influenced_by->effect;
// "control pheromones"
Certain powers lose their potency or trigger certain effects in your simulation when they are applied to several humans. You want to track the number of people who are connected to each power.
Define the function power_intensity
, that takes a human and returns the intensity of their power as an int.
If the person has no power, the return value should be 0
.
Otherwise the intensity should reflect the caster and all currently influenced people.
human jean_grey{};
manifest_power(jean_grey, "uplifting personality");
human scott{};
human logan{};
human ororo{};
use_power(jean_grey, ororo);
use_power(jean_grey, logan);
use_power(jean_grey, scott);
power_intensity(jean_grey);
// 4
Sign up to Exercism to learn and master C++ with 19 concepts, 97 exercises, and real human mentoring, all for free.