数组
数组
数组是什么?
1.数组是数据的集合。
2.数组的数据是同类型的 不能是又有整数 又有小数等。
3.数组地址是连续的。
为什么要用?
需要大量数据同时也不关心名称的时候,比较方便。
对于数据个数比较多的时候,像我们想定义100个整型变量存放数据,你是不是我们直接定义 int a,b,c,d,e…到100个就好啦? 这显然是极其麻烦的并且傻瓜的一种做法。所以我们需要用到数组。
一维数组
如何定义?
类型 + 名字 + 多少个(多大)
例子:
创建一个能装100个数据的整型变量
整型 名字 个数
int array[100];
创建一个能装100个数据的浮点型变量
浮点型 名字 个数
float array[100];
如何初始化?
整型:
//定义了一个整型数组 个数5个 分别为1 2 3 4 5
int array[5] = {1,2,3,4,5};
//定义了一个整型数组 个数5个 部分初始化 分别为1 2 后面的默认没写是为 0 的
int array[5] = {1,2};
错误写法(越界):
int array{5} = {1,2,3,4,5,6,7,8}; //超出范围了 越界了 定义了5个 存放了8个
其他类型也一样。
浮点型:
//定义了一个浮点型数组 个数5个 分别为1.1 2.2 3.3 4.4 5.5
float array[5] = {1.1,2.2,3.3,4.4,5.5};
//定义了一个浮点型数组 个数5个 部分初始化 分别为1.1 后面的默认 为0.0
float array[5] = {1.1};
有时候你定义一个数组都得数一数元素个数,其实可以不用写多少个。
也可以这样定义:
int array[] = {1,2,3,4,5};
如何引用?
int a[3] = {1,2,3};
其实就是 a[0] = 1; a[1] = 2; a[3] = 2; 下标是从0开始算的。
我们可以通过下标的方式访问。
打印单个数组数据
比如我想打印第数组的第5个数据即 array[4]
printf("%d\n",array[4]);
printf("%d\n",array[2*2]);
打印整个数组数据
首先打印得知道数组数据个数。你可能会说定义的时候不就写上了吗?
那我想说 int array[] = {1,2,3,4,5}; 这样呢?难道又去数数?
我们可以通过数组的大小来确定数组数据个数。
int 类型是4字节。如果数组大小为8字节 那是不是就是数组个数为2。
像下面这样,比较推荐第一种来写。
length = sizeof(array)/sizeof(array[0]);
length = sizeof(array)/sizeof(int);
length = sizeof(array)/sizeof(4);
通过for循环来:
int array[5] = {1,2,3,4,5};
int array1[] = {1,2,3,4,5};
/*知道数组个数*/
for(i=0;i<5;i++)
{
printf("%d ",array[i]);
}
/*=============================================*/
length = sizeof(array1)/sizeof(array1[0]);
/*不知道数组个数*/
for(i=0;i<length;i++)
{
printf("%d ",array1[i]);
}
输出结果:1,2,3,4,5 (两个都是)
一维数组在内存中的存储
我们通过代码打印一下每个数据的地址吧
int array[] = {5,4,3,2,1};
int length; //计算数组个数
int i;
length = sizeof(array)/array[0];
for(i=0;i<length;i++)
{
printf("array[%d]地址:%p\n",i,&array[i]);
}
输出结果:
array[0]地址:0060FED4
array[1]地址:0060FED8
array[2]地址:0060FEDC
array[3]地址:0060FEE0
发现了地址是从低往高存放的,地址是连续的 一个占4字节。
一维数组运用举例
一、找最大值
#include <stdio.h>
#include <stdlib.h>
int main()
{
int i = 0;//遍历数组
int array[5]; //存放数据
int max = 0; //存放最大值
while(i != 5)
{
printf("请输入第 %d 个数\n",i+1);
scanf("%d",&array[i]);
i++;
}
printf("数据: ");
for(i=0;i<5;i++)
{
printf("%d ",array[i]);
}
printf("\n");
max = array[0]; //把第一项给max
/*找最大值*/
for(i=1;i<5;i++)
{
if(array[i] > max)
{
max = array[i];
}
}
printf("最大值为%d\n",max);
system("pause");
return 0;
}
运行结果:
二、冒泡法排序(从小到大)
就是每次将相邻的两个数对比,把小的放到前面,如果有5个数:5 4 3 2 1。首先先5跟4对比 4最小 4放到前面 变成 4 5 3 2 1。然后继续5跟3对比,3最小,3放在前面。变成 4 3 5 2 1…直接看图吧。
我们会发现一个规律,总共进行了4轮(数组个数-1),每一轮都在减一。
那么通过for循环嵌套,如果i 代表是大的循环(4轮)j 表示里面互换次数的循环。
大循环好办 就是数组个数-1。 你5个 就for循环就是 4。
for(i=0;i<length-1;i++)
{
}
那么内循环 j 我们看图发现 第一轮进行了4次对比,正好是数组个数-1。第二轮进行了3次,第三轮进行了2次,第四轮进行了1次。
就是要j 第一次循环4次 第二次 循环3次 第三次 循环2次 第四次 1次。
刚好就是数组个数-i 就是大循环的次数-i
for(j=0;j<(length-1)-i;j++)
{
我们得知数组个数为5 5-1=4.
/* 第一轮i=0; 当j=0; 4-i = 4 可以进行4次循环
第二轮i=1; 当j=1; 4-i = 3 可以进行3次循环
第三轮i=2; 当j=2; 4-i = 2 可以进行2次循环
第四轮i=3; 当j=3; 4-i = 1 可以进行1次循环 */
}
完整程序:
#include <stdio.h>
int main()
{
int array[] = {5,4,3,2,1};
int length; //计算数组个数
int i,j;
int max;
length = sizeof(array)/sizeof(array[0]); //计算出数组个数
/*开始排序*/
for(i=0;i<length-1;i++)
{
for(j=0; j<(length-1)-i; j++)
{
if(array[j] > array[j+1]) //判断哪个大
{
max = array[j]; //大的给max
array[j] = array[j+1]; //把小的放在前面。
array[j+1] = max; //大的值放回后面
}
}
}
/*排序结果输出*/
printf("从小到大:");
for(i=0;i<length;i++)
{
printf("%d ",array[i]);
}
system("pause");
return 0;
}
输出结果:1 2 3 4 5
二维数组
如何定义?
二维数组就相当于一个矩阵,有行有列。比如下面定义的就是两行三列。
int array[2][3];
如何初始化?
类型 + 名字 + [行]][列];
注意:二维数组变量定义的时候 必须要有列的大小 可以没有行
正确的定义并初始化:
int array1[2][3] = {{1,2,3},{4,5,6}};
int array2[2][3] = {1,2,3,4,5,6};
int array3[][3] = {{1,2,3},{4,5,6}};
int array4[][3] = {1,2,3,4,5,6};
错误定义:
int array[][] = {{1,2,3},{4,5,6}}; //至少也要有列
int array[2][] = {{1,2,3},{4,5,6}}; //没有列
int array[2][3] {{1,2,3},{4,5,6},{7,8,9}};//越界 超出范围
如何引用?
比如:
int array[2][3] = {{1,2,3},{4,5,6}};
排列顺序:
array[0][0] —→ array[0][1] —→ array[0][2]
array[1][0] —→ array[1][1] —→ array[1][2]
下标是从0开始算的。
打印单个数据
比如我想打印第一行,第二个数据 即 2 array[0][1];
printf("%d\n",array[0][1]);
printf("%d\n",array[0*0][1*1]); //显得有点多余。。。
打印整个数组数据
首先得知道行数 和 列数:
sizeof(array[0][0]); 为一个元素占用的空间,
sizeof(array[0]); 为一行元素占用的空间,
sizeof(array); 为整个数组占用的空间,
行:line = sizeof(array)/sizeof(array[0]);
列:column = sizeof(array[0]/sizeof(array[0][0]))
通过for循环打印全部数组
int i,j;
int array[2][3] = {{1,2,3},{4,5,6}};
int size,line,column;
size = sizeof(array);//获取大小
line = sizeof(array)/sizeof(array[0]); //行
column = sizeof(array[0])/sizeof(array[0][0]); //列
printf("size:%d\n",size);
printf("line:%d\n",line);
printf("column:%d\n",column);
for(i=0;i<line;i++)
{
for(j=0;j<column;j++)
{
printf("array[%d][%d]:%d \n",i,j,array[i][j]);
}
}
输出结果:
size:24
line:2
column:3
array[0][0]:1
array[0][1]:2
array[0][2]:3
array[1][0]:4
array[1][1]:5
array[1][2]:6
二维数组在内存中的存储
二维数组是按行存放的,在内存中先放第一行的 然后 第二…
通过编程来查看地址:
#include <stdio.h>
int main()
{
int i,j;
int array[2][3] = {{1,2,3},{4,5,6}};
int size,line,column;
size = sizeof(array);//获取大小
line = sizeof(array)/sizeof(array[0]); //行
column = sizeof(array[0])/sizeof(array[0][0]); //列
printf("size:%d\n",size);
printf("line:%d\n",line);
printf("column:%d\n",column);
for(i=0;i<line;i++)
{
for(j=0;j<column;j++)
{
printf("array[%d][%d]地址:%p \n",i,j,&array[i][j]);
}
}
system("pause");
return 0;
}
输出结果:
size:24
line:2
column:3
array[0][0]地址:0060FEC4
array[0][1]地址:0060FEC8
array[0][2]地址:0060FECC
array[1][0]地址:0060FED0
array[1][1]地址:0060FED4
array[1][2]地址:0060FED8
跟一维数组一样。
二维数组运用举例
一、推箱子地图绘制
#include <stdio.h>
int main()
{
int i,j;
int Map[8][8] = {
0,0,1,1,1,0,0,0,
0,0,1,2,1,0,0,0,
0,0,1,0,1,1,1,1,
1,1,1,3,0,3,2,1,
1,2,0,3,4,1,1,1,
1,1,1,1,3,1,0,0,
0,0,0,1,2,1,0,0,
0,0,0,1,1,1,0,0,
};
/*地图显示*/
for(i=0;i<8;i++)
{
for(j=0;j<8;j++)
{
switch(Map[i][j])
{
case 0: printf(" "); break;
case 1: printf("▇"); break;//围墙
case 2: printf("★"); break; //目的地
case 3: printf("●"); break; //箱子
case 4: printf("♂"); break;//人
}
if((j+1)/8==1) //够8次 换行
{
putchar('\n');
}
}
}
return 0;
}
输出结果:
如果有的运行结果不像图中这样 把符号变成其他。改成*啥的
二、实现二维数组行列互换。
#include <stdio.h>
int main()
{
int array[2][3] = {{1,2,3},{4,5,6}};
int newArray[3][2];
int i,j;
/*原有数组输出数据*/
for(i=0;i<2;i++)
{
for(j=0;j<3;j++)
{
printf("array[%d][%d]: %d ",i,j,array[i][j]);
}
printf("\n");
}
/*变换行列*/
for(i=0;i<2;i++)
{
for(j=0;j<3;j++)
{
newArray[j][i] = array[i][j];
}
}
/*新数组输出数据*/
printf("\n=======变换行列结果========\n\n");
for(i=0;i<3;i++)
{
for(j=0;j<2;j++)
{
printf("newArray[%d][%d]: %d ",i,j,newArray[i][j]);
}
printf("\n");
}
return 0;
}
输出结果:
array[0][0]: 1 array[0][1]: 2 array[0][2]: 3
array[1][0]: 4 array[1][1]: 5 array[1][2]: 6
=======变换行列结果========
newArray[0][0]: 1 newArray[0][1]: 4
newArray[1][0]: 2 newArray[1][1]: 5
newArray[2][0]: 3 newArray[2][1]: 6
字符数组
如何定义?
字符很显然我是用char类型对吧。
定义一个char类型数组 数组个数是10个。
char string[10];
如何初始化?
定义一个字符数组 存放HelloWorld。
char string1[10] = {'H','e','l','l','o','W','o','r','l','d'}; //这种比较麻烦
char string2[10] = "HelloWorld"; //这种是比较简单 没那么麻烦
char string3[] = "HelloWorld"; //这种更简单 省略数组个数
如何引用?
1.通过访问下标的方式 一个一个字符的输出。——for循环 或者 while()循环
通过访问下标的方式:————string1字符数组
int i;
// i < string1整体大小 除以 string1首元素大小
for(i=0;i<sizeof(string1)/sizeof(string1[0]);i++)
{
printf("%c",string1[i]);
}
printf("\n"); //结束来个换行
输出结果:HelloWorld
2.通过%s输出字符串———string2 string3
//打印string2 内容:
printf("string2:%s \n",string2); string2 是指 字符串数组的首地址
//打印string3 内容:
printf("string3:%s \n",string3); string3 是指 字符串数组的首地址
输出结果:
输出结果:
string2:HelloWorld@
string3:HelloWorld
这里会发现 string2这个 打印出来字符串 在最后面有个乱码 这是为啥呢?
其实 在字符串中 字符串会默认以 ‘\0’ 这个符号 作为结束的标志。
解决办法就是把 string2[10] 改成 string2[11]
我们可以打印一下string3这个大小。就知道其实后面有没有多了个元素。
printf("string3的大小:%d\n",sizeof(string3));
输出结果:
11
我们也可以通过检测’\0’ 来输出字符串:
int i = 0;
while(string3[i] != '\0')
{
printf("%c",string3[i]);
i++;
}
输出结果:
HelloWorld
字符数组在内存中的存储
其实这个可以猜到 char 类型是占一个字节的 那么他们的存储是连续的一次是一个字节。
字符数组运用举例
字符串逆序
就是最前面的元素和最后面的元素互换 第二个元素和倒数第二个元素互换…
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main()
{
char string[] = "ABCDEF";
int number; //存放插入位置
int i;
char str; //存放新字符
int length;//获取字符串长度
int moveNumber; //获取移动的次数
printf("字符串为:%s\n",string);
printf("你想在第几个插入字符?\n");
scanf("%d",&number);
getchar(); //吸收回车键
printf("插入的字符为:\n");
scanf("%c",&str);
getchar();
printf("你想在%s 字符串 在第 %d 个插入字符 '%c' \n",string,number,str);
length = strlen(string); //获取字符串的长度
printf("字符长度:%d\n",length);
if(length == number)
{
string[length+1] = string[length];//为防止乱码 往右移一位 在Linux上 不需要加这行代码
string[length] = str; //插入数据
printf("==============================\n");
printf("插入后结果:%s\n",string);
printf("==============================\n");
return 0; //程序结束
}
moveNumber= length-number+1; //获取移动次数
printf("移动次数:%d\n",moveNumber);
//每一次循环移动一次
for(i=0;i<moveNumber;i++)
{
string[length-i] = string[length-1-i]; //往右移动
}
string[number-1] = str; //插入字符
printf("==============================\n");
printf("插入后结果:%s\n",string);
printf("==============================\n");
return 0;
}
运行程序结果:
字符串为:ABCDEF
你想在第几个插入字符?
2
插入的字符为:
G
你想在ABCDEF 字符串 在第 2 个插入字符 'G'
字符长度:6
移动次数:5
==============================
插入后结果:AGBCDEF
==============================
//注:如果最后有乱码
//请把string[] = "ABCDEF" 改成 string[8] = "ABCDEF" 或者数组个数更大。
/*
为什么移动的时候是以下的代码? (都加上了-i)
string[length-i] = string[length-1-i]; //往右移动
为什么不是下面这个?
string[length] = string[length-1];
因为在移动的时候 for循环中 这个是永远都是把最后一个元素往右边移动
*/
去掉字符串左边空格
#include <stdio.h>
#include <string.h>
int main()
{
char string[] = " hello"; //左边两个空格
int length = strlen(string); //获取字符串的长度
int i = 0;
int moveCount = 0; //记录需要移动的次数
printf("长度为:%d\n",length);
while(string[i] == ' ' && string[i] != '\0') //检测左边有多少个空格
{
i++;
moveCount++; //记录次数
}
printf("字符串中左边有 %d 个空格.后面的每个元素需要移动次数 %d 次\n",moveCount,moveCount);
/*也可以这么计算,增加一个标志位flag*/
/*
int flag = 1;
for(i=0;i<length;i++)
{
if(string[i] == ' ' && string[i] != '\0' && flag == 1)
{
number++;
}
else
{
flag=0; //一旦出现了不是空格 则不再计算左边空格数据
}
}
*/
/*循环次数:总共的字符串长度 - 左边的空格数*/
for(i=0;i<length-moveCount;i++)
{
string[i] = string[i + moveCount]; //将剩下的元素移动到前面
}
string[length-moveCount] = '\0'; //紧接着后面加个'\0'
printf("sting:%s\n",string); //输出结果
return 0;
}
输出结果:
长度为:7
字符串中左边有 2 个空格.后面的每个元素需要移动次数 2 次
sting:hello
去掉字符串右边空格
#include <stdio.h>
#include <stdlib.h>
int main()
{
char string[] = " hello "; //字符串
int i;
int number = 0; //计算字符串右边空格数量
int length = strlen(string); //计算字符串长度
printf("字符串长度为%d\n",length);
for(i=0;i<length;i++)
{
if(string[i] == ' ' && string[i] != '\0') //计算字符串右边的空格个数
{
number++;
}
else
{
number=0;
}
}
printf("右边空格有%d个\n",number);
string[length-number] = '#'; //为了能看出空格 在最后加个#号
string[length-number+1] = '\0';
printf("string:%s\n",string);
return 0;
}
输出结果:
字符串长度为20
右边空格有7个
string: hello#
去掉字符串左边的空格和右边的空格
这个就是比较简单了 把上面的结合一下就行了。
#include <stdio.h>
#include <stdlib.h>
int main()
{
char string[] = " he llo ";
int i;
int leftNumber = 0;
int rightNumber = 0;
int flag = 1; //左边空格标志位
int length = strlen(string);
printf("字符串长度为%d\n",length);
/*计算字符串左边的空格数*/
for(i=0;i<length;i++)
{
if(string[i] == ' ' && string[i] != '\0' && flag == 1)
{
leftNumber++;
}
else
{
flag=0; //一旦出现了不是空格 则不再计算左边空格数据
}
}
printf("左边空格有%d个\n",leftNumber);
/*计算字符串右边的空格数*/
for(i=0;i<length;i++)
{
if(string[i] == ' ' && string[i] != '\0')
{
rightNumber++;
}
else
{
rightNumber = 0;
}
}
printf("右边空格有%d个\n",rightNumber);
/*循环次数:字符串总长度 去掉左边的空格数 和 右边的空格数*/
for(i=0;i<length-leftNumber-rightNumber;i++)
{
string[i] = string[i + leftNumber]; //有效的字符串往左移动 (左空格数)
}
string[length-leftNumber-rightNumber] = '#'; //为了方便好看出结果 在最后加上 # 号
string[length-leftNumber-rightNumber+1] = '\0';
printf("删除字符串的左边空格 和 右边空格结果 string:%s\n",string);
return 0;
}
输出结果:
字符串长度为18
左边空格有5个
右边空格有7个
删除字符串的左边空格 和 右边空格结果 string:he llo#
关于函数传数组作为参数知识
记住一点:函数传参数 过去只能 数组的首地址。而不是整个数组。
#include <stdio.h>
void aaa(int array[5])
{
printf("aaa函数结果\n");
printf("%d\n",sizeof(array));//查看大小
printf("%d\n",sizeof(array)/sizeof(array[0]));//查看个数
}
void bbb(int array[])
{
printf("bbb函数结果\n");
printf("%d\n",sizeof(array));//查看大小
printf("%d\n",sizeof(array)/sizeof(array[0]));//查看个数
}
int main()
{
int array[5] = {1,2,3,4,5};
printf("main函数结果:\n");
printf("%d\n",sizeof(array));//查看大小
printf("%d\n",sizeof(array)/sizeof(array[0]));//查看个数
aaa(array); //数组名就是地址 是数组第一个元素地址
bbb(array);
system("pause");
return 0;
}
输出结果:
main函数结果:
20
5
aaa函数结果
4
1
bbb函数结果
4
1
在Linux底下aaa,或bbb函数 sizeof(array)是8个字节。