目录

文章目录

  • 目录
  • 前文列表
  • 数组
  • 声明数组
  • 初始化数据
  • 访问数组元素
  • 二维数组
  • 指向数组的指针
  • 将数组指针作为实参传入函数
  • 从函数返回一个数组指针
  • 指针数组
  • 数组名和取数组首地址的区别

前文列表

《程序编译流程与 GCC 编译器》
《C 语言编程 — 基本语法》
《C 语言编程 — 基本数据类型》
《C 语言编程 — 变量与常量》
《C 语言编程 — 运算符》
《C 语言编程 — 逻辑控制语句》
《C 语言编程 — 函数》
《C 语言编程 — 高级数据类型 — 指针》

数组

数组是具有相同数据类型,并且按照一定顺序排列的一组变量的集合。

数组都是由连续的内存空间组成的,最低的地址对应第一个元素,最高的地址对应最后一个元素。数组中的特定元素可以通过索引访问,数组的索引从 0 开始。


特征:

  • 有序性:数组元素之间具有固定的先后顺序
  • 可索引:通过数组名和下标可以唯一地确定数组中的元素

声明数组

在 C 中要声明一个数组,需要指定元素的类型元素的数量

  • 其中元素的数量为不可变量,C/C++ 不允许对数组的长度做动态定义。
  • 数组名除了作为数组辨识名称之外,也表示了该数组存储空间的首地址,即数组名本身就是数组的内存入口地址,指向第一个元素。
  • 一次只能使用数组中的单个元素,而不能一次使用整个数组。
// 正确用法:一次只能使用数组中的单个元素。
// 将 aArray 的数据复制到 bArray 中。
int aArray[5] = {1, 2, 3, 4, 5};
int bArray[5] = {0};
for(int i = 0; i < 5; i++)
{bArray[i] = aArray[i];
}// 错误用法:
int aArray[5] = {1, 2, 3, 4, 5};
int bArray[5] = {0};
bArray = aArray;    //不可给整个数组赋值

需要注意的是,使用数组时,需要主动的对数组变量进行边界检查。C/C++ 在编译过程并没有缺省的边界检查动作,所以在程序运行过程中当数组下标索引值越界时,并不会立即触发错误,存在潜在的逻辑异常风险。

#include <stdio.h>
// 如下例所示,aArray[i*j] 在程序进行过程中,下标会超出其数组大小。
// 但是在编译和运行过程中,并不会报错,因此必须由编程人员对此边界进行处理!
int main(void)
{int i, j;int aArray[5] = {1, 2, 3, 4, 5};for (i = 0; i < 5; i++){for (j = 0; j < 5; j++){aArray[i * 5 + j] = i * 5 + j;printf("aArray[%d]=%d\r\n", i * 5 + j, aArray[i * 5 + j]);}}return 0;
}/*
[root@c-dev ~]# ./main
aArray[0]=0
aArray[1]=1
aArray[2]=2
aArray[3]=3
aArray[4]=4
aArray[5]=5
aArray[11]=32718
aArray[10]=10
aArray[11]=11
aArray[12]=12
aArray[13]=13
aArray[14]=14
aArray[15]=15
aArray[16]=16
aArray[17]=17
aArray[18]=18
aArray[19]=19
aArray[20]=20
aArray[21]=21
aArray[22]=22
aArray[23]=23
aArray[24]=24
段错误
*/// 上错误示例可改为:
// 当然像示例中的简单数组越界在编程过程中是十分容易避免的,但对于复杂度高的问题,必须是要增加边界检查的。
int main(void)
{int i, j, idx;int aArray[5] = {1, 2, 3, 4, 5};for (i = 0; i < 5; i++){for (j = 0; j < 5; j++){idx = i * 5 + j;if (idx > sizeof(aArray) / sizeof(int)){printf("Err: idx over range!\r\nMax idx = %lu,idx=%d\r\n", sizeof(aArray) / sizeof(int), idx);return 0;}aArray[idx] = idx;printf("aArray[%d]=%d\r\n", idx , aArray[idx]);}}return 0;
}/*
[root@c-dev ~]# ./main
aArray[0]=0
aArray[1]=1
aArray[2]=2
aArray[3]=3
aArray[4]=4
aArray[5]=5
Err: idx over range!
Max idx = 5,idx=6
*/

