C++基础知识(8)

50、vector底层实现原理

vector就是一个动态的数组,当数组的元素超出capacity,其内部机制会申请一块更大的内存,把原本的元素移动到新申请的内存,然后释放之前的内存,新的capacity一般为原来的1.5-2倍。当vector的内存重新分配时,原来的迭代器将会失效。

vector内部维护三个指针,_first ptr指向第一个元素,_last ptr指向最后一个元素的下一个位置,_end ptr指向vector指向其申请内存的下一个位置。因此vector的size就是等于_last - _first,而capacity就等于_end - _first。

51、vector的元素类型可以是引用吗?
vector的底层实现要求连续的对象排列,引用并非对象,没有实际地址,因此vector的元素类型不能是引用。

52、vector的函数clear(),shrink_to_fit(),swap()之间的区别

vec.clear(),清除容器里的内容,但是不释放内存,size会变为0;

vector().swap(vec):清空内容并且释放内存,得到一个全新的vector

vec.shrink_to_fit():请求容器降低其capacity与size匹配

53、vector为什么是以1.5倍或者2倍扩容?

(1)随着数据增长越来越多,倍数方式扩容会比等长个数方式更加高效,因为能够减少拷贝次数。

(2)选择1.5倍和2倍扩容是为了能够复用之前申请的空间,如果大于2倍,那么每一次申请的空间都将会大于之前申请的总和,所以扩容倍数应该在(1,2)之间。

54、访问vector里面的元素有几种方式?

(1)利用[]访问,如果下标越界, 会引起未定义行为

(2)利用迭代器访问,vector 的迭代器属于随机访问迭代器, 允许进行 + - 整数操作,解引用越界的迭代器会引起未定义行为

(3)利用at访问,下标越界会抛出异常

std::vector<int> vec{1,2,3};
// 利用[]访问
std::cout<<"第2个元素为:"<<vec[1]<<std::endl;
// 利用迭代器访问
auto it = vec.begin();
std::cout<<"第2个元素为:"<<*(it+1)<<std::endl;
// 利用at访问
std::cout<<"第2个元素为:"<<vec.at(1)<<std::endl;

55、vector插入元素的方式

(1)尾插:使用 push_back或者emplace_back函数插入

(2)中间插入:使用 insert 函数进行插入. 需要配合迭代器指定插入的位置

std::vector<int> vec;
// 尾插
vec.push_back(1);
vec.emplace_back(3);
// 中间插入
auto it = vec.begin()+1;
vec.insert(it,2);
// 打印元素
for(auto& num:vec)
    std::cout<<num<<" ";
std::cout<<std::endl;

// 输出结果为:1,2,3

56、vector的emplace_back() 和 push_back() 的区别

两者在于底层实现的机制不同。push_back() 向容器尾部添加元素时,首先会创建这个元素,然后再将这个元素拷贝或者移动到容器中(如果是拷贝的话,事后会自行销毁先前创建的这个元素);而 emplace_back() 在实现时,则是直接在容器尾部创建这个元素,省去了拷贝或移动元素的过程。

#include <vector> 
#include <iostream> 
using namespace std;
class testDemo
{
public:
    testDemo(int num):num(num){
        std::cout << "调用构造函数" << endl;
    }
    testDemo(const testDemo& other) :num(other.num) {
        std::cout << "调用拷贝构造函数" << endl;
    }
    testDemo(testDemo&& other) :num(other.num) {
        std::cout << "调用移动构造函数" << endl;
    }
private:
    int num;
};

int main()
{
    cout << "emplace_back:" << endl;
    std::vector<testDemo> demo1;
    demo1.emplace_back(2);  

    cout << "push_back:" << endl;
    std::vector<testDemo> demo2;
    demo2.push_back(2);
}

/*
    运行结果
    emplace_back:
    调用构造函数
    push_back:
    调用构造函数
    调用移动构造函数
*/

/*
	如果把移动构造函数注释掉,再运行程序结果如下:
	emplace_back:
    调用构造函数
    push_back:
    调用构造函数
    调用拷贝构造函数
*/

由此可以看出,push_back() 在底层实现时,会优先选择调用移动构造函数,如果没有才会调用拷贝构造函数。总的来说emplace_back() 效率会更高,但是emplace_back() 是C++11新增的,如果要兼容旧版本还是使用push_back()比较好。

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

czy1219

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值