当前位置: 首页 > news >正文

[C语言]一维数组二维数组的大小

对于一维数组我们知道取地址是取首元素的地址,二维数组呢,地址是取第一行的地址,sizeof(数组名)这里计算的就是整个数组的大小,&数组名 表示整个数组,取出的是整个数组的地址,显示的是数组的首元素

记住这些前提后,我们看一下下面的题目会多少呢?

一维数组计算大小题目

第一组题目

#include <stdio.h>
int main()
{int a[] = { 1,2,3,4 };printf("%d\n", sizeof(a));  printf("%d\n", sizeof(a + 0));printf("%d\n", sizeof(*a));printf("%d\n", sizeof(a + 1));printf("%d\n", sizeof(a[1]));printf("%d\n", sizeof(&a));printf("%d\n", sizeof(*&a));printf("%d\n", sizeof(&a + 1));printf("%d\n", sizeof(&a[0]));printf("%d\n", sizeof(&a[0] + 1));
}

1、sizeof(a)是整个数组名所以为整个数组的大小,为4*4=16

2、sizeof(a+0)为首元素地址加0的元素地址既为1的地址,为4/8

3、sizeof(*a)是a的地址解码,是第一个元素1,大小为4

4、a+1是第二个元素的地址,同理也为4/8

5、a[1]也为第二个元素,大小为4

6、&a是去整个数组的地址,对于32地址线内存取4,对于64地址线内存取8

7、*&是取整个数组的地址再解码,因此sizeof里面就一剩一个数组名,大小为16

8、&a+1是区第二个元素的地址也为4/8

9、%a[0]是取第一个元素的地址,因此也为4/8

10、&a[0]+1是取第二个元素的地址,也为4/8

第一组题目完成了,我们来看看

第二组题目

#include <stdio.h>
int main()
{char arr[] = { 'a','b','c','d','e','f' };printf("%d\n", sizeof(arr));printf("%d\n", sizeof(arr + 0));printf("%d\n", sizeof(*arr));printf("%d\n", sizeof(arr[1]));printf("%d\n", sizeof(&arr));printf("%d\n", sizeof(&arr + 1));printf("%d\n", sizeof(&arr[0] + 1));printf("%d\n", strlen(arr));printf("%d\n", strlen(arr + 0));printf("%d\n", strlen(*arr));Cprintf("%d\n", strlen(arr[1]));printf("%d\n", strlen(&arr));printf("%d\n", strlen(&arr + 1));printf("%d\n", strlen(&arr[0] + 1));
}

1、arr为一个数组名,所以未整个数组的大小为6

2、arr+0代表了数组第一个元素地址,为4/8

3、*arr为数组第一个元素的值的大小为1

4、arr[1]也为数组第一个元素的值的大小为1

5、&arr是取数组arr的地址,大小为4/8

6、&arr+1是跳过了整个数组取下个地址的大小,既然是地址的大小输出的结果为4/8

7、&arr[0]+1为第二个元素的地址大小为4/8

8、strlen(arr)代表了求这个数组的字符长度,要看到'\0'strlen程序才会结束,而这个数组并没有'\0'所以其大小应为随机值

9、strlen(arr+0)也为和8一样的随机值

10、*arr是取数组地址的元素,strlen内部只能为地址,而其为元素,所以这个会显示错误

11、同10也会显示错误

12、&arr是取整个数组的地址,也是数组第一个元素的地址,和8、9一样是一样的随机值

13、&arr+1是跳过了整个数组后的第一个地址,取8、9不同是这个随机值比他们小6

14、地址是从第二个元素开始,所以输出的结果是比8、9小1

我们来看看这个程序运行结果是不是如此呢?

可以看到与我们预测的一样

第三组题目

