One of the C++ features that sets it apart from other languages
... is that you have to do memory allocation manually. But let's leave that aside:
- allocate on the heap when an object has to be long-lived, i.e. must outlive a certain scope, and is expensive or impossible to copy or move,
- allocate on the heap when an object is large (where large might mean several kilobytes if you want to be on the safe side) to prevent stack overflows, even if the object is only needed temporarily,
- allocate on the heap if you're using the pimpl (compiler firewall) idiom,
- allocate variable-sized arrays on the heap,
- allocate on the stack otherwise because it's so much more convenient.
Note that in the second rule, by "large object" I mean something like
char buffer[1024 * 1024]; // 1MB buffer
but not
std::vector<char> buffer(1024 * 1024);
since the second is actually a very small object wrapping a pointer to a heap-allocated buffer.
As for pointer vs. value members:
- use a pointer if you need heap allocation,
- use a pointer if you're sharing structure,
- use a pointer or reference for polymorphism,
- use a reference if you get an object from client code and the client promises to keep it alive,
- use a value in most other cases.
The use of smart pointers is of course recommended where appropriate. Note that you can use a reference in case of heap allocation because you can always delete &ref
, but I wouldn't recommend doing that. References are pointers in disguise with only one difference (a reference can't be null), but they also signal a different intent.