初始化数据

  • 指定数组长度的初始化:大括号 { } 之间的值的数目不能大于我们在数组声明时在方括号 [ ] 中指定的元素数目。
double balance[5] = {1000.0, 2.0, 3.4, 7.0, 50.0};
  • 不指定数组长度的初始化:数组的长度则为初始化时元素的个数。
double balance[] = {1000.0, 2.0, 3.4, 7.0, 50.0};
  • 对指定的元素进行赋值
balance[4] = 50.0;

访问数组元素

数组元素可以通过数组名称加索引进行访问。元素的索引是放在方括号内,跟在数组名称的后边。

将一个数组元素取出并赋值给新的变量:

double salary = balance[9];

二维数组

C 语言支持多维数组。多维数组声明的一般形式如下:

type name[size1][size2]...[sizeN];int threedim[5][10][4];
  • 二维数组

初始化二维数组:

int a[3][4] = {  {0, 1, 2, 3} ,   /*  初始化索引号为 0 的行 */{4, 5, 6, 7} ,   /*  初始化索引号为 1 的行 */{8, 9, 10, 11}   /*  初始化索引号为 2 的行 */
};//orint a[3][4] = {0,1,2,3,4,5,6,7,8,9,10,11};

访问二维数组元素:

int val = a[2][3];

指向数组的指针

数组变量名(标识符)的本质是一个指向数组中第一个元素的常量指针。

double balance[50];

如上,变量名 balance 是一个指向内存地址 &balance[0] 的指针,即数组 balance 的第一个元素的地址。使用数组名作为常量指针是合法的,反之亦然。因此,*(balance + 4) 是一种访问 balance[4] 数据的合法方式。

将数组指针作为实参传入函数

如果函数想接受一个数组(实际上是指针数组入口的指针)作为实参,那么函数必须使用以下三种方式之一来声明函数的形式参数,每种方式都是告诉编译器函数将要接收一个整型指针。同样地,也可以传递一个多维数组作为形式参数。

  • 方式 1
void myFunction(int *param){}
  • 方式 2
void myFunction(int param[10]){}
  • 方式 3
void myFunction(int param[]){}

示例:

#include <stdio.h>/* 声明一个函数形参为整型指针类型 */
double getAvg(int arr[], int size);int main(){/* 定义并初始化一个数组变量 */int balance[5] = {1000, 2, 3, 17, 50};/* 传递一个指向数组的指针作为函数实参 */double avg = getAvg(balance, 5);printf("AVG: %f", avg);return 0;
}double getAvg(int arr[], int size){int i;double avg;double sum = 0;for(i = 0; i < size; ++i){sum += arr[i];}avg = sum / size;return avg;
}

从函数返回一个数组指针

C 语言不允许函数返回一个完整的数组,但是可以返回一个指向数组的指针。注意,C 不支持在函数外部返回局部变量的地址,除非定义局部变量为 static 变量。

int * myFunction(){}

示例:

#include <stdio.h>
#include <stdlib.h>
#include <time.h>/* 定义返回整型指针类型结果的函数 */
int * getRandom(){static int r[10];int i;srand((unsigned)time(NULL));for(i = 0; i < 10; ++i){r[i] = rand();printf("r[%d] = %d\n", i, r[i]);}return r;
}int main(){/* 定义一个整型指针变量 */int *p;int i;p = getRandom();for(i = 0; i < 10; i++){printf("*(p + %d): %d\n", i, *(p + i));}return 0;
}

运行:

$ ./main
r[0] = 640773756
r[1] = 1617898688
r[2] = 2004130180
r[3] = 494154148
r[4] = 1999308605
r[5] = 959614519
r[6] = 81389324
r[7] = 1893093458
r[8] = 2121376870
r[9] = 1095386666
*(p + 0): 640773756
*(p + 1): 1617898688
*(p + 2): 2004130180
*(p + 3): 494154148
*(p + 4): 1999308605
*(p + 5): 959614519
*(p + 6): 81389324
*(p + 7): 1893093458
*(p + 8): 2121376870
*(p + 9): 1095386666

