1
#include <stddef.h>
2
#include <stdlib.h>
3
#include <stdio.h>
4
#include <vector>
5data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
6
#include <stdexcept>
7data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
8data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
9
#if defined(_MSC_VER) && _MSC_VER<1400
10
#include <new.h>
11data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
namespace std
{ using ::set_new_handler; using ::new_handler; }
12
#else
13
#include <new>
14
#endif
15data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
16data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
17data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
void* allocate(size_t size)
{
18
if (size==0) size = 1;
19data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
for (void* p=0;;)
{
20
p = malloc(size);
21data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
if (p)
{
22
printf("allocate %p\n",p);
23
return p;
24
}
25
std::new_handler handler = std::set_new_handler(0);
26
std::set_new_handler(handler);
27
if (handler)
28
handler();
29
else
30
throw std::bad_alloc();
31
}
32
}
33data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
34data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
void deallocate(void* ptr)
{
35
printf("deallocate %p\n",ptr);
36
free(ptr);
37
}
38data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
39data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
void* operator new(size_t size)
{ return allocate(size); }
40data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
void* operator new[](size_t size)
{ return allocate(size); }
41data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
void operator delete(void* ptr)
{ deallocate(ptr); }
42data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
43data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
class C
{
44
static int count;
45
public:
46
static bool fail;
47data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
C()
{
48
if (fail)
49
throw std::exception();
50
printf("C::C(),%d\n",++count);
51
}
52data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
~C()
{
53
printf("C::~C(),%d\n",count--);
54
}
55data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
C(const C& )
{
56
printf("C::(const C&),%d\n",++count);
57
}
58data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
59data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
60
//void* operator new(size_t,void* place) { return place; }
61data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
void* operator new(size_t size)
{ return allocate(size); }
62data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
void* operator new[](size_t size)
{ return allocate(size); }
63data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
void operator delete(void* ptr)
{ deallocate(ptr); }
64
};
65
bool C::fail;
66
int C::count;
67data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
68data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
struct S
{
69
static bool fail;
70data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
S()
{
71
if (fail)
72
throw std::exception();
73
printf("construct\n");
74
}
75data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
~S()
{
76
printf("destroy\n");
77
}
78
};
79
bool S::fail;
80data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
81data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
void test_class(int dim)
{
82
if (dim<=0)
83
return;
84
C::fail = dim==4;
85
C* arr = new C[dim];
86
delete[] arr;
87
}
88data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
89data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
90data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
void test_global(int dim)
{
91
if (dim<=0)
92
return;
93
S::fail = dim==4;
94
S* arr = new S[dim];
95
delete[] arr;
96
}
97data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
98data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
int main()
{
99
using namespace std;
100
int dim = 0;
101
for (printf("input dim: ");scanf("%d",&dim)==1;printf("input dim: "))
102data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
103data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
try
{
104
test_class(dim);
105
}
106data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
catch (std::exception& )
{
107
printf(" ---- catch an exception ----\n");
108
}
109data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
try
{
110
test_global(dim);
111
}
112data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
catch (std::exception& )
{
113
printf(" ---- catch an exception ----\n");
114
}
115
}
116
}
117data:image/s3,"s3://crabby-images/13de6/13de6130588e8a001331bf125b484ea2f97d951e" alt=""
有两个底层分配函数allocate和deallocate,它们使用malloc和free。
用这两个函数实现全局的3个op new,op new[], op delete,没有op delete[]
还用这两个函数实现了C的3个op new,op new[], op delete,同样没有op delete[]
用如下参数编译
cl /EHsc /MD /analyze /W3
你看看结果吧。
我用vc8、9测过(vc6不支持动态crt库,vc10我没装)。
反正两处delete[] arr;都没有调用 op delete。
它们调用那个全局的,没有被重写的op delete[]。
如果静态链接,该全局默认的op delete[]会被inline, 再调用该编译单元中定义的op delete。
如果动态链接,op delete[]不会被inline,会调用crt库中提供的op delete。
总之,这两处delete[] arr;都没有调用deallocate。
当然, 你可以说你只静态链接到crt库。
也可以说你的allocate和deallocate底层实现绝对会一直保持与vc提供的crt兼容。
但是,你的代码的用户了解么?
难道你打算在文档中写“使用我的库的代码者,使用的crt库必须满足XXX要求,必须自己测试YYY”,只是为了你自己可以少写一个 op delete[]?
这不是程序库开发者的态度。
还有两个小问题。
C* pc = static_cast<C*>(malloc(sizeof(*pc));
new (pc) C; // 编译错误
C* pc2 = new (std::nothrow) C; // 编译错误
当然, 你还是可以说你绝对不会用这种东西, 你是实用主义嘛。
但是你的库的使用者呢?
“出发点只是想找到一个经过验证的(大的、成功的产品使用过的)简便的工具”
你觉得这可以说明该产品中的每个细节都是无可挑剔的么?
越是大的产品,测试越不容易,更不容易暴露其中的问题,同时也许忽悠客户也更容易。
确实没有任何事物都是完美的,但不能连追求完美的心都舍弃了。
同时,从实用角度出发,让该库变得更完美,所付出的代价非常小,“按规则”办事就可以了,10来行代码的事,何乐而不为?
规则可以见《EffCpp》或者《C++CodingStandard》。
回复 更多评论