C语言:指针进阶(1)

文章详细介绍了C语言中的指针概念,包括字符指针、指针数组和数组指针的定义、使用及区别。字符指针可以指向单个字符或字符串,数组指针则是一个指向数组的指针。文章通过示例代码阐述了指针数组和数组指针的声明与使用,特别是数组指针在多维数组中的应用,并指出数组指针在处理多维数组时的便利性。
摘要由CSDN通过智能技术生成

 

目录

1. 字符指针

2. 指针数组

3. 数组指针

3.1 数组指针的定义

3.2 &数组名VS数组名

3.3 数组指针的使用


指针的主题,我们在初级已经接触过了,

我们知道了指针的概念:

1. 指针就是个变量,用来存放地址,地址唯一标识一块内存空间。

2. 指针的大小是固定的4/8个字节(32位平台/64位平台)。

3. 指针是有类型,指针的类型决定了指针的+-整数的步长,指针解引用操作的时候的权限。

4. 指针的运算。

1. 字符指针

在指针的类型中我们知道有一种指针类型为字符指针 char* ;

#include<stdio.h>

int main()
{
	char ch = 'w';
	char* pc = &ch;
	*pc = 'a';
	printf("%c\n", ch);
	return 0;
}

还有一种使用方式如下:

int main()
{
	char* p = "abcdef";//这里是把一个字符串放到pstr指针变量里了吗?
	//
	//p是指针变量,在x86环境下是4给字节,而这里字符串字节数有6个
	//
	//把"abcdef"字符串首字符的地址放入指针变量
	printf("%c\n", *p);//a
	printf("%s\n", p);
	//打印字符串,只需要字符串的首地址,依次向后打印,直到遇见'\0'
	return 0;
}
int main()
{
	const char* p = "abcdef";
	//这里是常量字符串,常量不能被改变,因此加上const修饰,拒绝访问"abcdef"
	
	printf("%c\n", *p);//a
	printf("%s\n", p);
	
	return 0;
}
#include <stdio.h>
int main()
{
    char str1[] = "hello.";
    char str2[] = "hello.";
    const char* str3 = "hello.";
    const char* str4 = "hello.";
    if (str1 == str2)
        printf("str1 and str2 are same\n");
    else
        printf("str1 and str2 are not same\n");

    if (str3 == str4)
        printf("str3 and str4 are same\n");
    else
        printf("str3 and str4 are not same\n");

    return 0;
}

 这里str3和str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域,当 几个指针。指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化 不同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4相同。

2. 指针数组

指针数组是一个存放指针的数组。

下面指针数组是什么意思?

int* arr1[10]; //整形指针的数组

char *arr2[4]; //一级字符指针的数组

char **arr3[5];//二级字符指针的数组

//指针数组
int main()
{
	//int arr[10];//存放整形的数组
	//int* arr1[10];//存放整型指针的数组
	//char* ch[5];//存放字符指针的数组

	int a = 10;
	int b = 20;
	int c = 30;
	int* pa = &a;
	int* pb = &b;
	int* pc = &c;

	int* arr[3] = { &a,&b,&c };//arr就是一个指针数组
	int i = 0;
	for (i = 0; i < 3; i++)
	{
		printf("%d ", *(arr[i]));
	}
	return 0;
}
int main()
{
	int arr1[5] = { 1,2,3,4,5 };
	int arr2[5] = { 5,6,7,8,9 };
	int arr3[5] = { 3,4,5,6,7 };

	int* parr[3] = { arr1,arr2,arr3 };//指针数组
	//    相当于模拟了一个二维数组
	int i = 0;
	for (i = 0; i < 3; i++)
	{
		int j = 0;
		for (j = 0; j < 5; j++)
		{
			//printf("%d ", parr[i][j]);
			printf("%d ", *(parr[i]+j));
		}
		printf("\n");
	}
	return 0;
}

 

3. 数组指针

3.1 数组指针的定义

数组指针是指针?还是数组?

答案是:指针。

我们已经熟悉: 整形指针: int * pint; 能够指向整形数据的指针。

浮点型指针: float * pf; 能够指向浮点型数据的指针。

那数组指针应该是:能够指向数组的指针。

下面代码哪个是数组指针?

int *p1[10];

int (*p2)[10];

//p1, p2分别是什么?

int main()
{
	int a = 10;
	int* p = &a;//整形指针_是存放整型变量地址的,指向整型的指针
	char ch = 'w';
	char* pc = &ch;//字符指针_是存放字符变量地址的,指向字符的指针

	
	int* p1[10];//p1是数组名,10个元素,存放指针,指针数组

	//数组指针_指向数组的指针
	int(*p2)[10];// (*p2) p2现在是个指针_[10]说明指向数组———每个元素是int
	

	return 0;
}