指针数组

有一种情况,我们想用数组来存储指向 int 或 char 或其他数据类型的指针。下面是一个指向整数的指针数组的声明:

int *ptr[MAX];

在这里,把 ptr 声明为一个数组,由 MAX 个整型指针组成。因此,ptr 中的每个元素,都是一个指向 int 值的指针。

#include <stdio.h>const int MAX = 3;int main ()
{int  var[] = {10, 100, 200};int i, *ptr[MAX];for ( i = 0; i < MAX; i++){ptr[i] = &var[i]; /* 将 int 值的地址赋值到整型指针数组 */}for ( i = 0; i < MAX; i++){printf("Value of var[%d] = %d\n", i, *ptr[i] );}return 0;
}

数组名和取数组首地址的区别

int array[4] = {0};

以上述语句为例:

  • 数组名 array 本质是数组首元素的入口地址。
  • 数组首元素的入口地址 &array[0]
  • &array 取的是整个数组 array 的首地址。

虽然,&array&array[0]array 的数值相同,但本质却不同。这一点体现在 +1 偏移量上是非常明显的。

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

运行:

$ ./mainarray = 0x7ffeecd8c830&array = 0x7ffeecd8c830&array[0] = 0x7ffeecd8c830array + 1 = 0x7ffeecd8c834&array + 1 = 0x7ffeecd8c840&array[0] + 1 = 0x7ffeecd8c834sizeof(int) = 4sizeof(array[0]) = 4sizeof(array) = 16sizeof(&array) = 8
sizeof(&array[0]) = 8

结论

  1. 在 C 中,几乎所有使用数组的表达式中,数组名的值是个指针常量,也就是数组第一个元素的地址。 它的类型取决于数组元素的类型: 如果它们是 int 类型,那么数组名的类型就是 “指向 int 的常量指针“。
  2. 在以下两种场合,数组名并不是用指针常量来表示:1)当数组名作为 sizeof 运算符操作数时,返回整个数组的长度,而不是指向数组的指针的长度;2)当数组名作为地址运算符 & 的操作数时,返回的是一个指向整个数组的指针,而不是指向数组首元素的指针。
  3. 指针的 +1 是偏移量问题:一个类型为 X 的指针的移动,是以 sizeof(X) 为移动步进的。
    1. array+1:在数组首元素的内存地址的基础上,偏移一个 sizeof(array[0]) 单位。上例中,为 0x7fffe3743ae0 + 1 * sizeof(array[0]) == 0x7fffe3743ae0 + 1 * sizeof(int) == 0x7fffe3743ae0 + 4 == 0x7fffe3743ae4
    2. &array+1:在数组的首地址的基础上,偏移一个 sizeof(array) 单位。上例中,为 0x7fffe3743ae0 + 1 * sizeof(array) == 0x7fffe3743ae0 + 1 * sizeof(int) * 0x4 == 0x7fffe3743ae0 + 0x16 == 0x7fffe3743af0

简而言之,数组名 array 的指针是数组基类型(首元素)的地址长度,而 &array 的指针是整个数组的地址长度。