#include <stdio.h>
int main()
{char arr[] = "abcdef";printf("%d\n", sizeof(arr));printf("%d\n", sizeof(arr + 0));printf("%d\n", sizeof(*arr));printf("%d\n", sizeof(arr[1]));printf("%d\n", sizeof(&arr));printf("%d\n", sizeof(&arr + 1));printf("%d\n", sizeof(&arr[0] + 1));printf("%d\n", strlen(arr));printf("%d\n", strlen(arr + 0));printf("%d\n", strlen(*arr));printf("%d\n", strlen(arr[1]));printf("%d\n", strlen(&arr));printf("%d\n", strlen(&arr + 1));printf("%d\n", strlen(&arr[0] + 1));
}

1、arr一样是整个数组的大小,但是这里记住以字符串命名会多一个'/0'所以这里的大小为7

2、sizeof(arr+0)代表了首元素的地址所以结果为4/8

3、*arr是指的首元素的值,大小为1

4、arr[1]是第二个元素的值大小为1

5、&arr是整个数组的地址,既然是地址大小就为4/8

6、&arr+1为跳过整个数组后的第一个元素的地址大小为4/8

7、&arr[0]+1为第二个元素的地址大小为4/8

8、strlen(arr)为数组的长度,这里有'\0'所以大小为6

9、strlen(arr+0)也是从首地址开始起算,所以大小也为6

10、11、这里都是取的元素,并不是地址,所以这里的值显示错误

12、&arr是整个元素的起始地址,大小也为6

13、&arr+1是整个数组后的第一个元素地址,这里计算字符串长度大小为随机值

14、&arr[0]+1是第二个元素的地址这里的大小为5

第四组题目

#include <stdio.h>
int main()
{char* p = "abcdef";printf("%d\n", sizeof(p));printf("%d\n", sizeof(p + 1));printf("%d\n", sizeof(*p));printf("%d\n", sizeof(p[0]));printf("%d\n", sizeof(&p));printf("%d\n", sizeof(&p + 1));printf("%d\n", sizeof(&p[0] + 1));printf("%d\n", strlen(p));printf("%d\n", strlen(p + 1));printf("%d\n", strlen(*p));printf("%d\n", strlen(p[0]));printf("%d\n", strlen(&p));printf("%d\n", strlen(&p + 1));printf("%d\n", strlen(&p[0] + 1));
}

这里注意命名的是char*命名,取得是后面字符串首元素地址

1、sizeof(p)为首元素地址的大小为4/8

2、是第二个元素地址的大小也为4/8

3、为首元素大小为1

4、为首元素大小为1

5、&p为首元素的地址大小为4/8

6、&p+1为第二个元素地址大小为4/8

7、&p[0]+1为第二个元素地址大小也为4/8

8、strlen(p)为首元素后找到'\0'为止的大小为6

9、为第二个元素后找到'\0'为止的大小为5

10、11、均为char类型的并不适用strlen所以会报错

12、&p是取得指针p的地址,这里计算出来的是随机值,这里并不是取p内指向的地址!

13、也为随机值,但不一定是随机值减1因为&p地址内存放的字符不一定

14、相当于取p[0]的地址再加1即为p[1]的地址大小为5,可以这么理解&(*p)+1

二维数组计算大小

#include <stdio.h>
int main()
{int a[3][4] = { 0 };printf("%d\n", sizeof(a));printf("%d\n", sizeof(a[0][0]));printf("%d\n", sizeof(a[0]));printf("%d\n", sizeof(a[0] + 1));printf("%d\n", sizeof(*(a[0] + 1)));printf("%d\n", sizeof(a + 1));printf("%d\n", sizeof(*(a + 1)));printf("%d\n", sizeof(&a[0] + 1));printf("%d\n", sizeof(*(&a[0] + 1)));printf("%d\n", sizeof(*a));printf("%d\n", sizeof(a[3]));return 0;
}

我们知道二维数组的地址取的是第一行的地址,我们来看看这几个大小是多少

1、sizeof(a)里面是个数组名既计算整个数组的大小为3*4*4=48

2、sizeof(a[0][0])为第一行第一列元素大小为4