解释:

int (*p)[10];

//解释:p先和*结合,说明p是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个 指针,指向一个数组,叫数组指针。

//这里要注意:[]的优先级要高于*号的,所以必须加上()来保证p先和*结合。

int main()
{
	int a = 10;
	int* p = &a;

	int arr[10] = { 0 };
	//数组名是首元素的值
	//怎么拿到这整个数组de地址?
	&arr;
	printf("%p\n", arr); //00DCFE98————类型是int*
	printf("%p\n", arr+1);//00DCFE9C____相差了4个字节

	printf("%p\n", &arr[0]);//00DCFE98————类型是int*
	printf("%p\n", &arr[0]+1);//00DCFE9C____相差了4个字节

	printf("%p\n", &arr);00DCFE98
	printf("%p\n", &arr + 1); //00DCFEC0____相差了40个字节
	//由此可知&arr的类型___int (*)[10] ————————数组指针
	int(*p)[10] = &arr;

	/*char* arr[5];
	char* (*p)[5] = &arr;*/

	return 0;
}

3.2 &数组名VS数组名

对于下面的数组:

int arr[10];

arr 和 &arr 分别是啥?

我们知道arr是数组名,数组名表示数组首元素的地址。

那&arr数组名到底是啥?

//
//数组名该怎么理解呢
//
//通常情况下,我们说的数组名都是数组首元素地址
//但是有两个例外
//sizeof(数组名),这里的数组名表示整个数组,这里计算的是整个数组的大小
//&数组名,这里的数组名表示整个数组,取出的是整个数组的地址
//
//
//


//数组指针到底有什么用?
//

3.3 数组指针的使用

那数组指针是怎么使用的呢?

既然数组指针指向的是数组,那数组指针中存放的应该是数组的地址。

//数组指针到底有什么用?
//
void print1(int arr[], int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");
}
//
void print2(int* arr, int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d ", *(arr + i));
	}
	printf("\n");
}

//这不是推荐的的写法
void print3(int (*p)[10],int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		//*p相当于数组名,数组名又是首元素的地址,所以*p就是&arr[0]
		printf("%d ", *(*p + i));
	}
	printf("\n");
}
int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
	//写一个函数,打印arr数组的内容

	//这时形参是数组,实参也是数组的形式
	int sz = sizeof(arr) / sizeof(arr[0]);
	print1(arr, sz);

	//这里是形参写成指针的形式
	print2(arr, sz);

	//这里形参是数组指针的形式
	print3(&arr,sz);
	int(*p)[10] = &arr;

	printf("%d\n",  (*p)[0]);
	return 0;
}
void print1(int arr[3][5], int row, int col)
{
	int i = 0;
	for (i = 0; i < row; i++)
	{
		int j = 0;
		for (j = 0; j < col; j++)
		{
			//printf("%d ", arr+j);
			printf("%d ", arr[i][j]);
		}
		printf("\n");
	}
}
void print2(int (*p)[3][5])
{

	int i = 0;
	for (i = 0; i < 3; i++)
	{
		int j = 0;
		for (j = 0; j < 5; j++)
		{
			printf("%d ", (*p)[i][j]);
		}
		printf("\n");
	}
}
void print3(int(*p)[5],int row,int col)
{

	int i = 0;
	for (i = 0; i < 3; i++)
	{
		int j = 0;
		for (j = 0; j < 5; j++)
		{
			//printf("%d ", (*p)[j]);
			//p+i相当于数组指针指向第i行
			//*(p+i)相当于拿到第i行,也相当于第i行的数组名
			//数组名表示首元素地址,*(p + i)就是第i行第一个元素的地址
			
			//printf("%d ", (*(p + i)) [j]);
			printf("%d ", *(*(p + i) + j));
			//printf("%d ", p[i][j]);//编译器需要转换成*(*(p + i) + j)计算
		}
		printf("\n");
		//p++;
	}
}
int main()
{
	int arr[3][5] = { {1,2,3,4,5},{2,3,4,5,6},{3,4,5,6,7} };
	//写一个函数,打印arr数组
	int row = sizeof(arr) / sizeof(arr[0]);
	int col = sizeof(arr[0]) / sizeof(arr[0][0]);

	//形参是二维数组,实参也是二维数组的形式
	print1(arr,row,col);
	printf("\n");
	//形参是数组指针
	print2(&arr);
	printf("\n");
	//
	print3(arr,row,col);
	printf("\n");
	return 0;
}

 

总结

由上可见,数组指针一般不用于一维数组,多用于多维数组

学了指针数组和数组指针我们来一起回顾并看看下面代码的意思:

int arr[5];

int *parr1[10];

int (*parr2)[10];

int (*parr3[10])[5];

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值