要实现一个类的对象(实例)计数,即程序运行中此类有多少个对象存在,最容易的实现方法是使用静态数据成员。如下:
class Widget {
public:
Widget() { ++count; }
Widget(const Widget&) { ++count; }
~Widget() { --count; }
static size_t howMany() { return count; }
private:
static size_t count;
};
//cpp文件中
size_t Widget::count = 0;
但是如果程序中有多个需要实例计数的类,则在每个类中加入上面代码未免繁琐、易错。这种情况下,最好是实现一个通用计数类。它应该具备一下特点:
下面我们将逐步实现并完善这个通用的计数类。
class Counter {
public:
Counter() { ++count; }
Counter(const Counter&) { ++count; }
~Counter() { --count; }
static size_t howMany() { return count; }
private:
static size_t count;
};
// This still goes in an implementation file
size_t Counter::count = 0;
//方法一: embed a Counter to count objects
class Widget {
public:
..... // all the usual public
// Widget stuff
static size_t howMany() { return Counter::howMany(); }
private:
..... // all the usual private
// Widget stuff
Counter c;
};
//or:
//方法二: inherit from Counter to count objects
class Widget: public Counter {
public:
..... // all the usual public
// Widget stuff
private:
..... // all the usual private
// Widget stuff
};
template
class Counter {
public:
Counter() { ++count; }
Counter(const Counter&) { ++count; }
~Counter() { --count; }
static size_t howMany() { return count; }
private:
static size_t count;
};
// this now can go in header
template
这时Widget类利用它来进行实例计数:
//方法一: embed a Counter to count objects
class Widget {
public:
.....
static size_t howMany() {return Counter
private:
.....
Counter
};
//or:
//方法二: inherit from Counter to count objects
class Widget: public Counter
.....
};
上面方法一和方法二都可正确实现计数,我们继续探讨这两种方案的优缺点。
首先讲public继承,即class Widget: public Counter
Counter
// get base class ptr to derived class object
......
delete pw; // yields undefined results if the base class lacks a virtual destructor
template
class Counter {
public:
.....
protected:
~Counter() { --count; }
.....
};
除了上面两种方法,还可以使用private继承,即class Widget: private Counter
class Widget: private Counter
public:
// make howMany public
using Counter
..... // rest of Widget is unchanged
};
Counter
class SpecialWidget: public Widget,
public Counter
public:
};
总结
用模板实现的这个对象计数类可以满足绝大多数需求,但不适用于计数有继承关系的类。本文的核心思想来源于CUG上C++大师Scott Meyers的一篇文章并有所改动。