文章目录
1. 数组的创建
数组是一组相同类型元素的集合
创建形式是 :
type_t arr_name [const_n];
type_t 是指数组的元素类型
arr_name 数组的名称
const_n 是一个常量表达式,用来指定数组的大小
int arr1[10]; 整形数组
char ch[5]; 字符数组
注:数组创建,在C99标准之前
, [ ] 中要给一个常量才可以使用,但是不能使用变量
。在C99标准之后才可以是变量,为了支持变长数组。
变长数组不能初始化
变长数组的创建
char arr[ n ];
int mian()
{
char ch[] = {'a','b','c'}; //监视台显示a b c
char ch[] = "abc"; //监视台显示a b c \0
}
1.1 一维数组的使用
[ ] 就是下标引用操作符
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
printf("%d\n",arr[4]);
// 这就是打印第四个元素
// 但是 数组是从0开始的
// 第一个元素下标是0
// 最后一个是你指定大小-1
int sz = sizeof(arr)/sizeof(arr[0]);
//这个就是arr的字符大小除以第一个元素的大小
sizeof( ) 返回一个变量或者类型的大小(以字节为单位)
int main()
{
int arr[] = {1,2,3,4,5,6,7,8,9,10};
int sz = sizeof(arr) / sizeof(arr[0]);
//这个就是arr的字符大小除以第一个元素的大小
int i = 0;
//打印数组的每个元素的地址
for(i=0; i< sz; i++)
{
printf("&arr[%d] = %p\n",i,&arr[i]);
}
}
打印数组的每个元素的地址,每个地址4个字节
int main()
{
int arr[] = {1,2,3,4,5,6,7,8,9,10};
int sz = sizeof(arr) / sizeof(arr[0]);
//这个就是arr的字符大小除以第一个元素的大小
int i = 0;
//顺序打印每个元素
for(i = 0;i<=sz;i++)
{
printf("%d ",arr[i]);
}
//倒着打印每个元素
for(i = ze-1;i>=0;i--)
{
printf("%d ",arr[i]);
}
return 0;
}
2. 二维数组的创建
2.1
int arr[3][4]; 前一个行 后一个列
意思是三行四列
1 2 3 4
2 3 4 5
3 4 5 6
2.2 二维数组的初始化
int arr[3][4] = {1,2,3,4,2,3,4,5,3,4,5,6};
三行
一行四个元素
int arr[3][4] = {1,2,3,4,2,3,4,5,3,4,};
没写满所需的元素会用零补足
结果:
int arr[3][4] = {{1,2},{3,4},{5,6}};
三行四列分组
一行两个不足补0
int arr1[][] = {{1,2,3,4},{2,3}}; X
这样写会报错 因为二维数组必须要写零
int arr1[][4] = {{1,2,3,4},{2,3}}; √
2.2 二维数组的下标定位
1 2 3 4
2 3 4 5
3 4 5 6
代码:
int main()
{
int arr[3][4] = {1,2,3,4,2,3,4,5,3,4,5,6};
int i = 0;
prinf("%d\n",arr[2][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;
}
输入12个元素然后三行四列输出。
int main()
{
int arr[3][4];
int i = 0;
for(i = 0;i < 3;i++)
{
int j = 0;
for(j = 0; j < 4; j++)
{
scanf("%d",&arr[i][j]);
}
}
for(i = 0;i < 3;i++) 循环行
{
int j = 0;
for(j = 0; j < 4; j++) 循环列
{
printf("%d",arr[i][j]);
}
printf("\n");
}
return 0;
}
2.3 二维数组在内存中的存储
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("&arr[%d][%d] = %p\n", i, j, &arr[i][j]);
}
}
return 0;
}
二维数组在内存中也是连续存放的,但是它和一维数组访问形式不一样。
3. 数组越界
3.1 一维数组的越界
数组的下标是有范围限制的
数组的下规定是从0开始的,如果数组有n个元素,最后一个元素的下标就是n-1。
C语言本身是不做数组下标的越界检查,编译器也不一定报错,但是编译器不报错,并不意味着程序就
是正确的,
所以程序员写代码时,最好自己做越界的检查。
#include <stdio.h>
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int i = 0;
int sz = sizeof(arr)/sizeof(arr[0])
for(i = 0; i <= se; i++)
{ //数组下标从零开始 有n个元素,所以元素下表是n-1
printf("%d\n", arr[i]); //当i等于10的时候,越界访问了
}
return 0;
}
3.2 二维数组的越界
int main()
{
int arr[3][4] = { 1,2,3,4,5,6,7,8,9,10,11,12 };
int i = 0;
for (i = 0; i < 3; i++)
{
int j = 0;
for(j = 0;j <= 4; j++) //不要等于 因为是元素从零开始
{
printf("%d ", arr[i][j]);
}
}
return 0;
}
4. !!重点 !!数组名是什么
数组名表示首元素的地址
但是有2个例外:
- sizeof (数组名),这里的数组名表示整个数组,计算的是整个数组的大小,单位是字节
- &数组名,这里的数组名表示整个数组,取出的是整个数组的地址
除此1,2两种情况之外,所有的数组名都表示数组首元素的地址。
&arr+1会跳过整个数组 一维数组和二维数组同理
因为他取的是整个数组的地址
int main()
{
int arr[10];
printf("%p\n", arr);
printf("%p\n", arr+1);//arr就是首元素的地址
printf("--------------------\n");
printf("%p\n", &arr[0]);//首元素的地址
printf("%p\n", &arr[0]+1);
printf("--------------------\n");
printf("%p\n", &arr);//数组的地址
printf("%p\n", &arr+1);
int n = sizeof(arr);
printf("%d\n", n);
return 0;
}
4.1 !!!二维数组的数组名的理解 !!!
二维数组的数组名也表示首元素的地址
-
int arr[3][4];
-
printf("%p\n",arr);//二维数组的数组名也表示首元素的地址
-
printf("%p\n", arr+1);进入下一行 因为这是二维数组
-
sizeof(arr)计算的是数组大小
-
sizeof(arr[0])取得是一行数组大小
-
sizeof(arr[0][0])取的是一个元素的大小
-
&arr取的的整个地址
-
&arr+1会跳过整个数组 一维数组和二维数组同理
-
因为他取的是整个数组的地址
4.2 二维数组计算行列
int main()
{
int arr[3][4];
printf("%d\n", sizeof(arr) / sizeof(arr[0]));
printf("%d\n", sizeof(arr[0]) / sizeof(arr[0][0]));
return 0;
}
** 小结 **
二维数组可以看作一维数组。
以把二维数组理解为:一维数组的数组
行能省略 ,列不能省略
int arr[3][4];
展开就是下面这些
第一行一维数组的名字为 int arr[0][j]; 第一行的第j个元素
第二行一维数组的名字为 int arr[1][j]; 第二行的第j个元素
第三行一维数组的名字为 int arr[2][j]; 第三行的第j个元素
printf("%d\n",sizeof(arr))