上次说了静态数组可变长,今天知道原理了

cpp加油站
• 阅读 2115

之前发了一篇文章,讲c99变长数组的,链接如下:

多年老c++程序员在静态数组这里翻船了

发出去以后有了挺多的反馈,因为这并不是一个很难的知识点,所以如果接触过的自然而然是知道,但还真有挺多人表示不知道和不相信这个事,同时我上次也只是简单的说了一下这个事,没有去讲解这个变长静态数组的实现原理,今天补上。

先看一下思维导图:

上次说了静态数组可变长,今天知道原理了

1. 变长数组是长度一直可以变的吗

变长数组,那么是长度一直可以变的吗,到底什么时候这个长度会确定下来呢?

我们先看一下代码,如下:

#include <iostream>
using namespace std;

int main()
{
    int size = 1000;
    cout << "please input a number:";
    cin >> size;
    int arr[size];
    cout << "please input a number too:";
    cin >> size;
    cout << "arr's size is " << sizeof(arr)/sizeof(arr[0]) << endl;

    return 0;
}

假设我们第一次输入100,第二次输入10000,那么最后一个cout到底输出多少呢,答案是100。

这里的所谓变长数组,实际上指的是可以使用变量来作为数组的元素个数,在还未运行到声明数组的地方时,还可以通过改变变量的值来修改数组的元素个数,但是等到运行到数组声明的地方后,这个数组的大小就确定了,后续就不能再改变了,所以所谓的变长也只是相对于运行到这个数组声明的地方而言。

2. 变长数组是分配在堆上吗

当然不是,注意这里概念不要搞混淆了,变长数组不是动态数组,虽然是到运行时才确定大小,但说到底它还是局部变量,而局部变量,又没有动态申请内存的动作,当然也只会在栈上分配内存。

关于这一点我们可以结合gdb和寄存器地址来看一下,我的机器是x86架构,那么栈的内存分配方式就应该是从高地址往低地址分配,如果学过汇编的会知道,gcc编译的时候不添加-O1或者-O2这样的选项,那么栈底指针地址会保存在寄存器rbp中,而栈顶指针则会保存在rsp寄存器中,如果地址是在rsp和rbp之间的,那就可以肯定这段内存是保存在栈中了。

还是这段代码:

//test.cpp
#include <iostream>
using namespace std;

int main()
{
    int size = 1000;
    cout << "please input a number:";
    cin >> size;
    int arr[size];

    return 0;
}

使用g++ -g test.cpp(注意这里千万不能加优化选项)编译后,gdb ./a.out看一下:

(gdb) b main #打断点
Breakpoint 1 at 0x4007fd: file test.cpp, line 11.
(gdb) r
Starting program: /root/a.out 

Breakpoint 1, main () at test.cpp:11
11        return 0;
(gdb) n
6        int size = 1000;
(gdb) 
7        cout << "please input a number:";
(gdb) 
8        cin >> size;
(gdb) 
please input a number:10000
9        int arr[size];
(gdb) p $rbp  #打出基址寄存器的值
$1 = (void *) 0x7fffffffe840
(gdb) n
11        return 0;
(gdb) p &arr[0]
$3 = (int *) 0x7fffffff4ba0
(gdb) p $rsp  #打出栈顶指针的值
$4 = (void *) 0x7fffffff4ba0
(gdb) 

可以看到,arr数组的首地址是在(rsp, rbp)这个范围之间的,所以运行时的变长数组也是保存在栈中的,既然是在栈中,那么这段内存在这个变量作用域结束以后就会被释放掉。

3. 变长数组与alloca函数

同时这次也是涨知识了,我才知道c标准库中有个alloca函数,它的用法类似malloc,但内存因为是在栈上申请的,也是不需要手动释放的,这么一看,这个函数的原理和变长数组其实是一样的,都可以使用运行时才确定大小的变量值来申请栈空间,并且不需要手动释放。

知道以后,我也很好奇,所以还专门去研究了下这两者的汇编指令是不是一样,先看下c++代码:

#include <iostream>
#include <alloca.h>
using namespace std;

int main()
{
    int size = 1000;
    cout << "please input a number:";
    cin >> size;
    int arr[size];
    int *p = (int*)alloca(size);

    return 0;
}

然后通过gdb,我分别打印出来int arr[size]int *p = (int*)alloca(size)这两行代码所对应的汇编指令,做了一个对比,如图:

上次说了静态数组可变长,今天知道原理了

可以看得出来变长数组是在一开始先做了一些其他的动作,然后后面的指令跟alloca的指令基本就是一样的了,也就是说最终他两的实现是比较类似的。

4. 变长数组使用注意点

基于变长数组的特点,它其实相当于一个变相版的动态申请内存,只是不需要堆而言,而这种场景多应用于小型机里面,比如很多嵌入式环境,因为资源有限,是没有堆内存的,那如果又需要动态改变数组大小怎么办,就可以使用变长数组,但这时也需要限定一下大小,不然很容易就会造成栈溢出。

好了,本篇文章就为大家介绍到这里,觉得内容对你有用的话,记得顺手点个赞哦~

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
3个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Stella981 Stella981
3年前
KVM调整cpu和内存
一.修改kvm虚拟机的配置1、virsheditcentos7找到“memory”和“vcpu”标签,将<namecentos7</name<uuid2220a6d1a36a4fbb8523e078b3dfe795</uuid
Easter79 Easter79
3年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Stella981 Stella981
3年前
JS 对象数组Array 根据对象object key的值排序sort,很风骚哦
有个js对象数组varary\{id:1,name:"b"},{id:2,name:"b"}\需求是根据name或者id的值来排序,这里有个风骚的函数函数定义:function keysrt(key,desc) {  return function(a,b){    return desc ? ~~(ak
Stella981 Stella981
3年前
LeetCode 5561. 获取生成数组中的最大值
文章目录1\.题目2\.解题1\.题目给你一个整数n。按下述规则生成一个长度为n1的数组nums:nums00nums11当2<2i<n时,nums2inumsi
Wesley13 Wesley13
3年前
GNU C 与 ANSI C的区别
1.零长度数组GNUC允许使用零长度数组,定义变长度对象时比较方便structvar\_data{   intlen;   chardata\0\;};var\_data的大小仅为一个int型,data是常量地址,data\index\是访问其后的内存空间。structvar\_data\smal
达里尔 达里尔
10个月前
给数组添加新数据,判断数据是否重复
多选要进行数组拼接,希望判断往原数组里添的新数据是否重复,封装个简易方法languageconstdataArrayname:'aaa',id:1,name:'bbb',id:2;constnewDataname:'ccc',id:2;//要添加的新数
Python进阶者 Python进阶者
9个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这