C++Primer笔记-----day08

news/2024/7/5 6:11:20

==========================================================================
day08
==========================================================================
1. map又称为关联数组。

使用map来写单词计数程序

E1: map<string, size_t> word_count;
string word;
while (cin >> word) {
++word_count[word]; // map中的下标操作符[],如果word不在map中,则下标操作符会添加一个新元素.所以[]只可以用在非const的map
// 有时候我们只想知道一个元素是否在map中,但在不存在时并不想添加元素,就不能使用下标操作符
}
for (const auto &w : word_count)
cout << w.first << " occurs " << w.second << ((w.second > 1) ? "times" : "time")<<endl;

 

E2: map<string, size_t> word_count;
string word;
while (cin >> word) {
auto ret = word_count.insert({word,1}); // 插入一个元素,关键字为word,值为1.如果map中已存在,则什么都不做
if (!ret.second) // word 已在map中
++ret.first->second; // 递增计数器
}
for (const auto &w : word_count)
cout << w.first << " occurs " << w.second << ((w.second > 1) ? "times" : "time") << endl;

2.动态内存管理

#include<memory>

为了更容易更安全的使用动态内存,标准库提供了两种智能指针类型来管理动态对象。
shared_ptr 允许多个指针指向同一个对象 用法: shared_ptr<string> // 表示指针可以指向的类型为string
unique_ptr 独占所指的对象

标准库还定义了weak_ptr,是一种弱引用,指向shared_ptr所管理的对象

 

 

3.内存耗尽。 尽管一般内存足够,但还是可能出现内存耗尽的情况。当内存耗尽,new就会失败,默认情况下,就会抛出一个bad_alloc的异常。
我们可以在出现这种情况时阻止抛出异常,并返回一个空指针: int *p = new (nothrow) int; //如果分配失败,返回一个空指针
这种形式的new称为定位new(placement new) bad_alloc和nothrow都定义在#include<new>中

4.我们传递给delete的指针必须是指向动态分配的内存,或者是一个空指针(nullptr)
eg: int i,*pi1 = &i,*pi2 = nullptr;
double *pd = new double(33),*pd2 = pd;
delete i; //错误,i不是一个指针
delete pi1; //未定义的行为,pi1指向的是一个局部变量 编译器并不会报错
delete pi2; //正确,释放一个空指针总是没错的
delete pd; //正确
delete pd2; //未定义的行为,pd2指向的内存以及被释放过了 编译器并不会报错

delete之后重置指针值:
当我们delete一个指针后,指针值就变得无效了。虽然指针无效了,但在许多机器上指针仍然保存着(已经释放了的)动态内存的地址。在delete之后
指针就变成了悬空指针,即指向一块曾经保存数据但现在已经无效的内存的指针。
避免悬空指针的方法是,delete后,将指针置为nullptr。
但这也仅仅是有限的保护:当多个指针指向相同的内存,delete内存后重置指针为空的方法只对这个指针有效,对其他仍指向(已释放的)内存的指针是没有
作用的。
eg: int *p = new int(42);
autp q = p; // q和p指向相同的内存
delete p; // p和q均变为无效
p = nullptr; // 指出p不再绑定任何对象 但重置p对q没有任何作用,q还是个悬空指针。

 

5. 注意,内置类型的动态内存分配,需要显式地初始化 : int *p = new int(5); // 初始化为5
否则,其指向的值是未定义的。 而一些内置类或自定义有默认构造函数的类型,如果不显式初始化就会进行默认初始化。


6. make_shared函数类似于容器的emplace()函数(考虑insert与emplace的区别),make_shared用其参数来构造给定类型的对象。
例如,make_shared<string>时传递的参数必须与string的某个构造函数相匹配。
make_shered<string>(10,'a'); 如果什么参数都不传递,则对象进行值初始化。


7.智能指针与new的混用
如果我们不初始化一个智能指针,他就会被初始化为一个空指针。 shared_ptr<int> sp; // 空指针
除了使用make_shared函数初始化智能指针,我们还可以用new返回的指针来初始化智能指针。
shared_ptr<int> p(new int(42)); // 正确,可以使用直接初始化的形式
shared_ptr<int> p1 = new int(1024); // 错误,智能指针的构造函数是explicit的,不能将内置指针隐式转换为智能指针,必须使用直接初始化的形式

【不要将智能指针与普通指针混用】
考虑如下:
void process(shared_ptr<int> ptr)
{
//使用ptr
} // ptr离开作用域,被销毁

上述函数的参数是以传值方式传递的,因此实参会被拷贝到ptr中。拷贝一个shared_ptr会递增其引用计数。因此process运行过程中,引用计数至少为2

