【C++初阶8-vector】熟悉的ta

news2025/1/12 22:49:13

前言

本期看看这位熟悉又陌生的朋友——vector。

博主水平有限,不足之处望请斧正!


是什么

vecotr是序列容器,可变大小的数组。

*vector有矢量、向量的意思,用其命名可能想强调“序列”这个概念。

class template

std::vector

template < class T, class Alloc = allocator<T> > class vector; // generic template

*Alloc是空间配置器(内存池),后面讲。


怎么用

有了数据结构顺序表的基础加上string的接口使用,vector的接口基本一看就懂了。CPP文档

遍历
void t1()
{
    vector<int> v;
    
    v.push_back(1);
    v.push_back(2);
    v.push_back(3);
    v.push_back(4);
    
    for(size_t i = 0; i < v.size(); ++i)
    {
        cout << v[i] << ' ';
    }
    cout << endl;
    
    vector<int>::iterator it = v.begin();
    while(it != v.end())
    {
        cout << *it << ' ';
        ++it;
    }
    cout << endl;
    
    for(auto e : v)
    {
        cout << e << ' ';
    }
    cout << endl;
}
1 2 3 4 
1 2 3 4 
1 2 3 4

按这个说法……vector<char>string不是一样吗?

还是没法划等号的:

  1. string的结尾是\0
  2. 意义不同,接口也有差异。
迭代器构造
void t2()
{
    vector<int> v1;
    v1.resize(10, 1);
    
    for(auto e : v1) cout << e << ' ';
    cout << endl;
    
    vector<int> v2(v1.begin() + 1, v1.end() - 1);
    for(auto e : v2) cout << e << ' ';
    cout << endl;
}
1 1 1 1 1 1 1 1 1 1 
1 1 1 1 1 1 1 1

需要注意,迭代器都是左闭右开的。

reserve
void t3()
{
    vector<int> v;
    
    v.push_back(1);
    v.push_back(2);
    v.push_back(3);
    v.push_back(4);
    cout << v.capacity() << endl;
    
    v.reserve(10);
    cout << v.capacity() << endl;
    
    v.reserve(4);
    cout << v.capacity() << endl;
    
}

resrve仍然是不缩容的。缩容是有代价的,没法原地缩,只能异地开辟、拷贝、释放(涉及到内存管理)。这可以理解成是一种空间换时间。

resize

缩容 / 扩容并初始化(需要可以填数据)

void t4()
{
    vector<int> v;
    v.resize(20, 3);
    
    for(auto e : v) cout << e << ' ';
    cout << endl;
}
3 3 3 3 3 3 3 3 3 3 3 3 3 3 3 3 3 3 3 3
resizereserve

一疏忽就容易用混

void t5()
{
    vector<int> v;
    v.reserve(10); //size没变
    cout << "size:" << v.size() << endl;
    cout << "capacity:" << v.capacity() << endl;
    
    for(size_t i = 0; i < 10; ++i) v[i] = i;
    
    for(size_t i = 0; i < 10; ++i) cout << v[i] << ' ';
    
    cout << endl;
    
}

在这里插入图片描述

断言原因:[]内根据size判断是否越界。

正常情况下resize好用些。

assert

assert也是有缺点的,在release版本下会失效。不过都到release这一步了,程序基本没什么触发断言的地方了。

#内置类型的构造函数

看看resize

void resize (size_type n, value_type val = value_type());

size_type = unsigned int

value_type = The first template parameter

val的缺省值给了一个匿名对象,当模版参数绑定的是int,不就等于内置类型也有构造函数了?

是的,

void t6()
{
    int a = int();
    int b = int(10);

    cout << a << endl;
    cout << b << endl;
}
0 //默认初始化为0
10

而且这是有意义的,如果缺省值给了个0,自定义类型就照顾不到了,你0不一定能赋给自定义类型,所以

内置类型也有构造函数。

#vector的扩容机制
void t7()
{
    vector<int> v;
    size_t size;
    size = v.capacity();
    cout << "vector growing...\n" << endl;
    for (int i = 0; i < 100; ++i)
    {
        v.push_back(i);
        if (size != v.capacity())
        {
            cout << "capacity changed to " << v.capacity() << endl;
            size = v.capacity();
        }
    }
}

