C初阶2:数组

1. 初始化

1.1 整体初始化

试一下下面三段代码,分析输出。

  • 代码一
int arr[12];
for(int i=0;i<12;++i){
    printf("%d ",arr[i]);
}
  • 代码二
int arr[12] = {0};
for(int i=0;i<12;++i){
    printf("%d ",arr[i]);
}
  • 代码三
int arr[12] = {2};
for(int i=0;i<12;++i){
    printf("%d ",arr[i]);
}

结论:

  1. 数组未初始化,数组里面的值都是随机值。
  2. 数组初始化为{0},数组里面的值都是0
  3. 数组初始化为{非零值},数组里面第一个值是非零值,其他的值都是0

1.2 部分初始化

试一下代码,分析输出。

int arr[12] = {[2]=2,[5]=5};
for(int i=0;i<12;++i){
    printf("%d ",arr[i]);
}

指定下标的值被赋值,其他的值都是0。这是C99语法。

1.3 大小

试一下,下面代码分析结果。

char carr[12];
int iarr[12];
double farr[12];
printf("carr = %d\n",sizeof(carr));
printf("iarr = %d\n",sizeof(iarr));
printf("farr = %d\n",sizeof(farr));

sizeof给出整个数组所占据的内容的大小。数组大小=元素大小*数组个数。

问题:已知数组arr,如何求出数组元素个数?

1.4 赋值

试一下下面的代码

int days[]={31,28,31,30,31,30,31,31,30,31,30,31};
int arr = days;

2. 数组与指针

2.1 数组名是地址

数组名是数组第一个元素的地址。

int arr[] = {1,2,3,4,5,6,7,8};
printf("&arr=%p\n",&arr);
printf("arr=%p\n",arr);
printf("&arr[0]=%p\n",&arr[0]);

数组下标实现的操作指针也可以实现。数组名+下标表示下标i元素的地址。

No. 操作 下标 指针
1 i个元素值 arr[i] *(arr+i)
2 i个元素地址 &arr[i] arr+i

所以,遍历数组可以是

int arr[] = {1,2,3,4,5,6,7,8};
for(int i=0;i<8;++i){
    printf("%d\n",*(arr+i)); 
}

数组名是不可改变的。

int arr1[] = {1,2,3,4,5,6,7,8};
int arr2[] = {1,2,3,4,5,6,7,8};
arr2 = arr1

数组下标比较易于理解,数组指针更灵活更高效。
要点

  1. 数组名是数组第一个元素的地址。
  2. 数组名+下标表示下标i元素的地址。
  3. 数组名是不可改变的。

2.2 数组名放入指针

地址存储在指针中。可以用指针操作地址。

int arr[] = {1,2,3,4,5,6,7,8};
int* p = arr;
for(int i=0;i<8;++i){
    printf("%d\n",*(p+i)); 
}

3. 函数与数组

3.1 传递数组给函数

数组作为函数参数时,通常必须再用一个参数传入数组大小。

返回值类型 函数名(类型 参数名[],int size){

}

或者

返回值类型 函数名(类型* 参数名,int size){

}

例如:打印数组元素

#include <stdio.h>

void PrintArr1(int arr[],int n){
    for(int i=0;i<n;++i){
        printf("%d ",arr[i]);
    }
    printf("\n");
}
void PrintArr2(int* arr,int n){
    for(int i=0;i<n;++i){
        printf("%d ",arr[i]);
    }
    printf("\n");
}
int main(){
    int arr[] = {1,2,3,4,5,6,7,8,9};
    PrintArr1(arr,9);
    PrintArr1(arr,3);
    PrintArr1(arr+6,3);
    PrintArr1(arr+3,3);
}

试一下下面代码

#include <stdio.h>

void Test1(int a[]){
    printf("sizeof(a):%d\n",sizeof(a));
    printf("a:%p\t&a:%p\n",a,&a);
}
void Test2(int* p){
    printf("sizeof(p):%d\n",sizeof(p));
    printf("p:%p\t&p:%p\n",p,&p);
}
int main(){
    int arr[10];
    printf("sizeof(arr):%d\n",sizeof(arr));
    printf("arr:%p\t&arr:%p\n",arr,&arr);
    Test1(arr);
    Test2(arr);
}

数组作为参数时,数组退化成指针,不能利用sizeof获取数组大小,也就不能计算数组元素个数。

3.2 从函数返回数组

C 语言不允许返回一个完整的数组作为函数的参数。通过指定不带索引的数组名来返回一个指向数组的指针。

类型* 函数名() {
   return 数组名;
}

练习

  1. 给定数组随机填充指定范围的数据。
    使用stdlib.h中的srand(time(NULL))rand()
  2. 将一维数组的查找FindArr()、替换Replace()封装成函数。

FindArr()可以返回bool(有无)、int(下标)、int*(地址)有什么区别。

  • 返回值含义(返回值的含义以及值是人为定义的)
    返回值有时存在两种情况:合法值和非法值。
    如果有非法值的情况,通常使用一些特定的值指代特殊情况。例如:数组下标只能是0和正数。我们实现数组查找元素下标时,存在找不到元素的情况,这时,使用-1可以作为这种情况的返回值。

重点

int arr[] = {1,2,3,4,5,6};
// sizeof(数组名)
printf("sizeof(arr) = %ld\n",sizeof(arr));
// 数组名的值
printf("arr=%p\n&arr[0]=%p\n",arr,&arr[0]);

数组名是地址,是常量。

  1. 可以使用指针操作:解引用
  2. 数组作为函数参数和返回值时,传指针/地址。

数组下标与数组长度的关系

  1. arr长度为n,最后一个元素的下标为n-1
  2. 下标为i是第i+1个元素,前面有i个元素(不包含i),后面有n-(i+1)个元素(不包含i)。
  3. 下标i和j之间有j-i个元素(包含i,不包含j)

