Skip to content

Latest commit

 

History

History
2385 lines (1846 loc) · 70.8 KB

README.md

File metadata and controls

2385 lines (1846 loc) · 70.8 KB

C++ pills

Copyright (C) 2021 Michele Welponer

Table of contents generated with markdown-toc

Pointers

Pointers hold an integer that is a memory address! We say then a pointer points to a location in memory: that location is the starting location of a memory space (whose lenght is defined by the type). How to retrieve the memory address of a variable? Retrieve the memory address corresponding to the start of myVariable dereferencing that variable using & in front of the variable name.

dereference a variable

dereferencing a variable gives you the memory address at which that variable is stored. It is done by putting a & in front of the variable

std::cout << "myVariable mem address: " << &myVariable; 

declare a pointer

Declare a pointer using * right after its type:

int* myPointer;

set a pointer

int a = 3;
myPointer = &a;

now myPointer points to a and holds the memory address of a

dereference a pointer

dereferencing a pointer gives you the value contained at the memory address pointed by the pointer. It is done by putting a * in front of the pointer

Dereference the pointer (reading)

b = *myPointer;

this will assign/writes 3 (the value pointed by myPointer) to b

Dereference the pointer (writing)

*myPointer = 6;

this will assign/writes 6 to the variable pointed by myPointer i.e. a.

Increment the value held by the variable pointed by myPointer

++(*myPointer) or (*myPointer)++;

first dereference using the *, then increment. The difference here is the same between ++i and i++

pointer arithmetic

// the [] array operator is secretly a pointer, so it's somehow like to write
// char* str = "Hello", but sintattically it's not correct, so we write
char str[] = "Hello"; // and then, once we have an array we can 
char* ptr = str;

p*++, p++* and ++p* are all syntax errors

int b[] = {3, 0, 7};
int* p = b;

switch(4) {
  case 1:
    std::cout << *++p << std::endl; // move p, then dereference p -> 0
    std::cout << *++p << std::endl; // -> 7, [3 0 7]
    break;
  case 11:
    std::cout << *(++p) << std::endl; // move p, then dereference p -> 0
    std::cout << *(++p) << std::endl; // -> 7, [3 0 7]
    break;
  case 2:
    std::cout << ++*p << std::endl; // dereference p, then increment -> 4
    std::cout << ++*p << std::endl; // -> 5, [5 0 7]
    break;
  case 22:
    std::cout << ++(*p) << std::endl; // dereference p, then increment -> 4
    std::cout << ++(*p) << std::endl; // -> 5, [5 0 7]
    break;
  case 3:
    std::cout << *p++ << std::endl; // dereference p, then move p -> 3
    std::cout << *p++ << std::endl; // -> 0, [3 0 7]
    break;
  case 33:
    std::cout << *(p++) << std::endl; // dereference p, then move p -> 3
    std::cout << *(p++) << std::endl; // -> 0, [3 0 7]
    break;
  case 4:
    std::cout << (*p)++ << std::endl; // dereference p, then print, then increment -> 3
    std::cout << (*p)++ << std::endl; // -> 4, [5 0 7]
    break;
  //default:
    // code block
}

std::cout << b[0] << " " << b[1] << " " << b[2] << std::endl;

NB: if you are asking yourself where a pointer is stored, the answer is anywhere in the memory, as a pointer is just an integer variable holding the memory address of the variable it is pointing to. If you are curious to know the memory address of a pointer ptr you can use dereference &ptr or the function std::addressof(ptr);

References

References are like Pointers but less powerful. Reference are just syntax sugar, whatever you can do with references you can also do it using pointers. Declare and set a reference using &, is part of the type. When we declare a reference we need to assign reference immediately.

int& myRef1 = myVar; // myRef1 now is an alias of myVar
int& myRef2 = *myPtr; // myRef2 now is an alias of the variable pointed by myPtr*
myRef2 = 3; // assigns/writes 3 to the variable referenced by myPtr

Passing by reference

If we want to write a function that increments a variable, we can pass that variable inside the function using a pointer, or more concisely using a reference (i.e. by reference)

// using a pointer: int* value means create a new pointer named
// value which points to the memory address we pass as argument
// so if we want to increment the variable we need to first 
// de-reference the pointer to get the variable and then increment
void incrementUsingPointer(int* value) {
  (*value)++; // dereference first then increment
  std::cout << "value inside function:" << *value << std::endl;
}
// more simply, by reference: int& value means create a
// reference (alias) named value and make it point exactly 
// to the same variable we pass
void incrementUsingReference(int& value){
  value++;
  std::cout << "value inside function:" << value << std::endl;
}
void increment(int value){
  value++;
  std::cout << "value inside function:" << value << std::endl;
}

int main(){
  int a = 5;
  int& ref = a;
  
  incrementUsingPointer(&a); // value inside function:6
  std::cout << a << std::endl; // => 6
  
  incrementUsingReference(ref); // value inside function:7
  std::cout << a << ref << std::endl; // => 77
  
  incrementUsingReference(a); // value inside function:8
  std::cout << a << ref << std::endl; // => 88

  increment(a); // value inside function:9
  std::cout << a << ref << std::endl; // => 88

  increment(ref); // value inside function:9
  std::cout << a << ref << std::endl; // => 88
  
  // and with classes
  Entity e1; // instance on the stack
  incrementUsingPointer(&e1);
  incrementUsingReference(e1);
  
  Entity* e2 = new Entity(); // instance on the heap
  incrementUsingPointer(e2);
  incrementUsingReference(*e2);
}

NB: if we don't pass the variable by reference or by pointer to the function and we simply use increment(int value) a new int value variable will be created, initialized and incremented inside the function, but the external variable (the one we pass to the function won't be modified!)

reasuming

code meaning
T* p = nullptr; declare a pointer p of type T, p holds a memory address, for now set to null
T& r = v; declare reference r of type T to the variable v, r is now an alias of v
*p dereference pointer p i.e. get the data/variable/object pointed by p
&v dereference variable v i.e. get the memory address of v, this address can be assigned to a pointer

Examples:

int a = 3; // declare a variable
int* ptr = nullptr; // declare a pointer
ptr = &a; // dereference variable a to get its address and assign it to pointer ptr
std::cout << *ptr << std::endl; // dereference the pointer to get the data, => 3

int& ref = a; // create a reference i.e. alias of variable a
std::cout << ref << std::endl; // => 3

int& ref = *ptr; // create a reference i.e. alias of what the pointer points, i.e. variable a
std::cout << ref << std::endl; // => 3

ref++; // increment using the reference
std::cout << a << *ptr << ref << std::endl; // 444

a++; // increment the variable a
std::cout << a << *ptr << ref << std::endl; // 555

(*ptr)++; // dereference first to get the data, then increment it 
std::cout << a << *ptr << ref << std::endl; // 666

Static keyword

general use

Declare a variable or a function static (outside of a class/struct)

static int myVariable = 5;
static void myFunc();

myVariable will be visible only in that specific translation unit (file). It means I can define another variable with the same name myVariable inside another translation unit without incurring in linking problems at compile time. It's like defining the variable "private" for the specific translation unit.

NB: if I don't declare a variable static int var = 5;, in another translation unit I can read that variable declaring external int var;, to say that variable is defined and set in an external translation unit

So the moral is: try to always mark functions and variable static unless you actually need them to be linked across different translation units.

inside a local scope

Inside a function for example, but could be inside a for loop or whatever other type of scope

void func() {
  static int i = 0;
  std::cout << ++i << std::endl;
}
// outside func() i is not accessible
func() // first time i is declared and incremented => 1
func() // second time just incremented => 2

First time we call myFunction i is declared, set and incremented. All other times, just incremented. It's like declaring i globally outside of the function, but declaring it inside a function we also guarantee that i can be modified only inside the function.

