Raw Pointers
🚀 Basics
-
new and delete:
- If you use new, you must use delete — otherwise, memory leak! see code.
new[]
:size must be provided as part of the signature.1 2 3
new int[10](); // value-initialized to 0 new int[5]{1, 2, 3}; // OK: extra elements zero-initialized new int[]{1, 2, 3}; // ❌ error: size must be explicit
- new[] must match delete[]. A simple
delete
will yield undefined behaviour. - delete will take not into account the number of objects, butdelete[]
will.
- delete[] should not be used on regularchar*
, unless it ischar* c = new char[size_];
.
-
Structs that contain pointers
- Default-initialized pointers inside a struct are
nullptr
— but that means no memory is allocated. → You mustnew
those inner pointers manually.
- Default-initialized pointers inside a struct are
📌 Using Pointers
- Initialization:
- Pointers are not auto-initialized.
- Always explicitly initialize to nullptr in constructors or field declarations.
-
Getting array size is unsafe:
sizeof(int*) / sizeof(int); // ❌ not safe
- When to use raw pointers:
- Small, performance-critical code with clear ownership semantics.
-
If your function doesn’t care about ownership, prefer:
1 2 3
void f(widget& w); // ✅ clear, safe void f(widget* w); // ✅ OK if nullable void f(shared_ptr<widget>& w); // ❌ misleading — might reset or alias others
- After
delete
orfree
, set the pointer tonullptr
:- Avoids use-after-free or double-free errors.
1 2
delete ptr_; ptr_ = nullptr;
- ⚠️ Cautions: Always initialize your pointers! Uninitialized pointers cause random crashes or data corruption.
- Avoids use-after-free or double-free errors.
- Drawbacks of
int* i
:- Is it a pointer to a single int or an array?
- Does it own the memory?
- If yes:
- Who deletes it?
- Use delete or delete[]?
- Is it dangling?
- Did you miss deleting one copy?
- If yes:
- Raw pointers have no metadata — they don’t convey ownership or lifetime info. That’s why modern C++ prefers smart pointers.
- const and Pointers
int* const ptr
→constant pointer to int
const int* ptr
→pointer to constant int
const shared_ptr<T>
≈T* const
→ shared_ptr can’t be reassigned, but pointee can be mutatedshared_ptr<const T>
→ pointer to const object- If you’re using
shared_ptr<const T>
, you must initialize in initializer list, especially for class members.
- If you’re using
📚 Arrays
arr[1] = 10;
will compile even if arr doesn’t own the memory. → Might segfault at runtime depending on what arr points to.- Multidimensional arrays:
- For
arr[1][2]
to work, arr must beint**
, notint*
. - Or use
std::vector<std::vector<int>>
for safety. - No clean way to convert pointer back to array form:
- Raw pointers have no size metadata.
- You lose bounds-checking and iteration safety.
- For
Casting
static_pointer_cast
vs dynamic_pointer_cast
:
RTTI
is run time type information, if a base class has at least a virtual function, then base class ptr can be dynamic_cast to derived class- (downcast, upcast is derived -> base)
dynamic_cast
is only for downcasting. dynamic_cast can happen in runtime.- If downcasting fails (not derived class), nullptr is returned.
- You can downcast references as well → throws
std::bad_cast
- Safer, but slower than
static_cast
static_cast
happens during compile time, no RTTI is needed.- If you’re sure you can downcast to a specific type, use static_cast since it’s cheaper, and the language allows you to do so
static_pointer_cast
vsstatic_cast
:static_pointer_cast
works on shared_ptrs, because you can’t cast its type directly.1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
#include <iostream> using namespace std; class B { //virtual void fun() {} // NEED TO ADD THIS!! }; class D : public B { }; int main() { B* b = new D; D* d = dynamic_cast<D*>(b); // 1. use dynamic_cast if we're not sure if we can succeed if (d != NULL) cout << "works"; else cout << "cannot cast B* to D*"; // 2. cpp still allows you to use static_ptr_cast std::static_pointer_cast<DerivedClass>(ptr_to_base)->f(); // 3. even static_pointer_cast static_cast<DerivedClass*>(ptr_to_base.get())->f(); // equivalent to above return 0; }