3、a[0]为第一行,里面只有一个数组名既为第一行的大小为1*4*4=16

4、a[0]+1为第一行首元素地址加1为第一行第二个元素地址的大小为4/8

5、*(a[0]+1)为第一行第二个元素地址的解引用为其整型的大小为4

6、a+1为第二行地址,地址的大小为4/8

7、*(a+1)为第二行地址的解引用即为第二行数组的大小a[1]大小为16

8、&a[0]+1为第一行的地址加1的大小为第二行的地址大小为4/8

9、为8第二行地址的存储数据大小为16

10、*a代表了第一行元素相当于a[0]大小为16

11、a[3]为第四行元素,注意这里的系统并未访问到第四行元素,sizeof里面的值不参与运算,但是系统会认为此数组存在第四行元素所以还是显示其大小为16

相关文章:

  • Android:adb命令
  • 驱动开发中的DMA是什么
  • OPPO 后端二面,凉凉。。。
  • 2023年蓝桥杯模拟省赛——列名
  • Qt5.9.6+VS2015 部署PCL1.8.1
  • Qt笔记 信号和槽
  • vue中动态显示时间
  • JavaScript 面试题
  • Vue2 和Vue3 双向数据绑定的区别和原理
  • word转pdf怎么转换?这几个转换技巧收好
  • Python将 PDF 转换为 png 图片的教程
  • 【vue2源码】模版编译
  • 室友打团太吵?一条命令断掉它的WiFi
  • Nanya(南亚科技)DRAM芯片选型详解
  • 10:00面试,10:06就出来了,问的问题有点变态。。。
  • 实现windows 窗体的自己画,网上摘抄的,学习了
  • Angular 响应式表单之下拉框
  • Java,console输出实时的转向GUI textbox
  • Javascripit类型转换比较那点事儿,双等号(==)
  • JavaScript 无符号位移运算符 三个大于号 的使用方法
  • JS+CSS实现数字滚动
  • js中forEach回调同异步问题
  • LeetCode18.四数之和 JavaScript
  • MobX
  • OpenStack安装流程(juno版)- 添加网络服务(neutron)- controller节点
  • PHP 程序员也能做的 Java 开发 30分钟使用 netty 轻松打造一个高性能 websocket 服务...
  • SpringCloud集成分布式事务LCN (一)
  • Work@Alibaba 阿里巴巴的企业应用构建之路
  • 服务器之间,相同帐号,实现免密钥登录
  • 记一次用 NodeJs 实现模拟登录的思路
  • 理解在java “”i=i++;”所发生的事情
  • 每天一个设计模式之命令模式
  • 设计模式 开闭原则
  • 新手搭建网站的主要流程
  • 用Canvas画一棵二叉树
  • #设计模式#4.6 Flyweight(享元) 对象结构型模式
  • (30)数组元素和与数字和的绝对差
  • (cljs/run-at (JSVM. :browser) 搭建刚好可用的开发环境!)
  • (JSP)EL——优化登录界面,获取对象,获取数据
  • (分布式缓存)Redis哨兵
  • (附源码)ssm高校运动会管理系统 毕业设计 020419
  • (原)记一次CentOS7 磁盘空间大小异常的解决过程
  • (转)Linux整合apache和tomcat构建Web服务器
  • ****** 二十三 ******、软设笔记【数据库】-数据操作-常用关系操作、关系运算
  • .net 受管制代码
  • .NET 自定义中间件 判断是否存在 AllowAnonymousAttribute 特性 来判断是否需要身份验证
  • .net反混淆脱壳工具de4dot的使用
  • .NET应用架构设计:原则、模式与实践 目录预览
  • ::
  • [BetterExplained]书写是为了更好的思考(转载)
  • [C# WPF] DataGrid选中行或选中单元格的背景和字体颜色修改
  • [c#基础]DataTable的Select方法
  • [CTSC2014]企鹅QQ
  • [dart学习]第四篇:函数
  • [docker] Docker容器服务更新与发现之consul