vs2019下:1.5x

vector growing...
capacity changed to 1
capacity changed to 2
capacity changed to 3
capacity changed to 4
capacity changed to 6
capacity changed to 9
capacity changed to 13
capacity changed to 19
capacity changed to 28
capacity changed to 42
capacity changed to 63
capacity changed to 94
capacity changed to 141

g++:2x(2倍左右比较合适,往下太少导致频繁扩容;往上太多用不完浪费)

vector growing...
capacity changed to 1
capacity changed to 2
capacity changed to 4
capacity changed to 8
capacity changed to 16
capacity changed to 32
capacity changed to 64
capacity changed to 128

提前开空间,可以避免扩容

void t7()
{
    vector<int> v;
    v.reserve(100);
    size_t size;
    size = v.capacity();
    cout << "vector growing..." << endl;
    for (int i = 0; i < 100; ++i)
    {
        v.push_back(i);
        if (size != v.capacity())
        {
            cout << "capacity changed to " << v.capacity() << endl;
            size = v.capacity();
        }
    }
}
vector growing...
find
void t8()
{
    vector<int> v;
    v.push_back(1);
    v.push_back(2);
    v.push_back(3);
    v.push_back(4);
    
    vector<int>::iterator it = find(v.begin(), v.end(), 3);
    if(it != v.end())
        v.insert(it, 5);
    
    for(auto e : v) cout << e << ' ';
    cout << endl;
}
1 2 5 3 4

需要注意,vector没有find。因为有迭代器,find只需要用迭代器遍历找就行,所以实现一个放进<algorithm>最方便,每个容器都能用。

swap
//1.成员函数
void swap (vector& x);

//2.非成员函数
template <class T, class Alloc>
  void swap (vector<T,Alloc>& x, vector<T,Alloc>& y);

大佬真是用心良苦,怕我们用错成 swap(v1, v2),还专门写了个非成员函数的swap给咱用。就算写错,也不会走算法库的深拷贝,而是走这个更匹配的。

使用方面,能讲的大概也就这些了,和string非常像。


OJ练手

1. 只出现一次的数字

给你一个 非空 整数数组 nums ,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。

你必须设计并实现线性时间复杂度的算法来解决此问题,且该算法只使用常量额外空间。

示例 1 :

输入:nums = [2,2,1]
输出:1

示例 2 :

输入:nums = [4,1,2,1,2]
输出:4

示例 3 :

输入:nums = [1]
输出:1

提示:

  • 1 <= nums.length <= 3 * 104
  • -3 * 104 <= nums[i] <= 3 * 104
  • 除了某个元素只出现一次以外,其余每个元素均出现两次。
思路和算法:

vector内的数异或起来,每个位上,相同的会抵消,不同的会留下,最终ret的32位由“不同的”组成

class Solution {
public:
    int singleNumber(vector<int>& nums) 
    {
        int ret = 0;
        for(auto e : nums) ret ^= e; //相同抵消
        return ret;
    }
};

2. 杨辉三角

给定一个非负整数 *numRows,*生成「杨辉三角」的前 numRows 行。

在「杨辉三角」中,每个数是它左上方和右上方的数的和。

img

示例 1:

输入: numRows = 5
输出: [[1],[1,1],[1,2,1],[1,3,3,1],[1,4,6,4,1]]

示例 2:

输入: numRows = 1
输出: [[1]]

提示:

  • 1 <= numRows <= 30
思路和算法:

杨辉三角的第n行有n个元素,除去第一个和最后一个,每一个都是上一行的元素 + 上一行左边的元素

1
1 1
1 2 1
1 3 3 1
杨辉三角:vv[i][j] = vv[i-1][j-1] + vv[i-1][j]
class Solution {
public:
    vector<vector<int>> generate(int numRows) 
    {
        //构造二维数组
        vector<vector<int>> vv;
        vv.resize(numRows); //vv含numRows个vector<int>

        for(int i = 0; i < numRows; ++i) 
        {
            vv[i].resize(i+1); //每一行含i+1个int
            // vv[i][0] = vv[i][vv[i].size() - 1] = 1;
            vv[i].front() = vv[i].back() = 1;
        }

        for(size_t i = 0; i < vv.size(); ++i)
        {
            for(size_t j = 0; j < vv[i].size(); ++j)
            {
                if(vv[i][j] == 0) vv[i][j] = vv[i-1][j-1] + vv[i-1][j];
            }
        }

        return vv;
    }
};