shared_ptr<int> p(new int(42)); // 引用计数为1
process(p); // 拷贝p递增它的引用计数; 变为2
int i = *p; // 出了process函数,局部变量ptr被销毁,引用计数变为1,所以仍然可以使用p指向的这块内存

再考虑如下:
int *x(new int(1024));
process(shared_ptr<int>(x)); // 引用计数为1
int j = *x; // 未定义的行为,x变为了空悬指针。
上述调用,我们把一个临时的shared_ptr传给了process,当调用结束,这个临时对象就被销毁了,引用计数就会递减,变为0,因此,临时对象被销毁,
它所指的内存被释放,但x继续指向已经被释放的内存,从而成了一个空悬指针。

所以,使用一个内置指针访问一个智能指针所负责的对象是危险的, 因为我们无法知道对象何时被销毁。

8.之前一直以为野指针和空悬指针一个意思,但错了。
空悬指针(dangling pointer):指向已经销毁的对象或已经回收的地址。
野指针:没有初始化的指针就是野指针。 如:int *p;


9.永远不要用get函数初始化另一个智能指针或为智能指针赋值。
get函数,p.get() ,返回p中保存的指针,返回的是一个内置指针。
get函数是为这样一种情况设计的:我们需要向不能使用智能指针的代码传递一个内置指针。

要非常注意的一点是:get返回的指针,不能用delete

shared_ptr<int> p(new int(42));p指向
int *q = p.get();
....
delete q; // 未定义的行为,因为智能指针p会自动释放内存,而再调用delete q 就会造成了二次释放内存。

10. 因为shared_ptr可能有多个指向了同一块内存,在改变指向的底层对象之前,我们先检查自己是不是当前对象仅有的用户,
如果不是,在改变之前要制作一份新的拷贝。 用到reset函数,其中一种的参数是一个内置指针。(reset函数有三种)
用法:p.reset(q); 令p指向q

if(!p.unique())
p.reset(new string(*p)); // 我们不是唯一用户,分配新的拷贝。
*p += newVal; //现在我们是唯一用户了,可以改变对象的值。

 

转载于:https://www.cnblogs.com/ll-10/p/9893257.html


http://www.niftyadmin.cn/n/4556924.html

相关文章

kafka整合springboot2.0(六)

1、生产者 消费者pom <!--spring boot 版本依赖--> <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.1.4.RELEASE</version><relativePath/> <…

lower_case_table_names默认值问题

在不同操作系统中它的值是不一样的 Unix&#xff1a;0 Windows&#xff1a;1 macOS&#xff1a;2

我是菜鸟 学好C++C#有什么秘诀吗

要熟记各种库函数和语法结构&#xff1b;多用c的类 另外还有多看别人的源码 都需要百分之百的兴趣坚持 日后用的多了自然就理解含义了 ||| 学习任何语言都没有捷径 有些看不懂的程序就背下来 多练习 多练 另外多交流讨论也是必不可少的 引用等练习程序编写尽量一题多解&#xf…

RocketMQ SpringBoot 采坑

1、项目结构&#xff1a; 2、Provider 2.1、Provider pom <!--spring boot 版本依赖--> <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.1.4.RELEASE</v…

MemSQL Start[c]UP 2.0 - Round 1 F - Permutation 思维+线段树维护hash值

F - Permutation 思路&#xff1a;对于当前的值x&#xff0c; 只需要知道x k, x - k这两个值是否出现在其左右两侧&#xff0c;又因为每个值只有一个&#xff0c; 所以可以转换成&#xff0c;xk, x-k在到x所在位置的时候是否都出现&#xff0c;或者都不出现&#xff0c;即出现…

什么是索引?

最常见的索引类型涉及单个列&#xff0c;将来自该列的值的副本存储在数据结构中&#xff0c;从而允许快速查找具有相应列值的行。B树数据结构让索引可以快速找到一个特定的值&#xff0c;一组值&#xff0c;或者一个范围内的值&#xff0c;对应于子句 中的运算符&#xff0c;如…

学c语言有什么用

自己查看看 ||| 基础性的语法 它适合作为系统描述语言 学学还是有必要的 为以后打基础 由于汇编语言依赖于计算机硬件 即可用来编写系统软件 C语言是国际上广泛流行的、很有发展前途的计算机高级语言 早期的操作系统等系统软件主要是用汇编语言编写的&#xff08;包括 UNIX操作…

Spire.Doc修改目录字体大小

官方论坛 官方论坛的解决方案&#xff1a; static void Main(string[] args){Document doc new Document();doc.LoadFromFile("目录.docx");foreach (Section section in doc.Sections){//遍历body下面所有对象foreach (DocumentObject obj in section.Body.ChildO…