QScopedPointer Class▲
-
Header: QScopedPointer
-
CMake:
find_package(Qt6 REQUIRED COMPONENTS Core)
target_link_libraries(mytarget PRIVATE Qt6::Core)
-
qmake: QT += core
-
Inherited By: QScopedArrayPointer
-
Group: QScopedPointer is part of misc
Detailed Description▲
Managing heap allocated objects manually is hard and error prone, with the common result that code leaks memory and is hard to maintain. QScopedPointer is a small utility class that heavily simplifies this by assigning stack-based memory ownership to heap allocations, more generally called resource acquisition is initialization(RAII).
QScopedPointer guarantees that the object pointed to will get deleted when the current scope disappears.
Consider this function which does heap allocations, and has various exit points:
void
myFunction(bool
useSubClass)
{
MyClass *
p =
useSubClass ? new
MyClass() : new
MySubClass;
QIODevice *
device =
handsOverOwnership();
if
(m_value &
gt; 3
) {
delete
p;
delete
device;
return
;
}
try
{
process(device);
}
catch
(...) {
delete
p;
delete
device;
throw
;
}
delete
p;
delete
device;
}
It's encumbered by the manual delete calls. With QScopedPointer, the code can be simplified to:
void
myFunction(bool
useSubClass)
{
// assuming that MyClass has a virtual destructor
QScopedPointer&
lt;MyClass&
gt; p(useSubClass ? new
MyClass() : new
MySubClass);
QScopedPointer&
lt;QIODevice&
gt; device(handsOverOwnership());
if
(m_value &
gt; 3
)
return
;
process(device);
}
The code the compiler generates for QScopedPointer is the same as when writing it manually. Code that makes use of delete are candidates for QScopedPointer usage (and if not, possibly another type of smart pointer such as QSharedPointer). QScopedPointer intentionally has no copy constructor or assignment operator, such that ownership and lifetime is clearly communicated.
The const qualification on a regular C++ pointer can also be expressed with a QScopedPointer:
const
QWidget *
const
p =
new
QWidget();
// is equivalent to:
const
QScopedPointer&
lt;const
QWidget&
gt; p(new
QWidget());
QWidget *
const
p =
new
QWidget();
// is equivalent to:
const
QScopedPointer&
lt;QWidget&
gt; p(new
QWidget());
const
QWidget *
p =
new
QWidget();
// is equivalent to:
QScopedPointer&
lt;const
QWidget&
gt; p(new
QWidget());
Custom Cleanup Handlers▲
Arrays as well as pointers that have been allocated with malloc must not be deleted using delete. QScopedPointer's second template parameter can be used for custom cleanup handlers.
The following custom cleanup handlers exist:
-
QScopedPointerDeleter - the default, deletes the pointer using delete
-
QScopedPointerArrayDeleter - deletes the pointer using delete []. Use this handler for pointers that were allocated with new [].
-
QScopedPointerPodDeleter - deletes the pointer using free(). Use this handler for pointers that were allocated with malloc().
-
QScopedPointerDeleteLater - deletes a pointer by calling deleteLater() on it. Use this handler for pointers to QObject's that are actively participating in a QEventLoop.
You can pass your own classes as handlers, provided that they have a public static function void cleanup(T *pointer).
// this QScopedPointer deletes its data using the delete[] operator:
QScopedPointer&
lt;int
, QScopedPointerArrayDeleter&
lt;int
&
gt; &
gt; arrayPointer(new
int
[42
]);
// this QScopedPointer frees its data using free():
QScopedPointer&
lt;int
, QScopedPointerPodDeleter&
gt; podPointer(reinterpret_cast
&
lt;int
*&
gt;(malloc(42
)));
// this struct calls "myCustomDeallocator" to delete the pointer
struct
ScopedPointerCustomDeleter
{
static
inline
void
cleanup(MyCustomClass *
pointer)
{
myCustomDeallocator(pointer);
}
}
;
// QScopedPointer using a custom deleter:
QScopedPointer&
lt;MyCustomClass, ScopedPointerCustomDeleter&
gt; customPointer(new
MyCustomClass);
Forward Declared Pointers▲
Classes that are forward declared can be used within QScopedPointer, as long as the destructor of the forward declared class is available whenever a QScopedPointer needs to clean up.
Concretely, this means that all classes containing a QScopedPointer that points to a forward declared class must have non-inline constructors, destructors and assignment operators:
class
MyPrivateClass; // forward declare MyPrivateClass
class
MyClass
{
private
:
QScopedPointer&
lt;MyPrivateClass&
gt; privatePtr; // QScopedPointer to forward declared class
public
:
MyClass(); // OK
inline
~
MyClass() {}
// VIOLATION - Destructor must not be inline
private
:
Q_DISABLE_COPY(MyClass) // OK - copy constructor and assignment operators
// are now disabled, so the compiler won't implicitly
// generate them.
}
;
Otherwise, the compiler outputs a warning about not being able to destruct MyPrivateClass.
See Also▲
See also QSharedPointer
Member Function Documentation▲
[explicit] QScopedPointer::QScopedPointer(T *p = nullptr)▲
Constructs this QScopedPointer instance and sets its pointer to p.
QScopedPointer::~QScopedPointer()▲
Destroys this QScopedPointer object. Delete the object its pointer points to.
T *QScopedPointer::data() const▲
Returns the value of the pointer referenced by this object. QScopedPointer still owns the object pointed to.
T *QScopedPointer::get() const▲
Same as data().
bool QScopedPointer::isNull() const▲
Returns true if this object refers to nullptr.
void QScopedPointer::reset(T *other = nullptr)▲
Deletes the existing object it is pointing to (if any), and sets its pointer to other. QScopedPointer now owns other and will delete it in its destructor.
[explicit] bool QScopedPointer::operator bool() const▲
Returns true if the contained pointer is not nullptr. This function is suitable for use in if-constructs, like:
if
(scopedPointer) {
...
}
See Also▲
See also isNull()
bool QScopedPointer::operator!() const▲
T &QScopedPointer::operator*() const▲
Provides access to the scoped pointer's object.
If the contained pointer is nullptr, behavior is undefined.
See Also▲
See also isNull()
T *QScopedPointer::operator->() const▲
Related Non-Members▲
bool operator!=(const QScopedPointer<T, Cleanup> &lhs, const QScopedPointer<T, Cleanup> &rhs)▲
Returns true if lhs and rhs refer to distinct pointers.
bool operator!=(const QScopedPointer<T, Cleanup> &lhs, std::nullptr_t)▲
Returns true if lhs refers to a valid (i.e. non-null) pointer.
See Also▲
See also QScopedPointer::isNull()
bool operator!=(std::nullptr_t, const QScopedPointer<T, Cleanup> &rhs)▲
Returns true if rhs refers to a valid (i.e. non-null) pointer.
See Also▲
See also QScopedPointer::isNull()
bool operator==(const QScopedPointer<T, Cleanup> &lhs, const QScopedPointer<T, Cleanup> &rhs)▲
Returns true if lhs and rhs refer to the same pointer.
bool operator==(const QScopedPointer<T, Cleanup> &lhs, std::nullptr_t)▲
bool operator==(std::nullptr_t, const QScopedPointer<T, Cleanup> &rhs)▲
Obsolete Members for QScopedPointer▲
The following members of class QScopedPointer are deprecated. We strongly advise against using them in new code.
Obsolete Member Function Documentation▲
T *QScopedPointer::take()▲
This function is deprecated since 6.1. We strongly advise against using it in new code.
Use std::unique_ptr and release() instead.
Returns the value of the pointer referenced by this object. The pointer of this QScopedPointer object will be reset to nullptr.
Callers of this function take ownership of the pointer.
Obsolete Related Non-Members▲
void swap(QScopedPointer<T, Cleanup> &lhs, QScopedPointer<T, Cleanup> &rhs)▲
This function is deprecated since 6.1. We strongly advise against using it in new code.
Use std::unique_ptr instead; this function may let a pointer escape its scope.
Swaps lhs with rhs.