今天的分享就到这里了

这里是培根的blog,期待与你共同进步!

下期见~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/129408.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

独占指针 std::unique_ptr

学习智能指针之前需要知道的&#xff1a; 智能指针是原始指针的封装&#xff0c;在头文件<memory>中&#xff0c;优点就是自动分配内存&#xff0c;不用担心潜在的内存泄漏。不是所有的指针都可以封装成智能指针&#xff0c;很多时候原始指针更方便。各指针中&#xff0…

Webpack中常见的Loader?解决了什么问题?

一、是什么 loader 用于对模块的源代码进行转换&#xff0c;在 import 或"加载"模块时预处理文件 webpack做的事情&#xff0c;仅仅是分析出各种模块的依赖关系&#xff0c;然后形成资源列表&#xff0c;最终打包生成到指定的文件中。如下图所示&#xff1a; 在web…

【网络安全】——web渗透的前缀知识

作者名&#xff1a;Demo不是emo 主页面链接&#xff1a;主页传送门 创作初心&#xff1a;舞台再大&#xff0c;你不上台&#xff0c;永远是观众&#xff0c;没人会关心你努不努力&#xff0c;摔的痛不痛&#xff0c;他们只会看你最后站在什么位置&#xff0c;然后羡慕或鄙夷座…

ArcGIS基础实验操作100例--实验18合并表格

本实验专栏来自于汤国安教授《地理信息系统基础实验操作100例》一书 实验平台&#xff1a;ArcGIS 10.6 实验数据&#xff1a;请访问实验1&#xff08;传送门&#xff09; 基础编辑篇--实验18 合并表格 目录 一、实验背景 二、实验数据 三、实验步骤 方法一&#xff1a;导出…

whisper

Robust Speech Recognition via Large-Scale Weak Supervision 介绍 大规模弱监督的训练。先前的方法都是通过大量的无监督学习训练&#xff08;无监督的数据容易收集&#xff0c;所以通过大量无监督的学习可以训练出一个质量较好的encoder&#xff09;。但是用的时候还需要找…

Redis配置文件

Redis配置文件 自定义目录 /myreids/redis.conf Units 单位 配置大小单位&#xff0c;开头定义了一些基本的度量单位&#xff0c;只支持bytes&#xff0c;不支持bit。大小写不敏感 INCLUDES 包含 多实例的情况可以把公用的配置文件提取出来 网络配置相关 bind 默认情况…

【WSL】[01] windows subsytem linux 配置和使用 - ubuntu GUI安装

第【1】章前言&#xff1a; AI的训练和设计似乎ubuntu是必要的&#xff0c;而且&#xff0c;GPU的配置似乎也是要在Ubuntu下&#xff0c;某些模式版本才能兼容。单独搞一个编译服务器是个思路&#xff0c;但是&#xff0c;如果资金不够&#xff0c;也许要考虑在Windwos和Linux…

程序员出身备考PMP,如何避开备考误区顺利拿到3A成绩?

还在犹豫2023年如何才能提升自己的职场竞争力吗&#xff1f;PMP项目管理证书值得大家了解。掌握这些备考技巧&#xff0c;让你的PMP学习少走弯路。有计划明年3月参考PMP的小伙伴注意啦&#xff01; 今天小赛邀请了一位程序员出身的小伙伴&#xff0c;一起来看看他是如何在忙碌…

linux下如何使用configure/make/make install命令编译安装卸载程序

源码的安装一般由3个步骤组成&#xff1a;配置&#xff08;configure&#xff09;、编译&#xff08;make&#xff09;、安装&#xff08;make install&#xff09;。安装成功的源码就是所谓的可执行文件&#xff0c;在你不需要的时候&#xff0c;也是可以删除/卸载&#xff08…

leetcode1456. 定长子串中元音的最大数目