use in object oriented

Declaring a field or a method static inside a class/struct

struct Entity {
  static int x;
  int y; 
  void print() { std::cout << x << ", " << y << std::endl;}
  static void s_print() {std::cout << "x:" x << std::endl;}
};

x will be unique and shared among all the Entity instances. Non-static methods can access static and non-static variables. Static methods cannot access non-static variables.

another good example of static inside a scope

let's create a Singleton. We declare the private static instance globally, then we need a method to get() that instance.

class Singleton {
private:
  static Singleton* s_instance; // global declaration
public:
  static Singleton& get() {return *s_instance;}
  void hello() {}
};

Singleton* Singleton::s_instance = nullptr;

int main() {
  Singleton::get().hello();
}

we can write a most clean Singleton declaring the instance static inside a scope, e.g. inside the get function

class Singleton {
public:
  static Singleton& get() {
    static Singleton instance; // declaring inside the scope
    return instance;
  }
  void hello() {}
}

int main() {
  Singleton::get().hello();
}

Enums

Stands for enumeration, a set of values. It's a way to give a name to an integer value so the code becomes more readable.

enum Example /* optionally specify the type of integer */: unsigned char {
  A, B, C // A will be 0, B 1 and C 2
  // A = 4, B, C // B and C will be 5 and 6
  // A = 5, B = 3, C = 7
};

class Log(){
public:
  enum Level {
    Error, Warning, Info
  };
private:
  Level m_logLevel = Info;
public:
  void setLevel(Level level){ m_logLevel = level;}
};

int main(){
  Example value = B;
  Log log;
  log.setLevel(Log::Error);
}

Objects and Classes

Example of a simple class

class Entity {
private:
  std::string m_name;
  int m_age;
public:
  Entity(){} 
  const std::string& getName() const { return m_name;}
  void setName(const std::string& name) {
    this->m_name = name;
  }
};

NB: this is the current object pointer, -> means dereference the pointer and then use the . to access the member method/field.

constructor and destructor

the Entity object will be instantiated and m_name and m_age fields will be immediately initialized to Unknown and 0 in the constructor.

class Entity {
private:
  std::string m_name;
  int m_age;
public:
  /* the constructor */
  Entity() 
    : m_name("Unknown"), m_age(0) // initializer list more efficient!
  {
    // here we don;t need to un-initialize anything because we didn't allocate anything in the constructor on the heap, m_name and m_age are on the stack memory
    // standard way to initialize class members (less efficient!)
    //m_name = "Unknown"; 
    //m_age = 0;    
    std::cout << "Entity created" << std::endl;
  }
  
  /* the destructor */
  ~Entity() {
    /* deallocate here everything created on the 
    heap in the constructor */
    std::cout << "Entity destroyed!" << std::endl;
  }
};

the destructor ~Entity will be automatically called when the Entity object gets deleted. Inside the destructor we deallocate everything that was created in the constructor on the heap memory.

NB: if the Entity object was created on the heap (using new operator) we need to manually deallocate the instance using "delete" (or "delete[]" for arrays). If the Entity object was created on the stack, the object gets automatically deleted when the program gets out of scope, thus the destructor will be called.

class inheritance

class Player : public Entity {
private:
  int points;
public:
};

the sub class Player extends the base class Entity, so Player class contains everything (methods and fields) that Entity class contains. Player just provides more functionalities to the Entity base class. And it can override any Entity class method.

virtual functions for methods overriding

Virtual functions allow us to override methods in sub-classes.

class Entity {
public:
  virtual std::string getName(){return "Entity";}
};

class Player : public Entity {
private:
  std::string m_Name;
public:
  Player(const std::string& name)
    : m_Name(name){}
  std::string getName() override {return m_Name;}
};

int main() {
  Player* p = new Player("Mike");
  Entity* entity = p;
  // OR using polimorphism
  Entity* entity = new Player("Mike"); // polimorphism
  
  /* if Entity::getName() 
  is not virtual, this will output "Entity" */
  std::cout << entity->getName() << std::endl;
}

If any sub class overwrites a method, it's good to make that method virtual in the base class. And to mark with override the method in the derived class.

NB: If getName() in base class Entity is not declared virtual then the printout will be Entity.

interfaces

Interfaces (a.k.a. abstract methods or pure virtual functions) allow us to define a function in a base-class that does not have an implementation, and at the same time force the sub-classes to implement that function. You are using an interface class when you want to guarantee that the subclasses extending that interface have specific methods

class Printable {
public:
  virtual std::string getClassName() = 0;
};

NB: = 0 makes it a pure virtual function! so the method needs to be implemented in the subclasses. Printable being an interface, cannot be instantiated.

visibility

class Entity {
private:
  int x, y;
protected:
  std::string name;
public: 
  getCoordinates(){};
};

in general

  • all that goes under public can be seen from outside and inside the class,
  • all that goes under private can only be seen within the class itself,
  • all that goes under protected can be seen from within the class itself and the subclasses.
class Entity {
//private:
  int x, y; /* private by default */
};

in classes if I don't specify the visibility, it will be private by default, so like to write private: before the declaration of the methods/fileds

struct Entity {
//public:
  int x, y; /* public by default */
};

struct

in structs if I don't specify the visibility, it will be public by default, so like to write public: before the declaration of the methods/fileds.

FUN FACT: actually this is the unique difference between class and struct. In c++ struct remain just to maintain the compatibility with c that doesn't have class.

class c_entity {
  int m_a, m_b;
public:
  c_entity(int a, int b) : m_a(a), m_b(b) {}
};

struct s_entity {
  int a, b;
  /* s_entity(int a, int b) : m_a(a), m_b(b) {} // this is optional */
};

int main() {
  c_entity c = {3, 4}; // or c_entity(3, 4)
  s_entity s = {3, 4}; // or s_entity(3, 4) if constructor is defined
}

class header

example of a class Human that extends from LivingBeing

Everything inside Human.h

class Human : public LivingBeing {
private:
  std::string m_name;
  int m_age;
  Human* m_son;
public:
  Human() 
    :m_name("unknown"), m_age(0){}
  Human(const std::string& name, const int age)
    :m_name(name), m_age(age){}
  Human(const std::string& name, const int age, Human* son)
    :m_name(name), m_age(age), m_son(son{}
  ~Human(){
    if (this->m_son)
      delete son;
  }
    
  void print(){
    std::cout << "Human " << this->m_name << ", age " << this->m_age << std::endl;
  }
  
  int age(){ return this->m_age; }
  Human* son(){ return this->m_son; }
};

If we want to separate the file into .h and .cpp then it becomes:

Human.h

#pragma once
#include <string>

namespace myNamespace {
  class Human : public LivingBeing {
  private:
    std::string m_name;
    int m_age;
    Human* m_son;
  public:
    Human();
    Human(const std::string& name, const int age);
    Human(const std::string& name, const int age, Human* son);
    ~Human();
    void print();
    int age();
  }
}

Human.cpp

#include "Human.h"

namespace myNamespace {
  Human::Human()
    : m_name("unknown"), m_age(0){}
  Human::Human(const std::string& name, const int age)
    :m_name(name), m_age(age){}
  Human::Human(const std::string& name, const int age, Human* son)
    :m_name(name), m_age(age), m_son(son{}
  Human::~Human(){
    if (m_son)
      delete son;
  }
        
  void print(){
    std::cout << "Human " << this->m_name << ", age " << this->m_age << std::endl;
  }
  
  int age(){ return this->m_age; }
}

Arrays

arrays on the stack memory

int stackArr[4] // declare: stackArr is now a pointer to the 1st element of the array
int stackArr[] = {1, 2, 3, 4} // declare and initialize 
stackArr[3] = 666

NB: we cannot use increment and decrement (++/--) operator on the array name e.g., stackArr++, because the array name is not an lvalue pointer. But we can always write

int* ptr = stackArr;
std::cout << ++ptr << std::endl; // => 2
*(ptr+3) = 666 // this equals to stackArr[3] = 666

FUN FACT: in release mode if you read/write out of the array space nobody will complain (no bounds checking, you will be reading/writing where you are not supposed to!). So it is better to write some check code to avoid the situation.

arrays on the heap memory

int* heapArr = new int[10];
delete[] heapArr;

Why should you declare an array on the heap? The difference is lifetime: creating something in the stack will be destroyed as soon as out of the scope. In the heap it will persist until deleted. E.g. if you have a function returning an array for example, you have to use the new keyword!

In general it is better - whenever possible - to declare the arrays in the stack to avoid all the pointer jumping to get to the actual data. Example:

Entity {
public:
  int stackArr[10];
  /* if instead here I use an array on the heap 
  int heapArr = new[10]; my Entity will contain 
  a pointer to the array so to get the data it 
  will need to jump to another space in memory */
  Entity {
    for (int i=0; i<10; i++)
      stackArr[i] = 0;
  }
};

NB: the row arrays do not have any .size or .lenght operators so:

