C++ 何时使用动态分配(即使用newkeyword)?何时使用指针?

动态分配

在你的问题里。你用了两种方式创建对象。这两种方式基本的不同在于对象的存储时间。

当运行Object myObject;这句代码时。它作为自己主动变量被创建,这意味着当对象出了作用域时也会自己主动销毁。而当你使用new Object()这样的方式时,对象所拥有的内存是动态分配的。这表示直到你调用delete()方法对象才会被销毁。否则一直存在。当须要用动态分配内存来处理时,你应该仅仅使用动态分配的方式,也就是说,当你能够使用动态分配内存的时候就不要使用自己主动变量。

下面是可能会使用到动态分配的两种常见情况:

1.当想让对象在出了作用域后依旧存在——且确实就是之前存储在该内存中的对象。而不是对象的拷贝。假设你能够接受使用对象的拷贝或者移动(大部分情况下你应该这样),那么你更应该使用自己主动存储方式。

2. 当须要大量内存时,这样的情况下极易导致栈溢出。

当然假设这对你来说根本不是问题就更好了(大部分情况下这是不可能的)。这显然超出了C++的管辖范围,可是不幸的是,我们必须处理我们开发的系统中存在的这样的现实问题。

当你确实须要使用动态分配时,你应该将它封装到一个智能指针中或者其它能具有RAII特性的类型(比如标准容器)。智能指针提供动态分配内存的对象的全部权语义。比如std::unique_ptr和std::shared_ptr。

假设你可以合适的使用它,你基本上不须要自己管理内存(參见Rule of Zero这篇文章)。

指针

其实,指针除了用来实现动态分配内存外还有非常多其他的使用方法。可是当中大部分也都存在比它们更好的选择。就像前面说过的那样。除非你必须用到指针。否则不要贸然使用。

须要使用引用的情况:有的时候,你想调用的函数须要訪问你当前的对象本身(而不是它的拷贝)。那么你就须要使用指针作为參数进行传递(暂不论它是怎样分配的)。然而,在大部分情况下。使用引用会比指针更好,这也正是引用被设计的理由。注意一下,这里不须要像上面所说的那样去延长对象的生命周期。前面已经说过了,假设你能接受使用对象的拷贝。那么你就不是必需再使用引用了。

须要使用多态的情况:通常你仅仅能通过对象的指针或者引用来实现多态(也就是依据对象的动态类型来调用函数)。假设这就是你想要的,那么你就须要使用指针或者引用。相同,以指针为优先选择。

当对象可忽略时,通过传递一个空指针来实现对象是可选的属性:假设它是一个參数的话,你应该优先使用默认參数或者函数重载的方法。否则你应该选择一种能够封装这样的行为的类型,比如boost::optional(或者是std::optional)。

当你想减少文件间的编译依存关系从而节省时间:指针的一大特点在于你仅仅须要在前面声明一下指针指向的类型(而假设要使用实际的对象,你还须要定义一下)。

这样你就能减少你的编译单元之间的耦合性从而减少编译时间。

參考Pimpl idiom.

当你想调用C或者类似C风格的函数库的接口时:在这样的情况下。你不得不使用指针进行操作。

你唯一能做的事情就是要保证你的指针在不使用时要被释放。你也能通过智能指针来操作原指针,比如通过它来调用成员函数。

假设被调用库已经为你申请了空间而又希望你通过句柄来释放的话,利用智能指针封装起句柄并利用定制的析构器来释放内存无疑是一种合理的选择。

时间: 2024-10-27 04:44:13

C++ 何时使用动态分配(即使用newkeyword)?何时使用指针?的相关文章

C++ new delete操作符

//new delete操作符 #define _CRT_SECURE_NO_WARNINGS #include<iostream> using namespace std; /* 1.new delete 操作符号 都是 c++的关键字 类似于 C语言中的 malloc free 2.定义对象时,使用了new关键字,会为这个对象在堆上分配内存,不使用new 关键字会直接在栈上分配内存 new 关键字可以分配基础类型内存,数组类型内存,对象类型内存 对于基础类型内存和数组类型内存,new de

C/C++面试题目一

C/C++开发工程师面试题目(一)(附答案分析) 推荐:自己根据在面试中碰到做过的一些题目以及总结的题目,希望对面试的同学有所帮助. 一. 选择题 1. 下列类中(  )不是输入输出流类iostream的派生类. A. fstream      B. ofstream     C. strstream    D. ostrstream 答案:BD 解析:ofstream和ostrstream派生自ostream,而不是iostream.                              

