C 练习实例77-指向指针的指针-二维数组

时间:2024-03-15 19:52:17

关于数组的一些操作

#include<stdio.h>
#include<stdio.h>
void fun(int b[],int length)
{
	for(int i=0;i<length;i++){
		printf("%d ",b[i]);
	}
	printf("\n");
	for(int i=0;i<length;i++){	//数组作为形参传递,传递的是指针,所以能够改变值
		b[i]++;
	}
}
int main()
{
	int a[5]={1,2,3,4,5};
//	int b[5]=a;  不合法,数组的本质是常量指针,不允许被修改
//	也就是说b指向了一片空间,这片空间是随机分配的,它一辈子只能指向那片空间
	fun(a,sizeof(a)/sizeof(int));
	for(int i=0;i<sizeof(a)/sizeof(int);i++){
		printf("%d ",a[i]);
	}
	printf("\n");
	return 0;
}

运行结果

1 2 3 4 5
2 3 4 5 6

指向指针的指针

代码 

#include<stdio.h>
int main()
{
	int a=100;
	int *p1=&a;
	int **p2=&p1;
	
	printf("&a=%p\n",&a);
	printf("p1=%p\n",p1);
	printf("*p2=%p\n",*p2);
	
	printf("a=%d\n",a);
	printf("*p1=%d\n",*p1);
	printf("**p2=%d\n",**p2);
	
	return 0;
}

 结果

&a=000000000065FE14
p1=000000000065FE14
*p2=000000000065FE14
a=100
*p1=100
**p2=100

二维数组

代码

#include<stdio.h>
int main()
{
//	要点:数组名可以当指针用,它的值就是首地址,它指向的就是数组首元素
	int b[3]={1,2,3};//一维数组
	printf("b=%p\n",b);//首地址
	printf("&b=%p\n",&b);//首地址
	printf("&b[0]=%p\n",&b[0]);//首地址
	printf("*b=%d\n",*b);//首地址可以取出首元素
	printf("b[0]=%d\n",b[0]);
	
	printf("\n");
	/*
	二维数组的每一行其实就是一个一维数组
	a[0]可以认为是第一行数组的数组名,也可以认为是指向第一行数组的指针
	a[0]的值实际上就是第一行数组的首地址
	以此类推,a[1]是第二行,a[2]是第三行......
	*/
	
//	{a[0],a[1],a[2]}也可以看作是一个一维数组,a是它的名,也可以看作是指针
//	a的值就是就是一维数组a[0]的首地址
	int a[3][4]={{0,1,2,3},{4,5,6,7},{8,9,10,11}};//二维数组
	printf("a[0]=%p\n",a[0]);//第一行数组的首地址,也就是首地址
	printf("&a[0]=%p\n",&a[0]);//第一行数组的首地址,也就是首地址
	printf("*a=%p\n",*a);//第一行数组的首地址,也就是首地址
	printf("a=%p\n",a);//第一行数组的首地址,也就是首地址
	printf("&a=%p\n",&a);//第一行数组的首地址,也就是首地址
	printf("&a[0][0]=%p\n",&a[0][0]);//第一行数组的首地址
	printf("a[0][0]=%d\n",a[0][0]);//首元素
	printf("*a[0]=%d\n",*a[0]);//首地址可以取出首元素
	printf("**a=%d\n",**a);//首地址的首地址可以取出首元素-指针的指针
		
	return 0;
}

运行结果

b=000000000065FE14
&b=000000000065FE14
&b[0]=000000000065FE14
*b=1
b[0]=1

a[0]=000000000065FDE0
&a[0]=000000000065FDE0
*a=000000000065FDE0
a=000000000065FDE0
&a=000000000065FDE0
&a[0][0]=000000000065FDE0
a[0][0]=0
*a[0]=0
**a=0

在一维数组中,a的值=&a的值=&a[0]的值,*a=a[0]

在二维数组中,&a=a=*a=a[0]=&a[0]=&a[0][0],a[0][0]=*a[0]=**a[0],有点绕哈,是不是和指向指针的指针有点像。

数组指针(指向数组的指针)

它的定义如下:

#include <stdio.h>
int main()
{
	int a[3]={1,2,3};
	int *p=a;
	printf("*p+0=%d\n",*p+0);
	printf("*p+1=%d\n",*p+1);
	printf("*p+2=%d\n",*p+2);
}

输出结果

*p+0=1
*p+1=2
*p+2=3

数组指针,就是指向数组的指针,它存放的是数组的首地址,每加1其实是加了一个int类型的字节长度。

指针数组

指针数组,就是数组的每个元素是指针类型的,存放的是地址值。

