Following a question asked here earlier today and multitudes of similary themed questions, I'm here to ask about this problem from stadard's viewpoint.
struct Base
{
int member;
};
struct Derived : Base
{
int another_member;
};
int main()
{
Base* p = new Derived[10]; // (1)
p[1].member = 42; // (2)
delete[] p; // (3)
}
According to standard (1)
is well-formed, because Dervied*
(which is the result of new-expression) can be implicitly converted to Base*
(C++11 draft, §4.10/3):
A prvalue of type “pointer to cv D”, where D is a class type, can be converted to a prvalue of type “pointer to cv B”, where B is a base class (Clause 10) of D. If B is an inaccessible (Clause 11) or ambiguous (10.2) base class of D, a program that necessitates this conversion is ill-formed. The result of the conversion is a pointer to the base class subobject of the derived class object. The null pointer value is converted to the null pointer value of the destination type.
(3)
leads to undefined behaviour because of §5.3.5/3:
In the first alternative (delete object), if the static type of the object to be deleted is different from its dynamic type, the static type shall be a base class of the dynamic type of the object to be deleted and the static type shall have a virtual destructor or the behavior is undefined. In the second alternative (delete array) if the dynamic type of the object to be deleted differs from its static type, the behavior is undefined.
Is (2)
legal according to standard or does it lead to ill-formed program or undefined behaviour?
edit: Better wording
(2)
is ill-formed. I spent last two hours searching something like what Bo Persson said, i.e.(p + n)
uses static type ofp
to compute offset, but I got the feeling that paragraph concerningoperator+
doesn't imply this. – Vitus