|
|
|
|
Functions | ||||
operator newoperator new[] | ||||
(C++11) | ||||
Classes | ||||
(C++11) | ||||
(C++17) | ||||
Types | ||||
Objects | ||||
(C++20) | ||||
Object access | ||||
(C++17) |
Defined in header <new> | ||
Replaceable allocation functions | ||
void* operator new (std::size_t count); | (1) | |
void* operator new[](std::size_t count); | (2) | |
void* operator new (std::size_t count,std::align_val_t al); | (3) | (since C++17) |
void* operator new[](std::size_t count,std::align_val_t al); | (4) | (since C++17) |
Replaceable non-throwing allocation functions | ||
void* operator new (std::size_t count,conststd::nothrow_t& tag); | (5) | (noexcept since C++11) |
void* operator new[](std::size_t count,conststd::nothrow_t& tag); | (6) | (noexcept since C++11) |
void* operator new (std::size_t count,std::align_val_t al, conststd::nothrow_t& tag)noexcept; | (7) | (since C++17) |
void* operator new[](std::size_t count,std::align_val_t al, conststd::nothrow_t& tag)noexcept; | (8) | (since C++17) |
Non-allocating placement allocation functions | ||
void* operator new (std::size_t count,void* ptr); | (9) | (noexcept since C++11) (constexpr since C++26) |
void* operator new[](std::size_t count,void* ptr); | (10) | (noexcept since C++11) (constexpr since C++26) |
User-defined placement allocation functions | ||
void* operator new (std::size_t count,/* args... */); | (11) | |
void* operator new[](std::size_t count,/* args... */); | (12) | |
void* operator new (std::size_t count, std::align_val_t al,/* args... */); | (13) | (since C++17) |
void* operator new[](std::size_t count, std::align_val_t al,/* args... */); | (14) | (since C++17) |
Class-specific allocation functions | ||
void* T::operator new (std::size_t count); | (15) | |
void* T::operator new[](std::size_t count); | (16) | |
void* T::operator new (std::size_t count,std::align_val_t al); | (17) | (since C++17) |
void* T::operator new[](std::size_t count,std::align_val_t al); | (18) | (since C++17) |
Class-specific placement allocation functions | ||
void* T::operator new (std::size_t count,/* args... */); | (19) | |
void* T::operator new[](std::size_t count,/* args... */); | (20) | |
void* T::operator new (std::size_t count, std::align_val_t al,/* args... */); | (21) | (since C++17) |
void* T::operator new[](std::size_t count, std::align_val_t al,/* args... */); | (22) | (since C++17) |
Attempts to allocate requested number of bytes, and the allocation request can fail (even if the requested number of bytes is zero). These allocation functions are called bynew expressions to allocate memory in which new object would then be initialized. They may also be called using regular function call syntax.
Overloads(1-4) are implicitly declared in each translation unit even if the<new> header is not included.
Seenew expression for the criteria of selecting overload.
Contents |
count | - | number of bytes to allocate |
ptr | - | pointer to a memory area to initialize the object at |
tag | - | disambiguation tag used to select non-throwing overloads |
al | - | alignment to use, invalid value leads to undefined behavior |
Overloads(1-8) arereplaceable. The effects of the default versions are:
Onfreestanding implementations, it is implementation-defined whether the default versions of(1-8) satisfy the behaviors required above. Freestanding implementations are recommended that if any of these default versions meet the requirements of a hosted implementation, they all should. | (since C++26) |
Globaloperator
snew/delete replacement:
#include <cstdio>#include <cstdlib>#include <new> // no inline, required by [replacement.functions]/3void* operator new(std::size_t sz){std::printf("1) new(size_t), size = %zu\n", sz);if(sz==0)++sz;// avoid std::malloc(0) which may return nullptr on success if(void*ptr=std::malloc(sz))return ptr; throwstd::bad_alloc{};// required by [new.delete.single]/3} // no inline, required by [replacement.functions]/3void* operator new[](std::size_t sz){std::printf("2) new[](size_t), size = %zu\n", sz);if(sz==0)++sz;// avoid std::malloc(0) which may return nullptr on success if(void*ptr=std::malloc(sz))return ptr; throwstd::bad_alloc{};// required by [new.delete.single]/3} voidoperator delete(void* ptr)noexcept{std::puts("3) delete(void*)");std::free(ptr);} voidoperator delete(void* ptr,std::size_t size)noexcept{std::printf("4) delete(void*, size_t), size = %zu\n", size);std::free(ptr);} voidoperator delete[](void* ptr)noexcept{std::puts("5) delete[](void* ptr)");std::free(ptr);} voidoperator delete[](void* ptr,std::size_t size)noexcept{std::printf("6) delete[](void*, size_t), size = %zu\n", size);std::free(ptr);} int main(){int* p1= newint; delete p1; int* p2= newint[10];// guaranteed to call the replacement in C++11 delete[] p2;}
Possible output:
// Compiled with GCC-5 in C++17 mode to obtain the following:1) op new(size_t), size = 44) op delete(void*, size_t), size = 42) op new[](size_t), size = 405) op delete[](void* ptr)
Overloads ofoperator new
andoperator new[]
with additional user-defined parameters ("placement forms", versions(11-14)) may be declared at global scope as usual, and are called by the matchingplacement forms ofnew expressions.
The standard library's non-allocating placement forms ofoperator new
(9,10) cannot be replaced and can only be customized if the placementnew expression did not use the::new syntax, by providing a class-specific placementnew(19,20) with matching signature:void* T::operator new(std::size_t,void*) orvoid* T::operator new[](std::size_t,void*).
The placement formvoid* operator new(std::size_t,std::size_t) is not allowed because the matching signature of the deallocation function,voidoperator delete(void*,std::size_t), is a usual (not placement) deallocation function. | (since C++14) |
Both single-object and array allocation functions may be defined as public static member functions of a class (versions(15-18)). If defined, these allocation functions are called bynew expressions to allocate memory for single objects and arrays of this class, unless thenew expression used the form::new which bypasses class-scope lookup. The keywordstatic is optional for these functions: whether used or not, the allocation function is a static member function.
Thenew expression looks for appropriate allocation function's name firstly in the class scope, and after that in the global scope. Note, that as pername lookup rules, any allocation functions declared in class scope hides all global allocation functions for thenew expressions that attempt to allocate objects of this class.
When allocating objects and arrays of objects whose alignment exceeds__STDCPP_DEFAULT_NEW_ALIGNMENT__, overload resolution is performed twice: first, for alignment-aware function signatures, then for alignment-unaware function signatures. This means that if a class with extended alignment has an alignment-unaware class-specific allocation function, it is the function that will be called, not the global alignment-aware allocation function. This is intentional: the class member is expected to know best how to handle that class. | (since C++17) |
When allocating objects and arrays of objects whose alignment does not exceed__STDCPP_DEFAULT_NEW_ALIGNMENT__, overload resolution is performed twice: first, for alignment-unaware function signatures, then for alignment-aware function signatures. | (since C++20) |
#include <cstddef>#include <iostream> // class-specific allocation functionsstruct X{staticvoid* operator new(std::size_t count){std::cout<<"custom new for size "<< count<<'\n';return::operator new(count);} staticvoid* operator new[](std::size_t count){std::cout<<"custom new[] for size "<< count<<'\n';return::operator new[](count);}}; int main(){ X* p1= new X; delete p1; X* p2= new X[10]; delete[] p2;}
Possible output:
custom new for size 1custom new[] for size 10
Overloads ofoperator new
andoperator new[]
with additional user-defined parameters ("placement forms"), may also be defined as class members(19-22)). When the placementnew expression with the matching signature looks for the corresponding allocation function to call, it begins at class scope before examining the global scope, and if the class-specific placementnew is provided, it is called.
When allocating objects and arrays of objects whose alignment exceeds__STDCPP_DEFAULT_NEW_ALIGNMENT__, overload resolution for placement forms is performed twice just as for regular forms: first, for alignment-aware function signatures, then for alignment-unaware function signatures. | (since C++17) |
When allocating objects and arrays of objects whose alignment does not exceed__STDCPP_DEFAULT_NEW_ALIGNMENT__, overload resolution for placement forms is performed twice just as for regular forms: first, for alignment-unaware function signatures, then for alignment-aware function signatures. | (since C++20) |
#include <cstddef>#include <iostream>#include <stdexcept> struct X{ X(){throwstd::runtime_error("");} // custom placement newstaticvoid* operator new(std::size_t count,bool b){std::cout<<"custom placement new called, b = "<< b<<'\n';return::operator new(count);} // custom placement deletestaticvoidoperator delete(void* ptr,bool b){std::cout<<"custom placement delete called, b = "<< b<<'\n';::operator delete(ptr);}}; int main(){try{[[maybe_unused]] X* p1= new(true) X;}catch(conststd::exception&){}}
Output:
custom placement new called, b = 1custom placement delete called, b = 1
If class-leveloperator new
is a template function, it must have the return type ofvoid*, the first argumentstd::size_t, and it must have two or more parameters. In other words, only placement forms can be templates.
Even though the non-allocating placementnew(9,10) cannot be replaced, a function with the same signature may be defined at class scope as described above. In addition, global overloads that look like placementnew but take a non-void pointer type as the second argument are allowed, so the code that wants to ensure that the true placementnew is called (e.g.std::allocator::construct), must use::new and also cast the pointer tovoid*.
If the behavior of a deallocation function does not satisfy the default constraints, the behavior is undefined.
The following functions are required to be thread-safe:
Calls to these functions that allocate or deallocate a particular unit of storage occur in a single total order, and each such deallocation callhappens-before the next allocation (if any) in this order. | (since C++11) |
It is unspecified whether library versions ofoperator new
make any calls tostd::malloc orstd::aligned_alloc(since C++17).
For loading a large file, file mapping via OS-specific functions, e.g.,mmap
on POSIX orCreateFileMapping
(A
/W
) along withMapViewOfFile
on Windows, is preferable to allocating a buffer for file reading.
Feature-test macro | Value | Std | Feature |
---|---|---|---|
__cpp_lib_freestanding_operator_new | 202306L | (C++26) | freestanding support for replaceableoperator new[1] |
0 | (C++26) | no freestanding support | |
__cpp_lib_constexpr_new | 202406L | (C++26) | constexpr placementnew andnew[] |
The following behavior-changing defect reports were applied retroactively to previously published C++ standards.
DR | Applied to | Behavior as published | Correct behavior |
---|---|---|---|
CWG 521 | C++98 | any class derived fromstd::bad_alloc could be thrown, even if thestd::bad_alloc base is ambiguous or inaccessible | the exception thrown should match a handler of typestd::bad_alloc |
LWG 9 | C++98 | multiple calls for allocating zero bytes could yield the same pointer | only allowed if all such previously yielded pointers have been passed to deallocation functions |
LWG 206 | C++98 | replacing the replaceable allocation functions did not affect the default behaviors of the corresponding replaceable non-throwing allocation functions | the default behaviors change accordingly |
LWG 404 | C++98 | replacements of the replaceable allocation functions could be declaredinline | prohibited, no diagnostic required |
[static](C++23) | allocates memory usingAllocator (public static member function of std::generator<Ref,V,Allocator>::promise_type )[edit] |
deallocation functions (function)[edit] | |
(C++11) | obtains the current new handler (function)[edit] |
registers a new handler (function)[edit] | |
(deprecated in C++17)(removed in C++20) | obtains uninitialized storage (function template)[edit] |
allocates memory (function)[edit] | |
(C++17) | allocates aligned memory (function)[edit] |