给你字符串 s 和整数 k 。 请返回字符串 s 中长度为 k 的单个子字符串中可能包含的最大元音字母数。 英文中的 元音字母 为&#xff08;a, e, i, o, u&#xff09;。 示例 1&#xff1a; 输入&#xff1a;s “abciiidef”, k 3 输出&#xff1a;3 解释&#xff1a;子字符串…

Spring的7种事务传播方式

Spring事务传播行为体现在某个service方法调用另一个service方法&#xff0c;事务该如何进行下去。 Spring支持7中事务传播方式&#xff0c;在Propagation类中可以看到&#xff0c;如下&#xff1a; REQUIRED(0), SUPPORTS(1), MANDATORY(2), REQUIRES_NEW(3), NOT_SUPPORTED…

【操作系统】进程的属性及状态(三态五态七态)

文章目录进程的概念进程的属性1、结构性2、共享性3、动态性4、独立性5、制约性6、并发性进程状态1、三态模型2、五态模型3、七态模型进程的概念 程进程是一个可并发执行的具有独立功能的程序关于某个数据集合的一次执行过程&#xff0c;也是操作系统进行资源分配和保护的基本…

Java中内部类的讲解(Java系列8)

目录 前言&#xff1a; 1.内部类 1.1内部类的概念 1.2内部类的分类 1.2.1实例内部类 1.2.2静态内部类 1.2.3局部内部类 1.2.4匿名内部类 结束语&#xff1a; 前言&#xff1a; 这次小编主要与大家分享一下什么是内部类&#xff0c;那么接下来就和小编一起来见识一下内…

aloam学习笔记(一)

开始学习aloam框架&#xff0c;记录一下最开始运行aloam中出现的各种问题以及解决方式。 1.aloam地址 GitHub - HKUST-Aerial-Robotics/A-LOAM: Advanced implementation of LOAM 2.安装aloam的一些依赖 主要是两个ceres和pcl库 2.1安装ceres ceres官方地址&#xff1a;I…

宝马335i手动挡和M3手动挡的对比

感受篇 *动力 两车的动力都堪称强悍&#xff0c;但发力感受差异非常大。335具备典型的涡轮车特征&#xff0c;动力来的比较突兀&#xff0c; 低速跟车时油门很难控制&#xff0c;给小了车走得慢&#xff0c;给大了就往前窜。转速拉到3000转以后335的动力刺激度非常高&#xff…

房产管理系统安全可靠性分析

数图互通房产管理 高校房产管理系统是基于公司自主研发FMCenter平台开发的应用系统。 一、系统安全性分析&#xff1a; 1.支持SSL传输协议&#xff0c;可以实现链路层的加密传输。 2.提供基于角色的授权体系&#xff0c;角色可自…

辞旧迎新,社科院与杜兰大学金融管理硕士项目引领你在金融的世界遇到更好的自己

不知不觉中2022年剩下最后的2天了&#xff0c;这一年中一半的时间是核酸&#xff0c;另一半是辛酸。当我们的理想快被生活磨灭的时候&#xff0c;记得把生活调成自己喜欢的频率&#xff0c;尽力而为。告别过去才能跟未来更好的相逢&#xff0c;新的一年我们的在职读研从社科院杜…

webdriver的尝试:三 【移动端的使用appium-配置】

文章目录appium安装appium与inspectorinspector配置配置信息获取inspector启动遇到的错误记录启动appium Selenium webdriver到移动端使用的是appium。 官方网站 介绍是&#xff1a;appium是一个app的自动化测试框架&#xff0c;使用webdriver协议。且可用于&#xff0c;原生…

【机器学习实战】二、随机森林算法预测出租车车费案例

随机森林算法预测出租车车费案例 一、导入第三方库 import numpy as np import pandas as pd import matplotlib.pyplot as plt import seaborn as sns import sklearn二、加载数据集 train pd.read_csv(train.csv,nrows1000000) # 加载前1000000条数据 test pd.read_csv(…

Java-抽象类与接口

前言 温故而知新 最近从头来看当初学过的语法知识点, 温故而知新, 发现当初还有许多未掌握的知识, 所以我建议大家也要多温故, 可能当初有好多知识点是没掌握到的. 这篇博客的重点就是介绍抽象类与接口, 并阐述他们的区别. 一. 抽象类 1. 概念 在面向对象的概念中, 所有的…