深入理解C语言-深入理解数组
数组,作为C语言中常见的复杂数据类型,了解其本质有助于深入了解C语言
数组概念
元素类型角度:数组是相同类型的变量的有序集合测试指针变量占有内存空间大小
内存角度:联系的一大片内存空间

数组初始化
数组元素的个数可以显示或隐式指定
数组的初始化可以用{ 0 },也可以使用memset初始化
int main()
{
int i = 0;
int a[10] = { 1, 2 }; //其他初始化为0
int b[] = { 1, 2 };
int c[20] = { 0 };
for (i = 0; i < 10; i++)
{
printf("%d ", a[i]);
}
memset(a, 0, sizeof(a));
getchar();
return 0;
}
关于数组名的一些问题
- 数组首元素的地址和数组地址是两个不同的概念,前者代表一个元素大小,后者代表一个和数组大小
- 数组名代表数组首元素的地址,它是个常量,其本质就是一个变量的别名
- 数组首元素的地址和数组的地址值相等
- 数组的类型由元素类型和数组大小共同决定,e.g. int[5]
数组类型的自定义
typedef int(MYINT5)[5];
···
MYINT5i Array; //int Array[5];
数组指针
数组指针用于指向一个数组
- 通过数组类型定义数组指针
typedef int(ArrayType)[5];
ArrayType* pointer;
用数组类型加*定义一个数组指针
void main(void)
{
int a[5];//声明一个数组类型
typedef int(MYINT5)[5];//用数组类型 加*,定义一个数组指针变量
MYINT5 *array;
array = &a;
for (i = 0; i < 5; i++)
{
(*array)[i] = i;
}
for (i = 0; i < 5; i++)
{
printf("\n%d %d", a[i], (*array)[i]);
}
}
- 声明一个数组指针类型
typedef int (*MyPointer)[5];
定义一个数组指针类型,然后用类型定义变量
void main(void)
{
int b[5];//声明一个数组指针类型
typedef int (*MyPointer)[5];//用数组指针类型,去定义一个变量
MyPointer mypoint;
mypoint= &b;
for (i = 0; i < 5; i++)
{
(*mypoint)[i] = i;
}
for (i = 0; i < 5; i++)
{
printf("\n%d %d", b[i], (*mypoint)[i]);
}
}
- 直接定义
int (*pointer)[n];
直接定义一个数组指针变量
void main(void)
{
int c[5];
int (*pointer)[5] = &c;//直接声明一个数组指针变量
for (i = 0; i < 5; i++)
{
(*pointer)[i] = i;
}
for (i = 0; i < 5; i++)
{
printf("\n%d %d", c[i], (*pointer)[i]);
}
}
一维数组本质
int a[5] 一维数组名代表数组首元素的地址
int a[5] ==> a的类型为int*
二维数组本质
实验:
int arr[3][4];
printf("arr:%d, arr + 1:%d", arr, arr + 1);
此时输出打印的结果相差16,也就是4 * 4,通过这个契机,得出二维数组名的本质
arr的本质是一个数组指针,每次向后移动一维的长度
| 说明 | 写法 | 写法 | 写法 |
|---|---|---|---|
| 第0行第1列元素地址 | a [0]+1 |
*a+1 |
&a[0][1] |
| 第1行第2列元素地址 | a [1]+2 |
*(a+1)+2 |
&a[1][2] |
| 第 i 行第 j 列元素地址 | a [i]+j |
*(a+i)+j |
&a[i][j] |
| 第1行第2列元素的值 | *(a [1]+2) |
*( *(a+1)+2) |
a[1][2] |
| 第 i 行第 j 列元素的值 | *(a [i]+j) |
*( *(a+i)+j) |
a[i][j] |
正是因为有了这些特性,二维数组在做函数参数的时候,会退化为二级指针,这样设计的目的在于参数传递的时候不用传递太多数据,数组指针做函数参数,一般只会使用到二维
char buf[3][4] ==> char buf[][4] ==> char (*buf)[4]
多维数组的线性存储特性
多维数组在内存中是线性存储的,即按照低维到高维线性排序
char buf[2][3];
buf[0][0]·buf[0][1]·buf[0][2]·buf[1][0]·buf[1][1]·buf[1][2]
数组做函数参数
由于C语言中只会以机械式的值拷贝的方式传递参数(实参把值传给形参),所以在数组做函数参数的时候,是没法知道数组大小的,需要额外传递数组大小数据
int fun(char a[20], size_t b)
{
printf("%d\t%d",b,sizeof(a));
}
二维数组可以看做是一维数组
二维数组中的每个元素是一维数组
二维数组参数中第一维的参数可以省略
void f(int a[5]); ==> void f(int a[]); ==> void f(int* a);
void g(int a[3][3]) ==> void g(int a[][3]); ==> void g(int (*a)[3]);
深入理解C语言-深入理解数组的更多相关文章
- 深入理解C语言-深入理解指针
关于指针,其是C语言的重点,C语言学的好坏,其实就是指针学的好坏.其实指针并不复杂,学习指针,要正确的理解指针. 指针是一种数据类型 指针也是一种变量,占有内存空间,用来保存内存地址 指针就是告诉编译 ...
- 深入理解C语言-深入理解void
void的字面意思是"无类型",void *则为"无类型指针",void *可以指向任何类型的数据 void含义 void几乎只有注释和限制程序的作用,定义一个 ...
- 深入理解C语言-深入理解内存四区
数组与指针 当数组做函数参数的时候,会退化为一个指针 此时在函数内是得不到数组大小的 因此,数组做函数参数的时候需要传递数组大小,也就是多传递一个参数 void func(int arr[], int ...
- 深入理解C语言中的指针与数组之指针篇
转载于http://blog.csdn.net/hinyunsin/article/details/6662851 前言 其实很早就想要写一篇关于指针和数组的文章,毕竟可以认为这是C语言的根本 ...
- 深入理解C语言中的指针与数组之指针篇(转载)
前言 其实很早就想要写一篇关于指针和数组的文章,毕竟可以认为这是C语言的根本所在.相信,任意一家公司如果想要考察一个人对C语言的理解,指针和数组绝对是必考的一部分. 但是之前一方面之前一直在忙各种事情 ...
- 深入理解c语言_从编译器的角度考虑问题_纪念Dennis Ritchie先生
开源中国: Dennis Ritchie教授过世了,他发明了C语言,一个影响深远并彻底改变世界的计算机语言.一门经历40多年的到今天还长盛不训的语言,今天很多语言都受到C的影 响,C++,Java,C ...
- "深入理解C语言" 指针
本文对coolshell中的"深入理解C语言"这篇文章中提到的指针问题, 进行简要的分析. #include <stdio.h> int main(void){ ]; ...
- 理解C语言中指针的声明以及复杂声明的语法
昨天刚把<C程序设计语言>中"指针与数组"章节读完,最终把心中的疑惑彻底解开了.如今记录下我对指针声明的理解.顺便说下怎样在C语言中创建复杂声明以及读懂复杂声明. 本文 ...
- 【转载】理解C语言中的关键字extern
原文:理解C语言中的关键字extern 最近写了一段C程序,编译时出现变量重复定义的错误,自己查看没发现错误.使用Google发现,自己对extern理解不透彻,我搜到了这篇文章,写得不错.我拙劣的翻 ...
随机推荐
- 实体类,bean文件,pojo文件夹,model文件夹都一样
实体类,bean文件,pojo文件夹,model文件夹都一样,这些都是编写实体类,这是我暂时看到的项目文件
- linux终端相关概念解释及描述
基本概念: 1. tty(终端设备的统称): tty一词源于Teletypes,或者teletypewriters,原来指的是电传打字机,是通过串行线用打印机键盘通过阅读和发送信息的东西,后来这东西被 ...
- NOIP模拟9
#rank3,开心 话说这次考试时,心态并不是很好,考试前一天看了DeepinC大佬的博客,上次他rank15就 感觉炸成那样,那我上次rank30算什么?反正内心虚得一比;昨天晚上做梦梦到自己模拟赛 ...
- Neo4j (1)创建节点
节点模式的构成:(Variable:Lable1:Lable2{Key1:Value1,Key2,Value2}),实际上,每个节点都有一个整数ID,在创建新的节点时,Neo4j自动为节点设置ID值, ...
- Python数据分析:Numpy学习笔记
Numpy学习笔记 ndarray多维数组 创建 import numpy as np np.array([1,2,3,4]) np.array([1,2,3,4,],[5,6,7,8]) np.ze ...
- [dos]切换工作目录
dos切换目录: 最快速的方法: cd /d D:\your\heart 常规步骤: 1. d: 先必须切换盘符 2. cd D:\your\heart,其次切换工作目录
- linux磁盘内存满了?删除大文件依然不起作用
好久没有更新博客了,但并不代表自己没有遇到技术问题了.遇到了一大堆,也解决了一大堆.只是没有记下来的欲望了,似乎大脑就这样,忘不掉.啥都忘不掉了,即使忘掉了也知道如何百度了. 查看目录大小命令 du命 ...
- BUUCTF平台-web-边刷边记录-2
1.one line tool <?php if (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) { $_SERVER['REMOTE_ADDR'] = $_ ...
- RHEL 6.10系统安装配置图解教程
EL 6.10系统安装配置图解教程(rhel-server-6.5) 截止目前RHEL 6.x最新版本为RHEL 6.10,下面介绍RHEL 6.10的具体安装配置过程,需要的朋友可以参考下 一.安装 ...
- koa 实现下载文件
文件下载需要使用到koa-send这个插件,该插件是一个静态文件服务的中间件,它可以用来实现文件下载的功能. 1.下载页面 static/download.html <!DOCTYPE html ...