  1. if you declare the array in the stack use this trick:
lenght = sizeof(stackArr) / sizeof(int);
  1. if you declare the array in the heap just do:
static const int numOfElements = 10;
int* heapArr = new int[numOfElements];

array standard library

#include <array>
std::array<int, 10> stdArr;
std::cout << stdArr.size() << std::endl;

it includes bounds checking and it keeps track of the array size. But of course there is a bit of overhead.

reasuming

code meaning
T ar[n] declare array ar of n elements of type T
&ar[0] the memory address of the first element of ar
&ar the memory address of the first element of ar
ar the memory address of the first element of ar
T* p = ar create pointer p that points to the first element of array ar
*(ar + 1) get the second element of array ar
ar[1] get the second element of array ar
T* ha = new T[n] declare on the heap array ha of n elements of type T
delete[] ha delete/remove array ha from the heap

Strings

C style strings

also known as null-terminated strings. A string is just an array of characters. If we want to declare it completely manually we need to add the string termination char (0 or '\0') that tells the pointer that the string is terminated.

char str[] = "Hello";
char mystring[5] = {'M', 'i', 'k', 'e', \0};
std::cout << mystring << std::endl;

C string handling functions

NB: #include <cstring>

  • strcpy(destination, source); copies the source string to the destination string. The destination must be large enough to hold the source string including the null terminator.
  • strcat(destination, source); appends the source string to the destination string. The destination must have enough space to hold the combined string.
  • strcmp(str1, str2); compares two strings lexicographically. It returns 0 if they are equal, a negative value if str1 is less than str2, and a positive value if str1 is greater than str2.
  • strlen(str); returns the length of the string, not counting the null terminator.

NB: #include <cctype>

  • isalnum(char) checks if char is alphanumeric check if a char is alphanumeric

declaration via char pointer

Declaration of a string using a char pointer in c++ it is immutable in the sense that you cannot change the lenght of the string, so usually it is declared const. #include <string.h> is used just to overload the << operator for the cout, strlen and strcpy

#include <string.h>
const char* name = "Mike";
std::cout << name << ", " << strlen(name) << std::endl;

char* str1 = "Ciao";
char str2[40];
strcpy(str2, str1); // copy str1 into str2 

string standard library

Finally if we want to use the standard string implementation.

std::string myString = "ciao";
myString += "!";
std::cout << myString << std::endl;
  
/* useful string methods */
myString.size();
myString.find("ao");
bool contains = myString.find("c") != std::string::npos; // std::string::npos means the end of the string

Const

normal use

const int MAX_AGE = 90;

normal use of const: the 'variable' MAX_AGE becomes basically a constant, it is not supposed to be modified.

with pointers

const BEFORE the asterix: what the pointer points at can't be modfied

const int* a = new int; // i.e.: int const* a
// *a = 5; // I cannot modify what the pointer points at!

const AFTER the asterix: the pointer address can't be modified

int* const b = new int;
*b = 5; // I still can modify what the pointer points at
//b = &MAX_AGE; // but I cannot modify the pointer address

and of ourse: const BEFORE & AFTER the asterix: the pointer address and what it points at cannot be modified

const int* const c = new int;
//*b = 5; // I cannot modify what the pointer points at
//b = &MAX_AGE; // I cannot modify the pointer address

with classes

Why declaring method const? Non-const objects can call either const or non-const methods. If I declare a const object like entity or a reference, then I will be able to call only on those const methods. RULE: const objects can not call non-const functions

class Entity {
private: 
  int m_X, m_Y;
  int* age;
public:
  /* used by a non-const Entity or Entity reference/pointer */
  int getX() {
    m_X = 2; // this is allowed    
    return m_X;
  }
  