关于虚析构函数的作用和使用

作用:作为基类使用的类应该具有虚析构函数,以保证在删除基类指针(动态分配的对象)时.依据指针实际指向的对象进行适当的析构. 请看以下这段代码; #include <iostream> class A{ public: A(){ std::cout << "A constructor execute" << std::endl; } ~A(){ std::cout << "A destructor execute" &l

ACM:统计难题 解题报告-字典树(Trie树)

统计难题 Time Limit:2000MS     Memory Limit:65535KB     64bit IO Format:%I64d & %I64u Submit Status Description Ignatius最近遇到一个难题,老师交给他很多单词(只有小写字母组成,不会有重复的单词出现),现在老师要他统计出以某个字符串为前缀的单词数量(单词本身也是自己的前缀). Input 输入数据的第一部分是一张单词表,每行一个单词,单词的长度不超过10,它们代表的是老师交给Ignati

【数据结构】队列的基本概念

1. 队列的定义 (1) 定义 队列(queue)是只允许在一端进行插入操作,而在另一端进行删除操作的线性表. (2) 抽象数据类型 ADT 队列(Queue)Data 同线性表.元素具有相同的类型,相邻元素具有前驱和后继关系.Operation InitQueue(*Q):初始化操作,建立一个空队列Q: DestoryQueue(*Q):若队列Q存在,则销毁它: ClearQueue(*Q):将队列Q清空: QueueEmpty(Q):若队列Q为空,返回true,否则返回false: GetH

boost之智能指针

最近在看一本书<boost程序库完全开发指南>,boost是一个强大的C++程序库,丰富了C++的功能和表现力,使得C++软件开发更加简洁.优雅.灵活和高效.因为之前使用Qt做了很多项目,我发现boost和Qt有很多相似之处,比如时间.智能指针.字符串处理.函数信号槽等,在Qt中最好还是用Qt中封装好的东西来做.下面归纳最近所学习的boost只能指针知识. 一.为何要使用智能指针 C++程序员通常使用RAII机制(资源获取即初始化)来管理内存资源,在构造函数中申请资源,析构函数中释放资源.局部

c语言基础知识要点

C语言程序的构成 与C++.Java相比,C语言其实很简单,但却非常重要.因为它是C++.Java的基础.不把C语言基础打扎实,很难成为程序员高手. 一.C语言的结构 先通过一个简单的例子,把C语言的基础打牢. /* clang01_1.c */ #include <stdio.h> int main(void) { printf("这是劝学网的C语言教程.\n"); return 0; } C语言的结构要掌握以下几点: C语言的注释是/* ··· */,而不是//···,

指针辨析:悬垂指针、哑指针、野指针、智能指针

悬垂指针: 1:提出的原因: 请看下面的代码片段: [cpp] view plaincopyprint? int *p=NULL; void main() { int i=10;p=&i; cout<<"第一次:*p = "<<*p<<endl; cout<<"第二次:*p = "<<*p<<endl; } [cpp] view plaincopyprint? int *p=NULL;

C++ Primer 学习笔记_68_面向对象编程 --构造函数和复制控制[续]

面向对象编程 --构造函数和复制控制[续] 三.复制控制和继承 合成操作对对象的基类部分连同派生类部分的成员一起进行复制.赋值或撤销,使用基类的复制构造函数.赋值操作符或析构函数对基类部分进行复制.赋值或撤销. 类是否需要定义复制控制成员完全取决于类自身的直接成员.基类可以定义自己的复制控制而派生类使用合成版本,反之,基类使用合成版本,而派生类使用自己定义的复制控制也可以. 只包含类类型或内置类型的数据成员.不包含指针的类一般可以使用合成操作,复制.赋值或撤销这样的成员不需要使用特殊控制.但是:

More Effective C++

条款一:指针与引用的区别 指针与引用看上去完全不同(指针用操作符'*'和'->',引用使用操作符'.'),但是它们似乎有相同的功能.指针与引用都是让你间接引用其他对象.你如何决定在什么时候使用指针,在什么时候使用引用呢? 首先,要认识到在任何情况下都不能用指向空值的引用.一个引用必须总是指向某些对象.因此如果你使用一个变量并让它指向一个对象,但是该变量在某些时候也可能不指向任何对象,这时你应该把变量声明为指针,因为这样你可以赋空值给该变量.相反,如果变量肯定指向一个对象,例如你的设计不允许变量为