C语言二维数组

二维数组

  • 二维数组的创建
  • 二维数组的初始化
  • 二维数组的使用
  • 二维数组在内存中的存储
  • 数组越界
  • 数组做为函数传参
  • 数组名是什么
  • 二维数组的数组名

二维数组的创建

二维数组的创建和一维数组数组差不多,只不过一维数组只是一行一行
二维数组就是一个有行和列的矩阵,每一行代表一个数组
二维数组可以理解:一维数组的数组,每一行都可以看做是一个一维数组

例如:

int arr[3][4] = { {1,2,3,4}, {2,3,4,5}, {3,4,5,6} };
int arr[2][4] = {1,2,3,2,3,4,3,4,5};
char ch[3][4];
double ch[2][3];

二维数组的初始化

二维数组初始化和一维数组有细微的差别

int arr[3][4] = { {1,2}, {2,3}, {4,5} };  
这叫不完全初始化,后面的数值自动补 0
char 类型 补的是 \0

C语言二维数组_第1张图片
二维数组如果有初始化,行可以省略,列不能省略

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

二维数组的使用

二维数组的使用也是通过下标来使用的
例如:
int arr [ 3 ] [ 4 ] = { {1,2,3,4}, {2,3,4,5}, {3,4,5,6} };
这是创建 3 列 4 行的数组

假设我要找到:
第二行的 3   arr[1][1]
第一行的 4   arr[0][3]
第三行的 5   arr[2][2]

C语言二维数组_第2张图片
如果我们要打印二维数组里的全部元素
看代码:

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

二维数组在内存中的存储

二维数组在内存中也是连续存放的,和一维数组一样的,一行内部连续或者跨行都是一样的
地址之间的差值也是 4 个字节

C语言二维数组_第3张图片
从下面图片可以看出二维数组在内存中和一维数组存放的方法都是一样的
在内存中二维数组 arr[3][4] 可以看做 一维数组arr[12],这二种在内存的存放是一样的
不过存在形式不一样
C语言二维数组_第4张图片

数组越界

数组的下标是有范围限制的。
数组的下规定是从0开始的,如果数组有n个元素,最后一个元素的下标就是n-1。
所以数组的下标如果小于0,或者大于n-1,就是数组越界访问了,超出了数组合法空间的访问
C语言二维数组_第5张图片
例如:

int main()
{
 int arr[6] = {1,2,3,4,5,6};
    int i = 0;
    for(i=0; i<=10; i++) 当i等于7的时候,越界访问了
   {
        printf("%d\n", arr[i]);
   }
 return 0;
}

数组越界后,打印的都是随机值
C语言本身是不做数组下标的越界检查,编译器也不一定报错,但是编译器不报错,并不意味着程序就
是正确的,所以自己在写代码的时候要自己检查,免得写出BUG

那么二维数组越界呢
二维数组越界是访问后面的下标,数值也是出错了
C语言二维数组_第6张图片

数组做为函数传参

数组传参传的是数组,传过去后形参可以写成数组和指针的形式
本质上都是指针,传过去其实是数组的首元素地址

数组做为函数传参形参可以写成 2 种形式:

数组的形式:
1. void test1( int arr[])
形参指针的形式:
2. void test(int * arr)

数组名是什么

数组名是首元素的地址
但是有2个例外:

  1. sizeof(数组名) - 数组名表示整个数组,计算的是整个数组的大小,单位是字节
  2. &数组名 - 数组名表示的是整个数组,取出的是整个数组的地址

二维数组的数组名

二维数组的数组名也是表示首元素的地址
二维数组的首元素的地址代表的是第一行的地址
C语言二维数组_第7张图片
C语言二维数组_第8张图片

你可能感兴趣的:(C语言,c语言)