  /* This method will be used by a const Entity reference */
  /* const here means I cannot modify the field inside the method */
  int getX() const {
    //m_X = 2; // this is not allowed
    return m_X;
  }  
};

if an Entity is used as parameter in some function, passed by const reference (not to copy all the class object) the normal getX() will not work. We need to create another method getX() const that will work (promising that inside the const method I don't modify the field).

FUN FACT: if I declare the class field as MUTABLE i can still modify it's value inside the const method! :P

class Entity {
private: 
  int m_X;
  mutable int m_debugCount = 0;

  int getX() const {
    m_debugCount++; // allowed because the filed is mutable!
    m_X = 3; // not allowed because the method is const!
    return m_X;
  }  
}

The ternary operator

.. just a syntactic sugar for if else

static int s_level = 1;
static int s_speed = 2;

void printSpeed() {
  s_speed = s_level > 3 ? 10 : 5; // speed becomes 10 if lev>3 else 5
  std::cout << "s_speen: " << s_speed << std::endl;
}

int main() {    
  printSpeed();
  s_level = 7;
  printSpeed();
}

Instantiate classes and objects

#include <iostream>
#include <string>

class Entity 
{
private:
  std::string m_name;
public:
  Entity()
    : m_name("Unknown"), m_age(-1) {}
  Entity(const std::string&  name)
    : m_name(name), m_age(-1) {}
  Entity(const  int  age)
    : m_age(age), m_name("Unknown") {}
    
  std::string& getName() const { return m_name; }
  int getAge() const {return  m_age;}  
  void setName(const std::string& name) {m_name = name;}
  void setAge(const int age) {m_age = age;}
};

Stack Heap memory allocation

stack allocation : as soon as that variable gets out of scope the memory is freed. When that scope ends - whichever scope - the stack pops off and anything was inside gets lost (destructor gets automatically called). Data is stored in contiguous way, stack memory is limited but very fast.

int value = 4;
int array[] = {1, 2, 3, 4, 5};
Entity entity("Mike"); // equals to Entity entity = Entity("Mike");

heap allocation: when allocated an object inside the heap, it's gonna sit there untill you decide you no longer need it.

int* hvalue = new int;
*hvalue = 4;
int* harray = new int[5];
array[0] = 1;
array[1] = 2;
array[2] = 3;
Entity* hentity = new Entity("Mike");
// free the allocated memory on the heap!
delete hvalue;
delete[] harray;
delete hentity;

Allocating on the heap takes longer: when we write "new dataType" we require the compiler to find and allocate a dataType long free and contiguous space in heap memory and return a pointer to that memory address (i.e. malloc function). Allocation on the heap is indeed a very heavy task, compared to the allocation on the stack that just needs 1 cpu instruction. Moreover, when allocating on the heap, we need then to remember to manually free the allocated memory. So the rule of thumb is whenever possible instantiate on the stack otherwise on the heap. It can be impossible to use stack memory when for example:

  1. the object is inside a scope and needs to remain alive also outside of that scope, or when
  2. the object is too big to stay on the stack (stack is usually 1-2MB) or
  3. there are too many objects that need to be allocated.

NB: So, whenever we use the word new, we need then to delete!

the new keyword

The main purpose of new is to allocate memory on the heap specifically. When we write "new dataType" we require the compiler to find and allocate a dataType long free and contiguous space in heap memory and return a pointer to that memory address -- basically it finds a block of memory that is big enough to accomodate our needs and then gives us a pointer to that block of memory -- The last thing that new does is also call the constructor on the new allocated entity.

int main(){
  int* a = new int;
  int* aArr = new int[50]; // an array of 50 contiguous int on the heap
  
  Entity* e = new Entity();
  
  // this is the equivalent c version
  Entity* another_e = (Entity*)malloc(sizeof(Entity));
  // it will just allocate the memory and not call the constructor!
  
  Entity* eArr = new Entity[50];

  // remember to free the memory when you are done with it!
  delete a, e, another_e;
  delete[] aArr, eArr;
}

Implicit conversion

is the automatic conversion that c++ does from one type to another, see the following examples

// some cool printing function
void PrintEntity(const Entity& entity) { // cool printing }

int main()
{
  Entity a("Mike"); 
  // we can also write Entity a = std::string("Mike")
  
  Entity b(43); // we can also write Entity b = 43
  
  // implicit conversion
  PrintEntity(22); 
  // 22 was implicitly converted into an Entity with age 22
  
  // implicit conversion: "Cherno" is firstly casted from 
  // const char array to a std::string then implicitly converted 
  // into an Entity with name Cherno
  PrintEntity(std::string("Cherno"));
  // it can be casted also to Entity
  PrintEntity(Entity("Cherno"));
}

NB: explicit keyword in front of the contructor disable any form of implicit conversion, e.g.:

class Entity {
  Explicit Entity(std::string& name) : m_name(name) {} 
};

int main {
  // this will cause a compilation error !!!
  PrintEntity(std::string("Mike")); 
  // we need to explicitly say
  PrintEntity( Entity(std::string("Mike)) );
}

Operator overloading

+ - * / % ^ & | ~ ! = < > += -= *= /= %= ^= &= |= << >> >>= <<= == != <= >= <=> && || ++ -- , ->* -> ( ) [ ]

Operators are just functions! All these c++ operators can be overloaded, i.e. definer/change the behaviour of the operator in the program. To overload an operator, e.g. the operator "@" we just write the word operator followed by the operator we want to overload @, then we define the function normally with its parameters.

example of overloading opertor "<<" to print a standard std::vector<T> object

template<typename T>
std::ostream& operator<<(std::ostream& stream, const std::vector<T>& vector) {
  int size = vector.size();

  stream << "[";
  for (int i = 0; i < size; i++)
    i == size-1 ? stream << vector[i] : stream << vector[i] << ", ";
  stream << "]";

  return stream;
}

example of overloading operator "+" to define how to add two Vector2 objects

struct Vector2 {
  float x, y;
  
  Vector2(float x, float y) : x(x), y(y) {}

  // here is how we overload the + operator
  Vector2 operator+(const Vector2& other) const {
    return Vector2(x + other.x, y + other.y);
  }
};

example of overloading the operator "<<", to define how a custom Vector2 is to be printed in std::cout

std::ostream& operator<<(std::ostream& stream, const Vector2& vector) {
  stream << vector.x << ", " << vector.y;
  return stream;
}

example of overloading of operator "new", to define what to do each time the word new is used: keep track of heap allocation count, show the quantity of heap memory allocated

static uint32_t s_AllocCount = 0;
void* operator new(size_t size){
  s_AllocCount++;
  std::cout << "Allocating " << size << " bytes\n";
  return malloc(size);
}
std::cout << s_AllocCount << " allocations\n";

The this keyword

it is just the pointer to the current object instance that the method belongs to.

class Entity {
private:
  int x, y;
public:  
  Entity(int x, int y) {
    // this is a pointer so using -> we first dereference 
    // then we use the Entity object
    this->x = x; 
    this->y = y;
  }
  // some fancy printing function
  void print() {};
};

Smartpointers

are scoped pointers (created on the stack) that will create and point to objects created on the heap. Once out of the scope the pointer and the pointed objects get automatically deleted.

unique pointers

simple type of smarpointer whose created object cannot have more than one pointer reference. make_unique will create a new Entity that can have just one pointer reference. You cannot copy unique pointers: if you have two unique pointers pointing to the same loc of memory and one of them dies, the loc will be freed and suddenly the second pointer will point then to memory that has been freed.

// create a unique pointer to a new Entity
std::unique_ptr<Entity> u1_ptr = std::make_unique<Entity>(7, 4);
u1_ptr->print(); 

// create another unique pointer and assign u1
std::unique_ptr<Entity>  u2_ptr = move(u1_ptr);
u2_ptr->print(); // let's check u2_ptr

// this will cause segmentation fault!
/* u1_ptr->print(); */ 
// there can be just one unique pointer to the object !!!

shared pointers

smartpointer that can be copied and actually maintain a reference copy count. Will free Entity only when last pointer reference gets deleted.

// create a shared pointer to a new Entity
std::shared_ptr<Entity> s1_ptr = std::make_shared<Entity>(7, 4);
std::cout << "count: " << s1_ptr.use_count() << std::endl; // let's check s1 count => 1

{ // an innner scope
  // create another shared pointer and assign s1
  std::shared_ptr<Entity> s2_ptr = s1_ptr;

  // let's check s1 count
  std::cout << "count: " << s1_ptr.use_count() << std::endl; // => 2

  // at the inner scope exit ref count gets decremented 
}

// let's check again 
std::cout << "count: " << s1_ptr.use_count() << std::endl; // => 1

weak pointers

weak pointer doesn't increment the reference count of a shared pointer, so in the following example, when we get out of the inner scope, the memory gets freed. However we can still ask the weak pointer, are you expired? are you still valid?

std::weak_ptr<Entity> w_ptr; // create a weak pointer

{ // an innner scope
  // create a shared pointer to a new Entity
  std::shared_ptr<Entity> s_ptr = std::make_shared<Entity>(7, 4);

  // assign weak pointer the shared pointer
  w_ptr = s_ptr; // ref count remains 1

  // let's check if weak pointer is intact
  std::string state = w_ptr.lock() ? "intact" : "gone";
  std::cout << "state: " << state << std::endl; // => intact

  // at the inner scope exit ref count gets 0, 
  // Entity object gets freed
}

// let's check again 
std::string state = w_ptr.lock() ? "intact" : "gone";
std::cout << "state: " << state << std::endl; // => gone

The copy constructor

Unnecessary copying is to be avoided, but sometimes copying is essential. When we are playing with copying class objects we need to verify that the classes we wrote have a copy constructor. Let's write a custom String class to make an example

class String {
private:
  char* m_buffer;
  unsigned int m_size;
public:
  String(const char* string) {
    m_size = strlen(string);
    m_buffer = new char[m_size + 1];
    memcpy(m_buffer, string, m_size);
    m_buffer[m_size] = 0; // null termination char
  }

  /* this is the copy constructor!! that guarantees that 
  m_buffer of a String object copy will be created on a 
  new block of memory */
  String(const String& other)
    : m_size(other.m_size) {
    m_buffer = new char[m_size + 1];
    memcpy(m_buffer, other.m_buffer, m_size + 1);
  }

  /* this is the way to prevent the user to make copies 
  of the String objects */
  // String(const String& other) = delete;

  ~String() {delete[] m_buffer;}
}

int main() {
  String string = "Cherno";
  String another = string; // copying a String object
  /*  */
}

NB: if String class doeasn't have a copy construstor that makes a deep copy, the m_buffer of 'string' and 'another' will be the very same at the very same block of memory.

The arrow operator

If we have an object declared on the stack then to access fields we use . If we have an object on the heap, created with new, so basically we have a pointer to that object, we access its fields using -> Indeed -> is a shortcut of dereference a pointer and use a method of the dereferenced object. Therefore, when we use the word this inside the definition of a class, that is the pointer to the actual object we use this->

int main() {
  Entity e; // create an entity on the stack
  Entity* ptr = &e; // make ptr point to it
  
  /* create a reference and assign it the 
  the object dereferenced by ptr pointer */ 
  Entity& entity = *ptr;
  entity.print();

  /* the last two lines can be shortcutted like this */
  (*ptr).print() // dereference first and then use print()
  /* or even better */
  ptr->print();
  ptr->age = 5;
}

So what -> operator does is dereference that pointer into a reference type and then call its method or field. You can overload the -> operator in this way. Let's create a scoped pointer as an example

class Entity {
public:
  void print() const {..}
};

class ScopedPtr {
private:
  Entity* m_obj;
public:
  ScopedPtr(Entity* entity) : m_obj(entity) {}
  ~ScopedPtr() {delete m_obj;}

  /* that's how we overload the -> operator */
  const Entity* operator->() const {
    return m_obj;
  }
};

int main() {
  // entity automatically deleted when it goes out of scope
  ScopedPtr entity = new Entity();
  
  /* the overload of the -> operator may allow 
  this even if entity is not a heap allocated entity */
  entity->print(); 
}

Dynamic arrays

It is just an array for which it is not necessary to specify the size. So we declare a dynamic array using:

#include <vector>
#include <algorithm> // used by for_each

std::vector<int> v {1, 2, 3, 4, 5}; // ver1
std::vector<int> v = {1, 2, 3, 4, 5}; // ver2
std::vector<int> v(3); // size 3, initial value 0
std::vector<int> v(3, 5); // size 3, initial value 5
std::vector<Point> pointArray;

// using lambda
for_each( v.begin(), v.end(), [](int i) { cout << i << endl; } );

and we add elements to a dynamic array using pushback():

pointArray.push_back(Point(x, y));

Optimizations

There are a couple of optimizations we can do using a dynamic array:

  1. if we know in advance how many elements we are gonna put into the array we can "reserve" the memory so the array will be resized just one time and not every time a new element is pushed_back in.
pointArray.reserve(8); 
  1. we can use emplace_back passing just the parameters to build a new element, in place of push_back already istanciated elements. In this way the elements will not be copied around uselessly
pointArray.emplace_back(x, y);

Static and Dynamic libraries

static linking

the library gets basically put into your executable (it's inside your .exe file). Technically faster. Use static whenever possible.

Create a Dependencies directory and inside another directory with the name of the dependency e.g. GLFW. Inside it we put the include folder and the lib folder. In VB we go to project properties > c/c++ > general > additional include directories, we select all configurations Win32 and we add $(SolutionDir)Dependencies\GLFW\include. Now in my code I can #include <GLFW/glfw3.h>

but we haven' t linked our actual library. We need the definition for the glfw3 functions.

We go to project properties > linker > general > additional librariy directories, and we add $(SolutionDir)Dependencies\GLFW\lib-vc2015. Then we go to project properties > linker > input, and we add glfw3.lib

dynamic libraries

the library gets linked at run-time. In windows you can use some function like load_library or at the application lunch it loads a .dll file (dynamic link library)

Templates

It's kind of generics, but it can do much more, it's not limited by the generic type system. It's a template specifying the compiler how to create methods, classes, or whatever, at compile time, based on the specific usage.

template<typename T, int N>
class Array {
private:
  T m_array[N];
public:
  int getSize() const { return N; }
};

int main() {
  Array<int, 50> array;
  array.getSize();
}

The class Array will be compiled if and only when used; type T and size N will be substituted by the type and size we specify when we actually use it (at compile time :P).

Macros

Macros are pre-processor, pure text replacing, before everything gets compiled. A simple example:

#define WAIT std::cin.get()

int main(){
  WAIT;
}

A more useful example, using a parameter and a preprocess definition:

#ifdef PR_DEBUG // preprocess definition variable
#define LOG(x) std::cout << x << std::endl
#else
#define LOG(x) // this will basically remove LOG line in main
#endif

int main(){
  LOG("Hello");
}

It is common practice to use macros to add custom debugging or validation checks:

#include <iostream>

// Custom assert macro
#define MY_ASSERT(condition) \
  do { \
    if (!(condition)) { \
      std::cerr << "Assertion failed: " << #condition << " in " << __FILE__ << " at line " << __LINE__ << std::endl; \
      std::terminate(); \
    } \
  } while (false)

int main() {
  int x = 5;
  MY_ASSERT(x == 5); // Use the custom assert macro

  std::cout << "Program continues after the assertion check." << std::endl;
}

NB: here the macro is wrapped in a do { ... } while (false) block to allow the use of multiple statements within the condition, ensuring proper scoping.

Example:

#define MY_ASSERT(condition) \ 
  if (!(condition)) { \ 
    std::cerr << "Assertion failed: " << #condition << std::endl; \ 
    std::terminate(); \ 
  }

// Use the custom assert macro with multiple statements  
if (x == 5) 
  MY_ASSERT(x > 0); 
else  
  do_something_else();

the expanded code would look like this:

if (x == 5) 
  if (!(x > 0)) { 
    std::cerr << "Assertion failed: " << std::endl; 
    std::terminate(); 
  } 
  else 
    do_something_else();

do { ... } while (false) construct is commonly used in macros to ensure proper behavior in complex statements

Namespace

The primary purpose of namespaces is to avoid naming conflicts, i.e. with namespaces we can have symbols (classes, functions, variables) with the same signature in different contexts

namspace apple {
  void print(const char* str){//do something}
}

namspace orange {
  void print(const char* str){//do something else}
}

int main(){
  orange::print("ciao a tutti");
  // or
  using namespace apple;
  print("ciao a tutti");
}

Function pointers

You can actually assign functions to variables. Starting from that, you can also pass a function into another function as paramenter. There's a whole bunch of things that we can actually do with functions and function pointers that do create intresting and complex logic.

// a very simple function
void helloWorld(){
  std::coutn << "hello" << std::endl;
}

int main(){
  // assign myfunc the memory address of helloWorld function
  auto myfunc = &helloWorld; // because of implicit conversion we can also omit the ampersand &

  // the type auto will be void(*myfunc)() so I can also write
  void(*myfunc)() = helloWorld; 

  // even better, I can use typedef like this
  typedef void(*hworldfunc)() 
  // and then 
  hworldfunc myfunc = helloWorld;

  myfunc();
  myfunc();
}

I can also use functions that have parameters so it would become

void helloWorld(int value){
  std::coutn << "hello: " << value << std::endl;
}

int main(){
  typedef void(*hwfunc)(int) 
  hwfunc myfunc = helloWorld;

  myfunc(8);
  myfunc(4);
  myfunc(9);
}

a more useful example can be this:

// this is the callback function
void printValue(int value){
  std::cout << value << std::endl;
}

void foreach(const std::vector<int>& values, void(*func)(int)){
  for (int v : values)
    func(v);
}

int main(){
  std::vector<int> vector = {1, 2, 3};
  foreach(vector, printValue);
}

so here I use foreach function to which I pass a vector of elements and a function pointer. The function the pointer points to defines what I want to do with each element of the vector. Though we can rewrite this in a more simple way, avoiding to write an extra function, using lambda. Lambda is just a normal function except that it is not declared as a normal function.

void foreach(const std::vector<int>& values, void(*func)(int)){
  for (int v : values)
    func(v);
}

int main(){
  std::vector<int> vector = {1, 2, 3};
  foreach(vector, [](int value){std::cout << value << std::endl;});
}

Lambdas

ref. https://en.cppreference.com/w/cpp/language/lambda

Lambdas are a way to define "anonymous functions". A way to create a function without the need to "phisically" create a function. This sort of function, we want to treat it more like a variable unless then a formal function that exsists like a symbol in our compiled code. So when to use it? In C++, whenever we set a function pointer to a function, we can set it to a lambda instead.

std::vector<int> v {1, 2, 3, 4};

// defining my lambda: check if i is even or odd
auto is_even = [](int i) { return i % 2 == 0; };
// result is an iterator that points to the first match
// or to end() if no match are found
auto result = std::find_if(begin(v), end(v), is_even);

// if something is found dereference iterator to print the first match
(result != std::end(v))
? std::cout << "v contains an even number: " << *result << '\n'
: std::cout << "v does not contain even numbers\n";
// inside [] I can put captures, variables/objects that are outside
// my lambda function
// I can use:
// [=] captures everything by value
// [&] captures everything by reference
// zero to n variables/classes by reference or by value
// e.g. [a] to pass variable a by value
// [&a] to passa variable a by reference

int const max = 100;
std::vector<int> vec {178, 101, 123, 145, 45, 33};

auto lambda = [max](int i){ return i < max; };
auto it = std::find_if(begin(vec), end(vec), lambda);
(it != end(vec))
? std::cout << "first element <= max is: " << *it <<std::endl
: std::cout << "all element are > max." << std::endl;

Multidimensional arrays

A simple array is a pointer to the first element of the array

// on the stack
int a[50];
int* ptr = a;

// on the heap 
char* buffer = new char[8];

A 2d array in C++ is an array of pointers, each one pointing to another array

int x[3][4] = {{0,1,2,3}, {4,5,6,7}, {8,9,10,11}}
// OR
int x[3][4] = {0,1,2,3,4,5,6,7,8,9,10,11}

On the heap a 2d array in is still a buffer of pointers that points to arrays stored somewhere in the memory.

// a pointer to a pointer to an integer
int** a2d = new int*[50];
// with this we just have allocated memory
// specifically 50 integer pointers, so
// 4bytes * 50 = 200 bytes of memory

// now we need to initialize the 50 arrays 
for (int i = 0; i < 50; i++)
  a2d[i] = new int[50];

// so to delete we need to delete them one by one
for (int i = 0; i < 50; i++)
  delete[] a2d[i];
// and finally
delete[] a2d;

so for a 3d array it would be

int*** a3d = new int**[50];
for(int i = 0; i < 50; i++){
  a3d[i] = new int*[50];
  for(int i = 0; i < 50; i++)
    a3d[i][j] = new int[50];
}

Managing multidimensional arrays this way though is not very convenient, nor very efficient, because an 2d array ends up to be a buffer of pointers to not contiguous arrays, so we continuously jump from one part to another part of the memory. The more optimized version is a single dimension array

int* fake2darray = new int[3*3];

for(int i = 0; i < 3*3; i++){
  for(int j = 0; j < 3; j++)
    fake2darray[j + i * 3] = 0; 
}

Sorting std sort

ref. en.cppreference.com/w/cpp/algorithm/sort

sorting a simple array

#include <algorithm>
int n = 7; // array size
int a[] = {4,2,5,3,5,8,3};
sort(a, a+n);

here we just sort a simple vector of elements

#include <algorithm>
std::vector<int> values = {3, 5, 1, 4, 2};
std::sort(values.begin(), values.end()); // 1 2 3 4 5
std::sort(values.rbegin(), values.rend()); // 5 4 3 2 1

sorting a string

#include <algorithm>
string s = "monkey";
sort(s.begin(), s.end());

sorting classes based on some kind of comparison

bool compare(Man& m1, Man& m2){
  return m1.age() < m2.age();
}
std::vector<Man> v = {mike, alice, bob};
sort(v.begin(), v.end(), compare);

if we want to provide some kind of way to sort providing it with a lambda function

#include <functional> // to use std::greater

std::sort(values.begin(), values.end(), std::greater<int>()); // > 5 4 3 2 1
std::sort(values.begin(), values.end(), [](int a, int b){ return a < b; }); // > 1 2 3 4 5
std::sort(values.begin(), values.end(), [](int a, int b){ return a > b; }); // > 5 4 3 2 1

type Punning

Type punning means treat a memory of a specific type as another type

int a = 5;
std::cout << "a:" << a << std::endl;
double value = a; // this one here is an implicit conversion, so the compiler implicitly did
// double value = (double)a;
// what was actually done was a conversion from int to double 
// so we will find different things into the memory
std::cout << "value:" << value << std::endl;

what instead if we want to take that exsisting original int memory and treat it as a double? (type punning)

// we take memory address of 'a', cast it to double pointer, then dereference to get back the value 
value = *(double*)&a;
std::cout << "value:" << value << std::endl;

a more interesting example: let's type punning a struct into an array:

struct entity {
  int a, b;
};

int main() {
  entity e = {5, 8};
  int* arr = (int*)&e;
  std::cout << "arr: " << arr[0] << ", " << arr[1] << std::endl;
}

Union

A union is a special class type that can hold only one of its non-static data members at a time. Imagine we have a Vector2 and a Vector4 and we want to access Vector4 memory parts using Vector2, i.e. treating a Vec4 as a couple of Vec2.

struct Vector2 { float x, y; };
// operator overloading to print Vector2
std::ostream& operator<<(std::ostream& stream, Vector2& vector){
  stream << vector.x << ", " << vector.y;
  return stream;
}

Using type punning, we could write getA() and getB() to get respectively x, y and z, w

struct Vector4 {
  float x, y, z, w;
  Vector2& getA() { return *(Vector2*)&x; }
  Vector2& getB() { return *(Vector2*)&z; }
};

int main() {
  Vector4 v4 = {1, 2, 3, 4};
  std::cout << v4.getA() << std::endl;
}

using Union we can instead write something simplier like this:

struct Vector4 {
  union {  
    // an anonymous struct to structure the data
    struct { float x, y, z, w; }; 
    struct { Vector2 a, b; };
    // so now I can access data in two ways: 
    // 1. using x, y, z, w
    // 2. using a and b, where a happens to be x, y 
    // b happens to be z, w
  };
};

int main(){
  Vector4 v4 = {1, 2, 3, 4};
  v4.z = 33;
  std::cout << v4.b << std::endl;
}

Virtual Destructors

When inheritance and virtual functions come together, then we need to deal with virtual destructors too.

class Base {
public:
  Base(){std::cout << "B constr, ";}
  ~Base(){std::cout << "B destr, ";}  
};
class Derived : public Base {
public:
  Derived(){std::cout << "D constr, ";}    
  ~Derived(){std::cout << "D destr, ";}
};
int main(){
  Derived* derived = new Derived();
  delete derived;
}

this will printout B constr, D constr, B destr, D destr, as we expect! What will happen when we use a polymorphic like this:

int main(){
  Base* poly = new Derived();
  delete poly;
}

just the Base destructor will be called, leading to a memory leak! So to solve this, we need to set destructor of Base class as virtual, telling this way that the class may be extended, so call the derived destructors if present

class Base {
public:
  Base(){std::cout << "B constr";}
  virtual ~Base(){std::cout << "B destr";}  
}

So, whenever writing a class that you will be extending or might be subclassed, you need to 100% declare destructor as virtual

Casting

static cast

Type casting, means conversion within the type system that C++ provide us with. If I declare a variable of a specific type, I need to stick with that type unless either there is an implicit conversion (C/C++ knows how to convert from one type to another) or we write an explicit conversion, so we tell C/C++ how to convert from one type to another.

C style cast:

double value = 5.32;
int a = (int)value + 5.4; // explicit conversion

C++ style cast (static cast):

double value = 5.32;
int a = static_cast<int>(value) + 5.4;

the good thing about C++ style of casting is that if something goes wrong with my compilation and I get some compile check messages and I can search for casting in my code easily trying to figure out the problem

dynamic cast

We can use dynamic cast to check if an object is of a specific type. Suppose we have Base class, Derived and Another classes that both extend from Base.

class Base {
public: 
  Base(){} 
  virtual ~Base(){}
};
class Derived : public Base {
public:
  Derived(){}
  ~Derived(){}
};
class AnotherClass : public Base {
public:
  AnotherClass () {}
  ~AnotherClass () {}  
};

and we want to know if a pointer of type Base specialized to Derived or to AnotherClass

Derived* derived = new Derived()
Base* base = derived // suppose we don't know that base specialized to Derived

AnotherClass* ac = static_cast<AnotherClass*>(base); // will do the conversion
AnotherClass* ac = dynamic_cast<AnotherClass*>(base); // will check if base specialized to AnotherClass, if not it will return NULL, so we can check the type by writing if (!ac) {} or if (ac) {}

reinterpret cast

type punning :P

const cast

removes or add const :P

Singleton

it is basically a single instance of a class that you have around. So in some sense a singleton in c++ acts pretty much like a namespace, as it is a way to organize a bunch of global variables and static functions into a kind of "organized blob".

class SingletonExample {

public:
  // the method to retrieve the single instance
  static SingletonExample& get() {
    static SingletonExample instance;
    return instance;
  }

  void setStatus() { this->status = 1; }
  static int getStatus() { return get().status; }

private:
  int status; // a variable to prove the instance is unique
  // declare private constructor to prevent the creation of objects
  SingletonExample() : status(0) {
    std::cout << "constr" << std::endl;
  }
};

int main() {
  std::cout << SingletonExample::getStatus() << std::endl;

  // multiple references retrieve the very same instance 
  SingletonExample& se1 = SingletonExample::get();
  SingletonExample& se2 = SingletonExample::get();
  auto& se3 = SingletonExample::get();

  std::cout << "s1 " << se1.getStatus() << std::endl;
  se1.setStatus();

  std::cout << "s1 " << se1.getStatus() << std::endl;
  std::cout << "s2 " << se2.getStatus() << std::endl;
  std::cout << "s3 " << se3.getStatus() << std::endl;

  std::cout << SingletonExample::getStatus() << std::endl;
}

lvalues and rvalues

lvalue

simply means an object that has an identifiable location in memory (i.e. having an address).

int a; // declare an object of type int: a variable that has a location in memory
a = 1; // a is therefore an l-value expression
int b = a; // here a l-value can appear on right

9 = a; // Error! 9 is an r-value because it is a value, and doesn't have a location in memory, therefore cannot be on the left of the operator =

rvalue

A r-value is an expression, that can’t have a value assigned to it, which means r-value can appear on right but not on left hand side of an assignment operator(=)

// declare 'a', 'b'
int a = 1, b;  // objects of type 'int'
a + 1 = b; // Error! left expression is not a l-value
 
// declare pointer variable 'p', and 'q'
int *p, *q; // *p, *q are lvalue
*p = 1; // valid l-value assignment
p + 2 = 18; // Error! "p + 2" is not an l-value
q = p + 5; // q is l-value, "p + 5" is an r-value
*(p + 2) = 18; // dereferencing pointer: expression gives an l-value

 
p = &b; // assign pointer p the mem addrs of b
int arr[20]; // arr[12] is an l-value; equivalent to *(arr+12) Note: arr itself is also an l-value
&a = p; // Error! &a is an r-value`
 
struct S { int m; };
struct S obj; // obj and obj.m are l-values

// ptr-> is an l-value; equivalent to (*ptr).m
// Note: ptr and *ptr are also l-values
struct S* ptr = &obj;

Logical operators

used to perform logical operations on boolean expressions. Combine multiple conditions and evaluate the result as either true or false. Conditions are evaluated from left to right The logical AND operator (&&) returns true if both of its operands are true. The logical OR operator (||) returns true if at least one of its operands is true. The logical NOT operator (!) is a unary operator that negates the value of its operand

bool a = true;
bool b = false;
bool result = a && b;  // result false (immediately if a is false)
bool result = a || b;  // result is true (immediately if a is true)
bool result = !a;  // result is false

Bitwise operators

Bitwise operators are used to perform operations on individual bits of binary numbers. These operators allow you to manipulate the binary representation of integers at a bit level. There are six bitwise operators in C++:

The bitwise AND (&) operator compares the corresponding bits of two operands and returns 1 if both bits are 1, otherwise it returns 0 The bitwise OR (|) operator compares the corresponding bits of two operands and returns 1 if at least one of the bits is 1, otherwise it returns 0 The bitwise XOR (^) operator compares the corresponding bits of two operands and returns 1 if the bits are different, otherwise it returns 0 The bitwise NOT (~) operator is a unary operator that flips the bits of its operand. It returns the one's complement of the operand. For example:

int a = 5; // binary representation: 0101
int b = 3; // binary representation: 0011
int result = a & b; // binary representation: 0001 (1 in decimal)
int result = a | b; // binary representation: 0111 (7 in decimal)
int result = a ^ b; // binary representation: 0110 (6 in decimal)
int result = ~a; // binary representation: 1010 (-6 in decimal)

The left shift << operator shifts the bits of the left operand to the left by a specified number of positions. The vacant positions are filled with zeros.
NB: left shift multiplies the original number by 2

The right shift >> operator shifts the bits of the left operand to the right by a specified number of positions. The vacant positions are filled with the sign bit (for signed types) or with zeros (for unsigned types). NB: right shifts are equivalent to dividing a number by 2

int a = 5; // binary representation: 0101
int result = a << 2; // binary representation: 010100 (20 in decimal)
int result = a >> 2; // binary representation: 0001 (1 in decimal)

Threads

A thread is the smallest unit of execution in a process. Multithreading involves the execution of multiple threads concurrently within a single program. It allows multiple threads to run independently, sharing the same resources (memory) but having their own execution contexts (stack and registers). Once a thread is created, its function or callable object is executed concurrently with other threads.

#import <thread>

void myThreadFunction(){
  // code to be executed on a separate thread
}

int main(){
  std::thread myThread(myThreadFunction); // after this line myThread immediately 
  //starts to run in parallel (with the main thread)

  // Here we can do other work in the main thread  

  myThread.join(); // tells main thread to wait the end of myThread before proceeding
  
  // or let it run independently
  //myThread.detach();

  // do other stuff here (only once myThreadFunction gets out of scope!)
}

When multiple threads access shared data concurrently, synchronization mechanisms are necessary to avoid data races. Mutexes, condition variables, and atomic operations are commonly used for synchronization.

Mutex

A mutex is a synchronization primitive used in C++ to protect shared resources from concurrent access by multiple threads. It ensures that only one thread can access the protected resource at a time, preventing data races and maintaining data integrity.

#include <iostream>  
#include <thread>  
#include <mutex> 

std::mutex myMutex; 
int sharedVariable = 0; 

void myThreadFunction() { 
  std::lock_guard<std::mutex> lock(myMutex); // Modify sharedVariable safely 
  sharedVariable++; 
} 
int main() { 
  std::thread myThread1(myThreadFunction); 
  std::thread myThread2(myThreadFunction); 
  myThread1.join(); 
  myThread2.join(); 
  // sharedVariable is now safely modified by both threads

Condition variables

std::condition_variable is a synchronization primitive in C++ that provides a way for threads to wait for a particular condition to be satisfied. It is often used in conjunction with a std::mutex to protect shared data and coordinate the execution of multiple threads

example Producer/Consumer:

#include <iostream>
#include <thread>
#include <mutex>
#include <condition_variable>
#include <queue>

std::mutex mutex_;
std::condition_variable condVar;
std::queue<int> dataQueue;

const int MAX_QUEUE_SIZE = 5;

void producer() {
  for (int i = 0; i < 10; ++i) {
    std::unique_lock<std::mutex> lock(mutex_);
    
    // Wait until there is space in the queue
    condVar.wait(lock, [] { return dataQueue.size() < MAX_QUEUE_SIZE; });

    // Produce data and add it to the queue
    dataQueue.push(i);
    std::cout << "Produced: " << i << std::endl;

    // Notify consumers that data is available
    condVar.notify_one();
  }
}

void consumer() {
  for (int i = 0; i < 10; ++i) {
    std::unique_lock<std::mutex> lock(mutex_);
    
    // Wait until there is data in the queue
    condVar.wait(lock, [] { return !dataQueue.empty(); });

    // Consume data from the queue
    int data = dataQueue.front();
    dataQueue.pop();
    std::cout << "Consumed: " << data << std::endl;

    // Notify producer that space is available
    condVar.notify_one();
  }
}

int main() {
  std::thread producerThread(producer);
  std::thread consumerThread(consumer);

  producerThread.join();
  consumerThread.join();
}

example Thread Synchronization

#include <iostream>
#include <thread>
#include <mutex>
#include <condition_variable>

std::mutex mutex_;
std::condition_variable condVar;
bool dataReady = false;

void producer() {
  std::this_thread::sleep_for(std::chrono::seconds(2)); // Simulate work
  {
    std::lock_guard<std::mutex> lock(mutex_);
    dataReady = true;
  }
  condVar.notify_one();
}

void consumer() {
  {
    std::unique_lock<std::mutex> lock(mutex_);
    condVar.wait(lock, [] { return dataReady; });
    // Do something with the data
    std::cout << "Consumed data." << std::endl;
  }
}

int main() {
  std::thread producerThread(producer);
  std::thread consumerThread(consumer);

  producerThread.join();
  consumerThread.join();
}

std::lock_guard vs std::unique_lock

  • If you need a simple and concise way to lock and unlock a mutex within a scope, and no manual unlocking is required, use std::lock_guard
  • If you need more flexibility, such as manual unlocking, deferred locking, or using the lock with condition variables, use std::unique_lock

The choice between std::lock_guard and std::unique_lock depends on the specific requirements of your code and the level of control you need over the associated mutex.

BOOST library semplification

Example Mutex

#include <boost/thread.hpp>

boost::mutex myMutex;
int sharedVariable = 0;

void myThreadFunction() {
  boost::unique_lock<boost::mutex> lock(myMutex);
  sharedVariable++; // Modify sharedVariable safely
}

int main() {
  boost::thread myThread1(myThreadFunction);
  boost::thread myThread2(myThreadFunction);
  myThread1.join();
  myThread2.join();
}

Lockfree Queue

Example Producer/Consumer

Boost provides a lock-free queue implementation, removing the need for explicit locking and unlocking when accessing the shared data structure.

#include <iostream>
#include <thread>
#include <boost/lockfree/queue.hpp>

const int MAX_QUEUE_SIZE = 5;
boost::lockfree::queue<int> dataQueue(MAX_QUEUE_SIZE);

void producer() {
  for (int i = 0; i < 10; ++i) {
    while (!dataQueue.push(i)) {
      // Queue is full, spin-wait or implement backoff strategy
    }
    std::cout << "Produced: " << i << std::endl;
  }
}

void consumer() {
  int data;
  while (true) {
    while (dataQueue.pop(data)) {
      // Process the consumed data
      std::cout << "Consumed: " << data << std::endl;
    }
    // Queue is empty, spin-wait or implement backoff strategy
  }
}

int main() {
  std::thread producerThread(producer);
  std::thread consumerThread(consumer);

  producerThread.join();
  consumerThread.join();

  return 0;
}

Atomic operations

The boost::atomic library provides atomic operations, which can be used to perform operations on variables in a way that is guaranteed to be atomic and avoid data races. Atomic operations are guaranteed to be executed as a single uninterruptable operation and are often implemented in a lock-free manner, making them suitable for scenarios where lock contention might be a concern. Atomic operations are typically used for simple, low-level operations on shared variables without the need for explicit locking.

Example Atomic Operation

#include <boost/atomic.hpp>

boost::atomic<int> atomicVariable(0);

void myThreadFunction() {
  atomicVariable.fetch_add(1);
}

Standard input

std::cin.get() is a member function of the cin object, which is an instance of the istream class. This function is used to read a single character from the standard input

#include  <iostream>
//...
char ch;
std::cout << "Enter a character: "; 

ch = std::cin.get(); // to read any char (including spaces and new line chars)
ch = std::cin << std::ws >> ch // read char (ignoring leading whitespace)

"You entered: " << ch << std::endl;
//...

std::cin is also used to read words and strings

std::string input; 
std::cout << "Enter a word: "; 

std::cin >> input; // to read a single word (sequence of chars without spaces)
std::getline(std::cin, input); // to read a whole line (until a newline character is encountered. It discards the newline character and stores the entire line)

std::cout << "You entered: " << input << std::endl;

Input file stream

use the <filesystem> library to work with file and directory paths, including obtaining the current working directory.

#include <filesystem>

// Get the current working directory 
std::filesystem::path currentPath = std::filesystem::current_path();

use the std::ifstream class from the <fstream> header to check if a file exists. Use getline(stream, line) to update the current line from the file stream

#include <fstream>

// ...
std::string filename = "example.txt";
std::ifstream ifs(filename); // Create an ifstream object

// Check if the file is open, which indicates that it exists
if (ifs.is_open()) {
  std::cout << "File exists." << std::endl;
  
  // You can proceed to read from or write to the file here if needed
  std::string line;
  while(getline(ifs, line)) {
    if (line.find("somethingToSearch") != std::string::npos){
      // somethingToSearch has been found
    }
  }  
   
  ifs.close(); // Close the file after using it
} else { 
  std::cout << "File does not exist." << std::endl; 
}

CMake

see https://stackoverflow.com/questions/49996260/how-to-use-target-sources-command-with-interface-library

subdirectories

root/CMakeLists.txt
root/Graphs
root/include

CMakeLists.txt in root folder

cmake_minimum_required(VERSION 3.0.0)
project(thecherno VERSION 0.1.0)
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_FLAGS  "${CMAKE_CXX_FLAGS} -std=c++17")

add_subdirectory(Graphs)

include

Graphs/CMakeLists.txt
Grphs/include

CMakeLists.txt in subdir Graphs

cmake_minimum_required(VERSION 3.0.0)
project(Graphs VERSION 0.1.0)

add_executable(Graphs Graphs.cpp)

#target_sources(Graphs PRIVATE log.cpp)

target_include_directories(Graphs PUBLIC ${CMAKE_CURRENT_SOURCE_DIR}/include)
target_include_directories(Graphs PUBLIC ${CMAKE_CURRENT_SOURCE_DIR}/../include)

multiple translation units / main entry points

Graphs/CMakeLists.txt Grphs/include

cmake_minimum_required(VERSION 3.0.0)
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_FLAGS  "${CMAKE_CXX_FLAGS} -std=c++17")

project(miketests VERSION 0.1.0)
add_executable(main1 main.cpp anotherTransUnit.cpp)
add_executable(main2 anotherMain.cpp)

library

ToDo!