在STL中处理对象指针
作者:cker
提交者:eastvc 发布日期:2003-7-12 21:13:52
原文出处:http://how.digichina.net/program/c/00000242.htm
众所周知,STL的容器类是针对对象来进行存储管理的。虽然没有明确的限制不能在STL的容器类中包含指针,STL也不在乎您存放在容器中的对象还是指针。但仔细考虑STL的初衷,在容器类中使用指针是明显不合时宜的。
CKER的意思是您应该尽可能的直接将对象本身而不是将对象的指针放置在STL容器类中。直接在容器中存放指针一个最大的副作用就是可能导致内存泄漏。这个问题在char *类型上有突出的体现。
但某些时候,直接使用对象指针有明显的好处,这些情况大致可以概括为如下的几个方面:
1. 当对象十分庞大时,频繁的拷贝复制所带来的系统资源消耗十分昂贵。
2. 当您需要将同一个对象存放到多个容器中,这种情况应该不是很少见的。
3. 当您需要在同一个容器中存放由同一个父类派生而来的多个派生类对象。这也很常见。
事实上,CKER这个星期所开发的计算程序中,正需要面对第三种情况。考虑使用STL的种种好处。
CKER决定引入STL的list容器。
本来,使用BCB的TList对象,也可以实现同样的目的。
但
TLsit类在对象指针的个数超过5000个时,效率会有明显的下降。
TList类不是类型安全的,它根本不在乎引入的对象指针是何种类型的。
引入TList类就意味着要包含VCL.h头文件,这对我的计算模块的可移植性实在不是一件好事。
cker做了决定之后,面临两个与stl有关的问题。
第一个问题就是上面提到的如何在STL中处理对象的指针。
CKER的解决之道是创建一个封装指针的类。
代码如下:
// 定义一个用于STL容器的指针封装类
// 因为使用STL不宜直接向容器中添加指针。
////////////////////////////////////////////////////////////////////////////////
class PtrWrapper
{
private:
X* px;//指向类X的指针
public:
file://构造及拷贝构造函数
PtrWrapper(X* x = 0) : px(x) {}
PtrWrapper(const PtrWrapper& pw) : px(pw.px) {}
file://析构函数
~PtrWrapper() {}
PtrWrapper& operator= (const PtrWrapper& pw) { px = xw.px; }
file://重载运算符()返回对象X的指针
const X* operator() () const { return px; }
X* operator() () { return px; }
};
file://重载逻辑运算符 == , < , >
bool operator== (const PtrWrapper& pw1, const PtrWrapper& pw2) {
return (pw1.operator()() && pw2.operator()()) ? *pw1() == *pw2() : false;
}
bool operator< (const PtrWrapper& pw1, const PtrWrapper& pw2) {
return (pw1() && pw2()) ? *pw1() < *pw2() : false;
}
bool operator> (const PtrWrapper& pw1, const PtrWrapper& pw2) {
return (pw1() && pw2()) ? !(*pw1() < *pw2()) : false;
}
上述代码封装了一个指针。经过类PtrWrapper的封装后,您就没有必要直接使用指针了。
STL容器接触的是真正的对象,只不过这个对象封装了一个特定类型的指针。
下面举例说明PtrWrapper的使用。
// 假设您需要将指向类X的对象的指针放入STL容器。
//
class X
{
private:
int I;
public:
file://构造函数,拷贝构造函数,析构函数
X(int i) : I(i) {}
X(const X& x) : I(x.I) {}
~X() {}
file://重载运算符= ,()
X& operator= (const X& x) { I = x.I; }
int operator()() const { return I; }
};
file://重载逻辑运算符
bool operator== (const X& x1, const X& x2) {
return x1() == x2();
}
bool operator< (const X& x1, const X& x2) {
return x1() < x2();
}
<wbr></wbr>
file://下面是示例主程序
int main(int, char*[]) {
PtrWrapper bucket[5];
for(int i = 0; i < 5; ++i) {
bucket[i] = PtrWrapper(new X(i * i));
}
random_shuffle(bucket, bucket + 5);
list<PtrWrapper> list1;
copy(bucket, bucket + 5,
back_insert_iterator<list<PtrWrapper> > (list1)
);
cout << "List of PtrWrapper: (";
for_each(list1.begin(), list1.end(), print);
cout << ")" << endl;
set<PtrWrapper, greater<PtrWrapper> > set1;
copy(list1.begin(), list1.end(),
insert_iterator<set<PtrWrapper, greater<PtrWrapper> > >
(set1, set1.begin())
);
cout << "Set of PtrWrapper : [";
for_each(set1.begin(), set1.end(), print);
cout << "]" << endl;
deque<PtrWrapper> deque1;
copy(list1.begin(), list1.end(),
back_insert_iterator<deque<PtrWrapper> > (deque1)
);
cout << "Deque of PtrWrapper : (";
for_each(deque1.begin(), deque1.end(), print);
cout << ")" << endl;
return 0;
}
还有一个问题是如何在STL容器种存放派生类对象,下次讨论。
注:文中的代码原型来自于Internet,可惜网址没留下来。CKER整理之后贴出来,姑且妄称原创了。
您可以随意复制、分发、下载此文档。但未经本人同意,您不可以截取、改动本文片断,或用本文谋取任何形式的利益。
分享到:
相关推荐
1.9.1 stl_config.h 中的各种组态 027 组态3:static template member 027 组态5:class template partial specialization 028 组态6:function template partial order 028 组态7:explicit function template ...
这本书也不适合带领你学习面向对象(Object Oriented)技术 — 是的,STL 与面向对象没有太多关连。本书前言清楚说明了书籍的定位和合适的读者,以及各类基础读物。如果你的Generic Programming/STL实力足以阅读本书...
1-3-2 在派生类中实现类的基本函数,................... _ ............... 29 1-3-3 内联函数技术,........ ................................... 30 3133 ..... .. .. .. .. .. .. .. .. .. .. .. .. .....
访问对象是通过象指针一样的迭代器实现的;容器是象链表,矢量之类的数据结构,并按模板方式提供; 算法是函数模板,用于操作容器中的数据。由于STL以模板为基础,所以能用于任何数据类型和结构。
函数指针是C++编程语言中非常重要的概念,可以让我们更加高效地编写程序。因此,要学好C++编程语言,必须熟练掌握函数指针的使用方法。 6. 掌握异常处理。异常处理是C++编程语言中非常重要的概念,可以让我们更加...
包括复制构造函数,移动构造函数以及复制和移动分配如何重载运算符使用继承和类层次结构使用多态函数和动态绑定使用智能指针使用流I / O C ++ STL简介异常处理简介C ++开发的良好做法提交者乔治·卡林george.calin ...
而每一本C++的经典著作所提供的方案是使用智能指针(STL的标准类auto_ptr)。 在Object Pascal中,这个问题变得非常的简单,程序员不必为此大费周折。如果Object Pascal的类在构造函数中抛出异常,编译器会自动调用...
7.11 有关对象的思考:在电梯模拟程序中使用复合和动态对象管理 小结 术语 自测练习 自测练习答案 练习 第8章 运算符重载 8.1 简介 8.2 运算符重载的基础 8.3 运算符重载的限制 8.4 用作类成员与友元函数的...
7.11 有关对象的思考:在电梯模拟程序中使用复合和动态对象管理 小结 术语 自测练习 自测练习答案 练习 第8章 运算符重载 8.1 简介 8.2 运算符重载的基础 8.3 运算符重载的限制 8.4 用作类成员与友元函数的...
13. l 从标准C++转向STL语法的过程中出现的问题 13.1.l 用迭代器遍历容器 13.1.2 仔细研究迭代器 13.1.3 流迭代器 13.1.4 为什么使用end() 13.1.5 复制列表 13.1.6 列表中的列表 13.1.7 STL字符...
12_数组中括号与指针关系和数组名常量指针分析 13_字符串一级指针内存模型_传智扫地僧 14_字符串copy函数技术推演 15_字符串copy函数强化训练_判断null_引入辅助指针变量_传智扫地僧 16_项目开发模型强化_strstr_...
在这个视频中,我们将带您走进C++的世界,从基础语法开始,逐步深入到面向对象编程的核心概念。无论您是初学者,还是有一定编程经验的开发者,这个视频都将为您提供宝贵的参考和指导。 视频大纲: C++简介 C++的...
在个人实践中,Ivor Horton也是一名系统顾问。他从事程序设计教学工作已经超过了25年。 苏正泉,1995年毕业于解放军信息工程学院计算机及应用专业,高级工程师。在IT项目管理、软件开发、系统管理和网络管理方面都...
3.5.4 字符数据在内存中的存储形式及使用方法 41 3.5.5 字符串常量 41 3.5.6 符号常量 42 3.6 变量赋初值 42 3.7 各类数值型数据之间的混合运算 43 3.8 算术运算符和算术表达式 44 3.8.1 C运算符简介 44 3.8.2 算术...
3.5.4 字符数据在内存中的存储形式及使用方法 41 3.5.5 字符串常量 41 3.5.6 符号常量 42 3.6 变量赋初值 42 3.7 各类数值型数据之间的混合运算 43 3.8 算术运算符和算术表达式 44 3.8.1 C运算符简介 44 3.8.2 算术...
高级特性篇:深入探讨模板编程、标准模板库(STL)的使用,以及异常处理、文件操作等高级主题。 实战项目篇: 游戏开发项目:指导读者开发一个简单的控制台游戏,实践面向对象的设计思想。 数据结构库项目:从零开始...
24.STL中map和set的原理(关联式容器) 25.MFC的消息机制 26.消息映射 27.列举几种进程的同步机制,并比较其优缺点 28.数组和链表的区别 29.MFC主要要用到哪几个类?及其各个类的作用 30.MFC六大核心机制 31....
网络管理器使用此 ID 作为 STL 映射的关键字,以获取指向请求对象的回复处理槽的指针。 然后可以直接调用回复处理程序。 当请求对象向网络管理器注册它自己的指针时,ID 只不过是提供给请求对象的随机数。 网络管理...