Why doesn\'t new
return NULL
on failure? Why does it throw an exception only on failure?
As it returns a pointer to the object on successes
Just a historical note, not an answer to this question (there are plenty of good answers)... a long long time ago when dinosaurs roamed the earth and Turbo C++ was ruling the world of C++ programmers, new
actually returned NULL
. Quote from a book of those days (0bj3ct-0r13nt3d Pr0gr4mm1ng with ANSI and Turbo C++ - title intentionally obfuscated so that no one by mistake reads it anymore) page 115.
If the new operator fails to allocate memory it returns NULL which can be used to detect failure or success of new operator.
because of this legacy code is full with a lot of NULL checking ... a real nightmare to bring them up to the current standard...
However, here is a tiny piece of the C++ standard 5.3.4/13:
[ Note: unless an allocation function is declared with a non-throwing exception-specification (15.4), it indicates failure to allocate storage by throwing a std::bad_alloc exception (Clause 15, 18.6.2.1); it returns a non-null pointer otherwise. If the allocation function is declared with a non-throwing exception-specification, it returns null to indicate failure to allocate storage and a non-null pointer otherwise. —end note ] If the allocation function returns null, initialization shall not be done, the deallocation function shall not be called, and the value of the new-expression shall be null.
which tells you that in some special cases, new
can return NULL
In C, all requests for dynamic memory are made by function calls; it's necessary that code check any allocation requests to see if they return null
, but there is a well-defined place where such checks can be made (i.e. right after the call). In C++, while it is possible for code to construct objects by calling new
, most objects are created by creating an object in which they play a part.
Given:
class Story
{
Potter harry;
Weasley ron,ginny,george,fred;
Grainger hermione;
Longbottom neville;
Creevy colin;
Story()
{
}
}
If the constructor for harry
tries to create a new
object and fails, the only way to prevent the system from trying uselessly to create objects for all the other fields would be to have the constructor throw an exception. Even though the constructor for harry
failed, the other constructors might succeed in creating those objects if they require less memory than a Potter
object; since the failure to create harry
would render the Story
object useless, however, any effort spent creating objects for the remaining fields would not only be wasted, but would require the system to expend even more effort destroying those uselessly-created objects.
Since in most cases the only thing the system would be able to do when new
fails is to throw an exception, it's easier to have new
throw the exception itself and have code catch the exception if necessary, than to require that all code which calls new
must check whether it succeeded and throw an exception if it didn't.