#include <stdio.h>
int main()
{
	int a=1;
	int b=2;
	int c=3;
	printf("%d,%d,%d\n",&a,&b,&c);
	int *array[3]={&a,&b,&c};
//int *->int型指针,array[3]->数组有3个元素,每个元素是int *型,即指向整形变量的指针类型
	printf("%d,%d,%d\n",array[0],array[1],array[2]);
	printf("%d,%d,%d\n",*array[0],*array[1],*array[2]);//*为取值运算符
	
	printf("%d\n",&array[0]);//(数组也有它自己的地址)
	printf("%d\n",array);//&array[0]=array=&array
	printf("%d\n",&array);//&array[0]=array=&array
	
//	int *p=array;××因为array是int *型的,所以我们要定义int *型的指针,才能指向这个数组
	int **p=array;//拆成3部分看容易理解,int *看作类型,*表示是int *类型的指针,p代表变量名
	//即指向指针的指针
	printf("%d,%d,%d\n",&array[0],&array[1],&array[2]);//(数组也有它自己的地址)
	printf("%d,%d,%d\n",p,p+1,p+2);//(通过指针拿到数组单元的地址)
	printf("%d,%d,%d\n",*(p+0),*(p+1),*(p+2));//(通过取值运算符*拿到数组单元的值)
	printf("%d,%d,%d\n",**(p+0),**(p+1),**(p+2));//(通过取值运算符*拿到数组单元的地址所指向的值)
	
//在自己动手实验的过程中,我发现变量的内存分配好像是自顶向下,是递减规律
//而数组单元间的地址值是递增规律
	
}

运行结果 

6684180,6684176,6684172
6684180,6684176,6684172
1,2,3
6684144
6684144
6684144
6684144,6684152,6684160
6684144,6684152,6684160
6684180,6684176,6684172
1,2,3

二维数组续

 

#include <stdio.h>
int main()
{
	int a[3][4]={{0,1,2,3},{4,5,6,7},{8,9,10,11}};
	int *p0=a[0];//第一层首元地址
	int *p1=a[1];//第二层首元地址
	int *p2=a[2];//第三层首元地址
	printf("*p0=%d\n",*p0);
	printf("*p1=%d\n",*p1);
	printf("*p2=%d\n",*p2);
	printf("*a[0]=%d\n",*a[0]);
	printf("*a[1]=%d\n",*a[1]);
	printf("*a[2]=%d\n",*a[2]);
	int (*p)[4]=a;//p指向int[4],即指向4个int类型的连续空间,4*4=16Byte
	printf("a[0]=%d\n",a[0]);
	printf("a[1]=%d\n",a[1]);
	printf("a[2]=%d\n",a[2]);
	printf("\n");
	printf("p0=%d\n",p0);
	printf("p1=%d\n",p1);
	printf("p2=%d\n",p2);
	printf("\n");
	printf("p+0=%d\n",p+0);//二维数组第一层
	printf("p+1=%d\n",p+1);//+1不是简单的加1,而是加到下一个16Byte的空间  第二层
	printf("p+2=%d\n",p+2);//二维数组第三层
	printf("\n");
	
	printf("*(*(p+0)+0)=%d\n",*(*(p+0)+0));
	printf("*(*(p+0)+0)=%d\n",*(*(p+1)+0));
	printf("*(*(p+0)+0)=%d\n",*(*(p+2)+0));
	printf("\n");
	
//	因此遍历二维数组就有了两种方式
//	方式一
	for(int i=0;i<3;i++){	//最外层控制行
		for(int j=0;j<4;j++){	//内层控制列
			printf("%d ",a[i][j]);
		}
		printf("\n");
	}
	printf("\n");
//	方式二
	for(int i=0;i<3;i++){	//最外层控制行
		for(int j=0;j<4;j++){	//内层控制列
			printf("%d ",*(*(p+i)+j));
		}
		printf("\n");
	}
//	总结:c语言里的二维数组是连续空间,所以可以进行上述操作
	return 0;
}

运行 

*a[0]=0
*a[1]=4
*a[2]=8
a[0]=6684096
a[1]=6684112
a[2]=6684128

p0=6684096
p1=6684112
p2=6684128

p+0=6684096
p+1=6684112
p+2=6684128

*(*(p+0)+0)=0
*(*(p+0)+0)=4
*(*(p+0)+0)=8

0 1 2 3
4 5 6 7
8 9 10 11

0 1 2 3
4 5 6 7
8 9 10 11

指针+字符串

#include <stdio.h>
int main()
{	
	//好好学习,天天向上。
	char *str1="好好学";
	char *str2="习,天";
	char *str3="天向上";
	char *str[3]={str1,str2,str3};
	printf("%s%s%s",str[0],str[1],str[2]);;
	return 0;
}

运行结果

好好学习,天天向上