C 语言编程 — 高级数据类型 — 数组相关推荐

  1. Go 语言编程 — 高级数据类型 — 数组

    目录 文章目录 目录 数组 定义数组 数组与切片的区别 访问数组元素 多维数组 初始化二维数组 访问二维数组 向函数传递数组 指针数组 数组 数组是具有相同数据类型的一组已编号且长度固定的数据项序列, ...

  2. C 语言编程 — 高级数据类型 — void 类型

    目录 文章目录 目录 前文列表 void 类型 前文列表 <程序编译流程与 GCC 编译器> <C 语言编程 - 基本语法> <C 语言编程 - 基本数据类型> & ...

  3. C 语言编程 — 高级数据类型 — 共用体

    目录 文章目录 目录 前文列表 共用体 定义共用体 访问共用体成员 前文列表 <程序编译流程与 GCC 编译器> <C 语言编程 - 基本语法> <C 语言编程 - 基本 ...

  4. C 语言编程 — 高级数据类型 — 结构体与位域

    目录 文章目录 目录 前文列表 结构体 定义结构体 初始化结构体变量 访问结构体成员 结构体的内存分布 将结构体作为实参传入函数 指向结构体变量的指针 位域 定义位域 使用位域结构体的成员 前文列表 ...

  5. C 语言编程 — 高级数据类型 — 枚举

    目录 文章目录 目录 前文列表 声明枚举类型 定义枚举类型的变量 枚举类型变量的枚举值 枚举在 switch 语句中的使用 将整型转换为枚举类型 前文列表 <程序编译流程与 GCC 编译器> ...

  6. C 语言编程 — 高级数据类型 — 字符串

    目录 文章目录 目录 前文列表 字符串 字符串拷贝 字符串比较 strcmp strncmp 前文列表 <程序编译流程与 GCC 编译器> <C 语言编程 - 基本语法> &l ...

  7. C 语言编程 — 高级数据类型 — 指针

    目录 文章目录 目录 前文列表 指针 声明一个指针变量 使用指针 空指针 悬空指针 野指针 指针的算术运算 指向指针的指针 将指针作为实际参数传入函数 从函数返回指针 一个古老的笑话 前文列表 < ...

  8. Go 语言编程 — 高级数据类型 — 结构体

    目录 文章目录 目录 结构体 访问结构体成员 向函数传递结构体 结构体指针 结构体标签(Struct Tag) 结构体 Golang 中,结构体是由一系列具有相同类型或不同类型的数据构成的数据集合.与 ...

  9. Go 语言编程 — 高级数据类型 — Interface、多态、Duck Typing 与泛式编程

    目录 文章目录 目录 Golang 的接口 Interface 实例存储的是实现者的值 如何判断某个 Interface 实例的实际类型 Empty Interface Interface 与多态 I ...

最新文章

  1. html 资源缓存,解决index.html缓存问题
  2. 在线CSS工具及相关资源收集
  3. linux vim编辑kconfig 无法wq,编译linux-3.15.5时遇到的几个错误
  4. 3、ShardingSphere 之 Sharding-JDBC 实现水平分库
  5. 理解SharePoint中的Managed Path
  6. 如何把 DropDownList 某一个 Item 的 Text 改成粗体 ?
  7. spring mvc 初步接触学习笔记
  8. C/C++基础知识:函数指针和指针函数的基本概念
  9. 使用Maven Jetty插件
  10. Java ByteArrayInputStream reset()方法及示例
  11. java math 函数_Java中Math类常用函数总结
  12. 前方危险-让很多“高逼格”高管深刻反思的文章
  13. Spring Cloud 之 Feign 使用HTTP请求远程服务
  14. Mike Krueger 加入Mono团队
  15. Service Mesh-Linkerd安装与使用
  16. C语言学生成绩排名系统
  17. 视频封装格式篇(TS)
  18. 携程的旅游知识图谱构建和应用
  19. android 获取渠道信息,Android 如何获取 umeng 的 渠道信息
  20. 网吧客户信息查询c语言,网吧经营管理之客户定位

热门文章

  1. php mysql table_关于php:MySQL Table不存在错误,但确实存在
  2. 选择之后触_如果有朋友在做选择时左右为难,我应该该给出什么样的建议
  3. qt调用c语言编写的dll文件,Qt之调用外部DLL - moki_oschina的个人空间 - OSCHINA - 中文开源技术交流社区...
  4. 计算机术语所见即所得,计算机应用基础作业四(16页)-原创力文档
  5. linux内核更新/修补程序,Ubuntu 18.04.3 LTS无需重启即可轻松修补Linux内核
  6. 小程序订单点击不同页面_小程序跳转页面参数丢失
  7. 从脑电图(EEG)中提取稳定的模式进行识别
  8. 使用Leap Motion Orion开发酷炫的手势识别VR/AR应用
  9. 足球?光头?AI:这道题可太难了
  10. GitHub现在已支持函数定义跳转,妈妈再也不怕我记性差了