4 classes in the following codes: A, B, C and D.
They all have a member operator new[]
.
Besides,
- B has a constructor;
- C has a destructor;
- D has a member
operator delete[]
.
The Parameter size
of member operator new[]
and the sizeof
of the 4 classes are output:
new[] A 40
new[] B 40
new[] C 48
new[] D 48
sizeof(A) 4
sizeof(B) 4
sizeof(C) 4
sizeof(D) 4
What's the reason for the differences of size
?
Codes(ugly I know):
#include <iostream>
using namespace std;
class A {
int i;
public:
static void* operator new[](std::size_t size) throw(std::bad_alloc) {
cout << "new[] A " << size << endl;
return malloc(size);
}
};
class B {
int i;
public:
static void* operator new[](std::size_t size) throw(std::bad_alloc) {
cout << "new[] B " << size << endl;
return malloc(size);
}
B() {}
};
class C {
int i;
public:
static void* operator new[](std::size_t size) throw(std::bad_alloc) {
cout << "new[] C " << size << endl;
return malloc(size);
}
~C() {}
};
class D {
int i;
public:
static void* operator new[](std::size_t size) throw(std::bad_alloc) {
cout << "new[] D " << size << endl;
return malloc(size);
}
static void operator delete[](void* p, std::size_t size) {
free(p);
}
};
int main() {
A* a = new A[10];
B* b = new B[10];
C* c = new C[10];
D* d = new D[10];
cout << "sizeof(A) " << sizeof(A) << endl;
cout << "sizeof(B) " << sizeof(B) << endl;
cout << "sizeof(C) " << sizeof(C) << endl;
cout << "sizeof(D) " << sizeof(D) << endl;
}
About OS and compiler:
Compiling: same results for clang++ and g++
clang++ test.cpp -o test -std=c++11
g++ test.cpp -o test -std=c++11
OS: Linux Mint 18.2 Cinnamon 64-bit
Compilers:
clang++ -v
clang version 3.8.0-2ubuntu4 (tags/RELEASE_380/final)
Target: x86_64-pc-linux-gnu
Thread model: posix
InstalledDir: /usr/bin
Found candidate GCC installation: /usr/bin/../lib/gcc/x86_64-linux-gnu/4.9
Found candidate GCC installation: /usr/bin/../lib/gcc/x86_64-linux-gnu/4.9.3
Found candidate GCC installation: /usr/bin/../lib/gcc/x86_64-linux-gnu/5.4.0
Found candidate GCC installation: /usr/bin/../lib/gcc/x86_64-linux-gnu/6.0.0
Found candidate GCC installation: /usr/lib/gcc/x86_64-linux-gnu/4.9
Found candidate GCC installation: /usr/lib/gcc/x86_64-linux-gnu/4.9.3
Found candidate GCC installation: /usr/lib/gcc/x86_64-linux-gnu/5.4.0
Found candidate GCC installation: /usr/lib/gcc/x86_64-linux-gnu/6.0.0
Selected GCC installation: /usr/bin/../lib/gcc/x86_64-linux-gnu/5.4.0
Candidate multilib: .;@m64
Selected multilib: .;@m64
Found CUDA installation: /usr/local/cuda
g++ -v
Using built-in specs.
COLLECT_GCC=g++
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-linux-gnu/5/lto-wrapper
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 5.4.0-6ubuntu1~16.04.4' --with-bugurl=file:///usr/share/doc/gcc-5/README.Bugs --enable-languages=c,ada,c++,java,go,d,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-5 --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --with-default-libstdcxx-abi=new --enable-gnu-unique-object --disable-vtable-verify --enable-libmpx --enable-plugin --with-system-zlib --disable-browser-plugin --enable-java-awt=gtk --enable-gtk-cairo --with-java-home=/usr/lib/jvm/java-1.5.0-gcj-5-amd64/jre --enable-java-home --with-jvm-root-dir=/usr/lib/jvm/java-1.5.0-gcj-5-amd64 --with-jvm-jar-dir=/usr/lib/jvm-exports/java-1.5.0-gcj-5-amd64 --with-arch-directory=amd64 --with-ecj-jar=/usr/share/java/eclipse-ecj.jar --enable-objc-gc --enable-multiarch --disable-werror --with-arch-32=i686 --with-abi=m64 --with-multilib-list=m32,m64,mx32 --enable-multilib --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 5.4.0 20160609 (Ubuntu 5.4.0-6ubuntu1~16.04.4)
prog.cc:7:51: error: ISO C++1z does not allow dynamic exception specifications
with GCC 8. @JohnZwinck (s)he mentions 64 bit. – Newtonnextsize
parameter todelete[]
. But just speculations. Also, if the class members were something more complicated than anint
, the results could have been different. – Diaphanedelete []
expression, the destructor must be called for every allocated object. Similar story if anoperator delete[](void &, size_t)
is provided, since it is necessary to pass the size from the new expression somehow. – DesolationA
andB
, you don't care because there is nothing to destruct (you don't have to know how many elements you'll have to destruct), and you don't need to keep the information to call the built-indelete[]
(the compiler does not need it). ForC
, the compiler needs to know how much elements must be destroyed when callingdelete[]
, and forD
it needs to know what to pass toD::operator delete[]
. You can verify these easily by looking at the generated assembly. – Jurel