What is a C++ unique pointer and how is it used? smart pointers part I

Introduction

Unique pointers are smart pointers that help us in memory management. Unique pointers guarantee they delete their object if they are destructed, therefore, one less hassle for programmers. They follow “Resource Acquisition Is Initialization” (RAII) rule.

Prerequisites

Here, I assume you are aware of raw pointers and auto keyword.

I compiled the examples using GCC 10.2 with flag -std=c++20.

To be focused, the headers and main() are omitted in some examples:

#include <iostream> // For std::cout
#include <memory> // For std::unique_ptr, std::make_unique

using namespace std; // dropping std::

// class definitions

int main(){

    // implementations

    return 0;
}

Unique pointer

A unique pointer is defined as

std::unique_ptr<int> p(new int); // p is allocated a new int on the heap
(.Get 1)

If the unique pointer is destructed, the allocated object on the heap is destructed too

{
  unique_ptr<int> p(new int);
  // make use of p

} // p is destructed, so the int object is destructed.
(.Get 1)

Compare the above code with raw pointers which are deleted explicitly by programmers.

A unique pointer can also be created with std::make_unique

#include <memory>
#include <iostream>
using namespace std;

struct Person{
  Person(string n):Name(n){}
  ~Person(){cout<<"Deleted!";}
  string Name;
};

int main(){
    auto p = make_unique<Person>("Jack"); // Person constructor is called.
    return 0;
}

Ownership

A unique pointer is a 1-to-1 relationship between a pointer (p) and its allocated object on the heap (new int).

unique_ptr<int> p(new int);
// p <-------->  object

p owns the object and the object has only one owner, p. So when programming, we can think of them as one entity.

(.Get 1)

A unique pointer cannot be copied or passed by value. However, the ownership of its object can be transferred.

A unique pointer can be empty too

unique_ptr<int> p; // empty pointer, contains null pointer

Operations

A unique pointer supports operations below

struct Person{
  Person(string n):Name(n){}
  string Name;
};

auto p = make_unique<Person>("Rose"); // create unique pointer
auto b = *p; // dereference pointer
p->Name = "Jack"; // access class members

There is a raw pointer inside a unique pointer that can be accessed:

auto r = p.get(); // get the raw pointer

Use the above raw pointer only for calculations and do not delete it as it is managed by a unique pointer.

The object allocated to the pointer can be changed but remember that it is automatically deleted:

#include<iostream>
#include<memory>
using namespace std;

int main(){

    auto q = make_unique<int>();
    cout<< q <<'\n'; // points to 0xb1feb0

    q = make_unique<int>(); // The data in 0xb1feb0 deleted.
    cout<<q; //points to new object in 0xb20ee0

    return 0;
}

The pointer can be reset to a new object

auto q = make_unique<int>(); // q created with an int object on the heap
q.reset(new int()); // The previouse object deleted, a new one is created.

Only one unique pointer owns the object on the heap:

auto q = make_unique<int>(); // q associate to a newly created object on the heap
auto p = q; // Error: the object belongs to q and cannot be shared.

However, the ownership of the object can be transfered via std::move():

auto q = make_unique<int>(); // q created with an int object on the heap
auto p = move(q); // p owns the q's object, q lost it (null pointer).
(.Get 1)

std::swap works with unique pointers

auto p = make_unique<int>(); 
auto q = make_unique<int>(); 
swap(p,q); // p points to q's object and vice versa.

A unique pointer can be checked if it is associated with an object

unique_ptr<int> a; // a created but is empty (null pointer)
if (a) // (bool) a returns false as it is not associated.
  cout<<*a; // This is not run.

Pass to a function

The function below takes the ownership of a unique pointer. To pass the pointer std::move() must be used:

#include<iostream>
#include<memory>
using namespace std;

struct A{
    ~A(){cout<<"Deleted.";}
};

void PassIn(std::unique_ptr<A> a)
{
    cout<< "Pointer received."<<'\n';

} // a and its object are deleted.


int main(){
    
    auto x = make_unique<A>();
    PassIn(move(x)) // Pointer received.
    ; // Deleted.
    
    if (!x) cout<< "x is empty."; // true: x is empty.
    
    return 0;
}

Return from function

A function can return a unique pointer. Consequently, it gives up the ownership of the pointer:

#include<iostream>
#include<memory>
using namespace std;

struct A{};

std::unique_ptr<A> PassOut()
{
    auto a =  make_unique<A>();
    return a;
} 


int main(){
    
    auto x = PassOut();

    if (x) cout<< "x has an object."; // true: x has an object.
    
    return 0;
}

Pass to observer function

If the function only works with the pointer’s object and doesn’t care about the ownership, we can pass the unique pointer by a reference or raw pointer. If null pointer should be handled, we pass it by a raw pointer:

#include<iostream>
#include<memory>
using namespace std;

struct Database{
    double GetAverageSalary(){return 1000;};
};

void ShowSalaryDifference(double salary, Database* db)
{
    if (!db) throw runtime_error("Database is null.");
    cout<< salary - db->GetAverageSalary();
} 

int main(){
    
    auto db = make_unique<Database>();
    ShowSalaryDifference(1200, db.get()); // 200
    
    return 0;
}

If we are sure db has an object, we can pass it by reference

void ShowSalaryDifference(double salary, Database& db)
{
    cout<< salary - db.GetAverageSalary();
} 


int main(){
    
    auto db = make_unique<Database>();
    ShowSalaryDifference(1200,*db); // 200
    
    return 0;
}

Class member: unique pointer vs raw pointer vs reference

If we design our program based on smart pointers, we can assume the below rules for a class member:

  • Unique pointer member: the class is the owner of the pointer’s object.
  • Raw pointer member: the class is an observer and not responsible for deleting the pointer’s object. It is deleted by a smart pointer outside of this class. The pointer can be null.
  • Reference member: it is guaranteed that the reference contains valid data while the class object is alive.

Performance

Accessing unique pointers is as fast as raw pointers. The class of the unique pointer contains only a raw pointer as the data member, so, the size of a unique pointer is the same as a raw pointer. All in all, unique pointers can safely replace raw pointers in high-performance calculations.

Factory Example

A factory that creates unique pointers is shown below

#include<iostream>
#include<memory>
using namespace std;

struct Base{};
struct Derived:Base{};

std::unique_ptr<Base> create(int option)
{   
    if (option == 0)
        return make_unique<Base>();
    else if (option==1)
        return make_unique<Derived>();
    else
        throw runtime_error("Wrong option.");
}
int main(){
    
    auto p = create(1);
    return 0;
}

Next study

I recommend after this have a look at my posts on shared pointers and weak pointers.

References

unique pointer from cppreference

isocpp

Tags ➡ C++ Pointers

Subscribe

I notify you of my new posts

Latest Posts

Comments

2 comments
Mark H 4-Mar-2022
An excellent explanation. Thank you.
Peter David Fagan 26-Jun-2022
This is a great article, thank Sorush Khajepor.