数组地址与数组名的关系

  1. 数组arr的首地址为arr,下标为i的地址为arr+i
  2. p=arr+i,那么下标对应为p-arr
  3. p前面有p-arr个元素(不包含p),后面有n-(p-arr+1)个元素(不包含p)。

4. 多维数组

4.1 声明

  • 语法
类型 数组名[元素个数1][元素个数2]...[元素个数N];

多维数组最常用形式是二维数组。二维数组相当于一个行列组成的表。

类型 二维数组名[行数][列数];

例如下面是一个43列元素类型为int的表。

int days[4][3];

4.2 初始化二维数组

多维数组可以通过在括号内为每行指定值来进行初始化。

int days[4][3]={
    {31,28,31},
    {30,31,30},
    {31,31,30},
    {31,30,31}
};

4.3 访问二维数组元素

二维数组中的元素是通过使用下标(即数组的行索引和列索引)来访问的。


// 获取一月份的天数
int n = days[0][0];
// 修改二月份的天数
days[0][1] = 29;
// 打印一月份天数
printf("%d",days[0][0]);

4.4 二维数组元素遍历

通常使用嵌套循环来处理二维数组。

for (int i = 0; i < 4; i++ ) {
    for (int j = 0; j < 3; j++ ) {
       printf("a[%d][%d] = %d\n",i,j,a[i][j]);
    }
}

4.5 二维数组输入

int n,m;
scanf("%d%d",&n,&m);
int arr[n][m];
for (int i=0;i<n;++i){
  for (int j=0;j<m;++j){
      scanf("%d",&arr[i][j]);
  }
}

练习

  1. 输入m个学生的n门课程成绩,输出各门课程成绩以及总成绩和平均值。

  2. 二维数组通常可用于表示数学中的矩阵。实现矩阵基本运算(加法、减法、乘法、转置)。


  3. mn月的日期,按星期以日历方式放入二维数组(列表示星期),并打印出来。

4.6 简化

初始化二维数组可以有如下简化写法:

  1. 省略内部嵌套括号。
int days[4][3]={31,28,31,30,31,30,31,31,30,31,30,31};
  1. 省略第一维大小,第二维不能省略。
int days[][3]={31,28,31,30,31,30,31,31,30,31,30,31};

问题
二维数组可以整体初始化吗?
练习

4.7 二维数组作为函数参数

二维数组作为函数参数,需要同属输入行和列的个数。

void PrintMatrix(int m[4][3],int r,int c)
for (int i = 0; i < r; i++ ) {
    for (int j = 0; j < c; j++ ) {
       printf("%d\n",m[i][j]);
    }
}

4.8 多维数组

大于二维的数组的用法与二维数组一样,只是使用比较少。

int days[][4][3]={
    {31,28,31,30,31,30,31,31,30,31,30,31}, // 平年
    {31,29,31,30,31,30,31,31,30,31,30,31}  // 闰年
};

printf("平年二月天数为%d\n",days[0][0][1]);// 平年第一季度第二个月
printf("闰年二月天数为%d\n",days[1][0][1]);// 闰年第一季度第二个月 

多维数组初始化只能第一个维度可以省略。

5 const数组

5.1 const数组是什么?

const int arr[]={1,2,3,4,5,};

数组变量已经是const指针,表示数组中的每一个元素都是const int,即每个元素不能通过arr改变。
例如:

const int arr[]={1,2,3,4,5,};
arr[0] = 0;

5.2 const数组怎么用?

保护数组值

因为数组作为函数参数是以地址方式传递的,所以函数内部可以修改数组的值。
为了保护数组不被函数破坏,可设置参数为const
例如:

int sum(const int arr[],int len);

或者

int sum(const int* arr,int len);

6 变量指针 vs 数组指针

变量指针:指向单个变量的指针。
数组指针:指向数组的指针。

#include <stdio.h>

int main () {
    int n = 10;
    int *p;
    p = &n; // p指针指向变量
    printf("*p = %d\n",*p);
    
    int arr[] = {1,2,3,4,5,};
    p = arr;// p指针指向数组
    printf("*p = %d\n",*p);
    printf("*(p+1) = %d\n",*(p+1));
    printf("*(p+2) = %d\n",*(p+2));
    printf("*(p+3) = %d\n",*(p+3));
    printf("*(p+4) = %d\n",*(p+4));
    
    return 0;
}

指针既可以指向一个基本类型变量又可以指向一个数组。所以在使用时要注意分辨。

7 项目

输入年份打印万年历。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 教育心理学是一门研究学校内外学生的发展、学习、动机、教学与评估的学科。 1、教师效能感:由于大众媒体的影...
    蕴韵1236阅读 288评论 0 6
  • 小编最近补看了欢乐喜剧人,节目段子各种搞笑!但是更迷上了这个煤炭一样黑的男人有没有!! ...
    慢慢狂奔阅读 10,473评论 0 0
  • 互联网+医疗行业,会孕育大量的医疗健康服务创业公司,在医生、患者、医院、医药、单科领域、智能硬件、医疗新媒体等领域...
    阿柯玛地德阅读 308评论 0 0
  • 1、描述:名字《明天》 2、顺序: 房子一路一花草一右侧树一左侧树一云朵一小鸟一人物(我,女儿,老公)本来画的火柴...
    淘伊_心向光明阅读 437评论 6 1
  • #玩卡不卡·每日一抽#每一位都可以通过这张卡片觉察自己: 1、直觉他叫什么名字?保罗 2、他几岁了?40 3、他现...
    我心安住阅读 137评论 0 0

友情链接更多精彩内容