一篇文章带你了解C语言指针进阶

目录
  • 1.字符指针
  • 2.指针数组
  • 3.数组指针
  • 4.函数指针
  • 5.数组传参
  • 总结

1.字符指针

我们已经知道了数组名在大部分时候表示数组的地址,指针本质上也表示一个地址,那么我们能否用指针来创建一个字符串呢?

int main()
{
	char arr1[] = "abcdef";
	char arr2[] = "abcdef";
	const char* p1 = "abcdef";
	const char* p2 = "abcdef";//const可省略,默认为常量字符串
	printf("%s\n", arr1);
	printf("%s\n", arr2);
	printf("%s\n", p1);
	printf("%s\n", p2);
	return 0;
}

需要注意的是字符指针创建的字符串是常量字符串,普通的字符串只要数组名不同即使字符串内容相同,也是不同的字符串,但是不论用多少个指针创建字符串,只要字符串内容相同,所有指针指向的就是同一个字符串,因此字符指针指向的字符串不能修改,在没有const修饰的情况下也是如此。

2.指针数组

我们知道数组可以存储整形,浮点型,或自定义的结构体,那么可不可以存储指针呢?答案是:可以。元素都为指针的数组就称为指针数组,形式为:数据类型* 数组名[ ],例如:储存整形指针的指针数组创建成int*p[10],表示的就是一个包含十个指针元素的数组,利用指针数组我们可以访问指针数组内的指针进而操作变量或其他数组。

int main()//利用指针数组分别遍历三个数组
{
	int arr1[] = { 1, 2, 3, 4, 5 };
	int arr2[] = { 2, 3, 4, 5, 6 };
	int arr3[] = { 3, 4, 5, 6, 7 };
	int*parr[] = { arr1, arr2, arr3 };
	int i, j;
	int Sz = sizeof(parr) / sizeof(parr[0]);
	int sz = sizeof(arr1) / sizeof(arr1[0]);
	for (i = 0; i < Sz; i++)
	{
		for (j = 0; j < sz; j++)
		{
			printf("%d ",*(parr[i]+j) );
		}
		printf("\n");
	}
}

3.数组指针

我们知道指针可以指向整形,浮点型,或自定义的结构体,那么能不能指向数组呢?答案也是:可以。指向数组的指针称为数组指针,形式为:数据类型(*p)[ ] ,例如int(*p)[10],表示的就是一个指向包含十个元素数组的指针,这和指针数组的区别在于要将*p用小括号括起来以确保它是一个指针而非数组(无括号的情况下p先和 [ ]结合)。

int main()利用数组指针遍历数组
{
	int arr[6] = { 1, 2, 3, 4, 5, 6 };
	int(*p)[6] = &arr;
	int i;
	int sz = sizeof(arr) / sizeof(arr[0]);
	for (i=0;i<sz;i++)
	printf("%d", (*p)[i]);
	//int*arr[5];
	//int*(*p)[5]=&arr;//int(*p[10])[5]-->p[10]是一个指针数组,剩下int(*  )[5]的是数组的类型,类型为数组指针,p是一个数组有十个元素,每个元素都是一个数组指针,指向的数组有五个int类型的元素
	return 0;
}

有了指针数组和数组指针,我们还可以进行两者的嵌套使用,例如:数组指针数组(int(*p[ ])[ ]),指针数组指针(int*(*p)[ ]),前者是一个数组里面的元素是指向另外一些数组的指针,后者是个指针指向一个元素全为另外一些指针的数组。

void print(int(*p)[5],int x,int y)//用数组指针接收数组地址
{
	int i,j;
	for (i = 0; i < x; i++)
	{
		for (j = 0; j < y; j++)
		{
			printf("%d ", *(*(p + i) + j));//arr[i]==*(arr+i)==*(p+i)==p[i]
		}
		printf("\n");
	}
}
int main()//利用数组指针遍历二维数组
{
	int arr[3][5] = { { 1, 2, 3, 4, 5 }, { 2, 3, 4, 5, 6 }, { 3, 4, 5, 6, 7 } };
	print(arr, 3, 5);//arr是首元素地址,二维数组的首元素是个第一行的一维数组
	return 0;
}

4.函数指针

我们知道指针可以指向整形,浮点型,自定义的结构体和数组,那么能不能指向一个函数呢?答案同样是:可以。函数指针的形式为:函数返回类型(*p)(参数类型,参数类型....)例如:int(*p)(int,int)表示的就是一个指向返回类型为整形,两个参数类型也是整形的函数的指针。注意:利用函数指针调用函数时有几个解引用操作符都可以没有也可以,但是括号不可或缺!

int Mul(int x, int y)
{
	return x*y;
}
int main()
{
	int a, b;
	scanf("%d%d", &a, &b);
	int(*p)(int,int) = &Mul;//函数指针,有无&都可以
	printf("%d", (p)(a, b));//用指针调用函数(*(void(*)()0))()
	printf("%d", (*p)(a, b));
	printf("%d", (**p)(a, b));
	printf("%d", (***p)(a, b));//任意个数的*都可以
	return 0;
}

5.数组传参

我们知道简单的一维数组的传参方式:数组名是首元素地址,我们可以直接用指针接收也可以直接用数组接收,那么指针数组和二维数组呢?指针数组传参传的也是其首元素的地址,而它的首元素是一个指针,指针的地址我们当然用二级指针接收,同样地我们也可以直接指针数组接收。二维数组传参传的是其首元素的地址,要注意的是,二维数组的首元素并不是第一个元素而是第一行元素所构成的一维数组,一维数组的地址我们当然用数组指针接收,同样地,我们也可以直接用二维数组接收。

void test1(int arr[])//直接数组接收
{
	printf("%d\n", arr[]);
}
void test2(int*arr)//指针接收首元素地址
{
	printf("%d\n", arr[]);
}
void test3(int arr[][5])//二位数组直接接收
{
	printf("%d\n", arr[][5]);
}
void test4(int(*arr)[5])//数组指针接收
{
	printf("%d\n", arr);
}
void test5(int*arr[])//直接指针数组接收
{
	printf("%d\n", arr);
}
void test6(int**arr)//二级指针接收
{
	printf("%d\n", arr);
}
int main()
{
	int arr1[5] = {1,2,3,4,5};
	int arr2[3][5] = {1,2,3,4,5};
	int* arr3[5] = {NULL};
	test1(arr1);
	test2(arr1);
	test3(arr2);
	test4(arr2);
	test5(arr3);
	test6(arr3);
	return 0;
}

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • C语言进阶:指针的进阶(1)

    目录 指针进阶 字符指针 字符指针的作用 字符指针的特点 指针数组 指针数组的定义 指针数组的使用 总结 指针进阶 我们在初阶时就已经接触过指针,了解了指针的相关内容,有: 指针定义:指针变量,用于存放地址.地址唯一对应一块内存空间. 指针大小:固定32位平台下占4个字节,64位8个字节. 指针类型:类型决定指针±整数的步长及指针解引用时访问的大小. 指针运算:指针解引用,指针±整数,指针-指针,指针关系运算. 本章节在此基础上,对C语言阶段指针进行更深层次的研究. 字符指针 字符指针,存入字符

  • C语言指针与引用的区别以及引用的三种用法案例详解

    1.指针与引用的区别: 指针是一块内存的地址值,而引用是一块内存的别名. 下面引自://www.jb51.net/article/221791.htm 从概念上讲.指针从本质上讲就是存放变量地址的一个变量,在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变. 而引用是一个别名,它在逻辑上不是独立的,它的存在具有依附性,所以引用必须在一开始就被初始化,而且其引用的对象在其整个生命周期中是不能被改变的(自始至终只能依附于同一个变量). 在C++中,指针和引用经

  • C语言进阶:指针的进阶(2)

    目录 数组指针 数组指针的定义 &数组名和数组名 数组指针的使用 反面用例 正面用例 Example 类型辨别方法 总结 数组指针 由前面的例子,不难得出,数组指针是指向数组的指针,是指针而非数组. 数组指针的定义 char ch = 'w'; char* pch = &ch;//字符地址存放在字符指针中 int a = 10; int* pint = &a;//整型地址存放在整型指针中 float f = 0.0; float* pf = &f;//浮点型地址存放在浮点型

  • C语言进阶:指针的进阶(4)

    目录 函数指针 函数指针的定义 函数指针的类型 函数指针的使用 Example 总结 函数指针 函数指针的定义 整型指针存放整型的地址:数组指针存放数组的地址:那么类比可得,函数指针存放函数的地址. 显然,函数指针指向函数,存放函数的地址.搞懂函数指针,先了解函数的地址. &函数名或函数名代表函数地址,与&数组名和数组名略有不同,&函数名和函数名完全一致. 函数的地址必然要放到函数指针里,函数指针的类型该如何写呢?(以Add函数为例) //整型指针 int* pa = &a

  • C语言进阶:指针的进阶(3)

    目录 数组传参和指针传参 一维数组传参 二维数组传参 一级指针传参 二级指针传参 总结 数组传参和指针传参 实践之中不免会碰到数组和指针作函数参数而如何设计形参的问题. 一维数组传参 一维数组传参,下列接收方式是否可行呢? //1. void test(int arr[]) {} //2. void test(int arr[10]) {} //3. void test(int* arr) {} int main() { int arr[10] = { 0 }; test(arr); retur

  • C语言进阶:指针的进阶(5)

    目录 函数指针数组 函数指针数组的定义 函数指针数组的使用 转移表 回调函数 指向函数指针数组的指针 总结 函数指针数组 //整型数组 - 存放整型变量 int arr[10]; //字符数组 - 存放字符变量 char ch[5]; //指针数组 - 存放指针变量 int* arr[10]; //函数指针数组 - 存放函数指针 int(*pfar[10])(int, int); 指针数组存放指针变量,函数指针数组存放函数指针,故元素类型为函数指针类型. 函数指针数组的定义 int Add(in

  • C语言指针笔试题全面解析

    目录 前言 一.指针笔试题 1.题目如图: 2.题目如图: 3.题目如图: 4.题目如图: 5.题目如图: 6.题目如图: 7.题目如图: 8.题目如图: 总结 前言 通过8道指针笔试题的解析,可以充分的复习到指针的相关知识,并且题目中会结合许多之前的相关知识,希望通过本篇文章,对大家所学的知识进行一个复习. 提示:以下是本篇文章正文内容,下面案例可供参考 一.指针笔试题 1.题目如图: 逐条语句分析: ①.定义了一个大小为5的整型数组,并进行了初始化 ②.定义了一个整型指针变量ptr用来存放地

  • 一篇文章带你了解C语言指针进阶

    目录 1.字符指针 2.指针数组 3.数组指针 4.函数指针 5.数组传参 总结 1.字符指针 我们已经知道了数组名在大部分时候表示数组的地址,指针本质上也表示一个地址,那么我们能否用指针来创建一个字符串呢? int main() { char arr1[] = "abcdef"; char arr2[] = "abcdef"; const char* p1 = "abcdef"; const char* p2 = "abcdef&qu

  • 一篇文章带你入门C语言:操作符

    目录 操作符 分类 算术操作符 移位操作符 整数存储规则 左右移位规则 赋值操作符 单目操作符 取地址操作符& 解引用操作符* 类型长度操作符sizeof 按位取反操作符~ ++ -- 操作符 强制类型转换操作符(type) 关系操作符= 逻辑操作符 短路运算 条件操作符 逗号表达式 下标引用.函数调用和结构成员 下标引用操作符[] 函数调用操作符() 结构成员操作符. -> 结构体定义 结构体使用 结构体地址 表达式求值 隐式类型转换 整型提升 如何整型提升 有符号数 无符号数 算术转换

  • 一篇文章带你了解C语言:入门基础(2)

    目录 操作符 算术操作符 移位操作符 位操作符 单目操作符 逻辑反操作! 操作符++,-- 逻辑操作符 条件操作符 逗号表达式 常见关键字 typedef extern static 修饰局部变量 修饰全局变量和函数 其它 #define定义常量和宏 定义常量 定义宏 指针 内存单元 指针变量 &取地址操作符,*解引用操作符 类型所占空间 结构体 定义结构体 使用结构体变量 总结 本节将结束对初识C语言的概述,只追求大概,不求精细. 本节包括的内容有操作符,常见关键字,#define定义常量和宏

  • 一篇文章带你了解C语言--数据的储存

    目录 前言 数据类型介绍 类型的基本归类 整形在内存中的存储 原码.反码.补码 大小端介绍 浮点型在内存中的存储 前言 前面我们学习了C语言的一些基本知识和基础的语法,想必大家对C语言都有了自己的认识. 当然只是学习这些知识还是不够的,我们需要进行更加深入的学习. 从本章开始,我们将进行C语言进阶阶段的学习,所以难度会有所增加. 数据类型介绍 前面我们已经学习了基本的内置类型: char //字符数据类型 short //短整型 int //整形 long //长整型 long long //更

  • 一篇文章带你入门C语言:函数

    目录 函数 定义 库函数 定义 介绍 Example 1 strcpy Example 2 memset 自定义函数 Example 1 Example 2 两数交换 链式访问 Example 1 函数声明 函数递归 Example 1 Example 2 函数迭代 Example 3 Example 4 总结 函数 定义 程序里的函数又被叫做子程序,他作为一个大型程序的部分代码,有一或多个语句项组成.函数负责完成某项特定任务,提供了对过程的封装和对细节的隐藏,这样的代码通常会被集成为软件库.

  • 一篇文章带你了解C语言浮点数之间的比较规则

    目录 你认为这段代码输出什么? 为什么不等于呢? 应该怎么解决? 那么怎么判断两个浮点数 f1 和 f2 相等呢. 伪代码 可以简化为 >> 怎么判断浮点数等于0? 还有一个问题 总结 你认为这段代码输出什么? int main() { float f1 = 1.1; float f2 = 2.2; if (f2 - 1.1 == f1) printf("等于"); else printf("不等于"); return 0; } 答案是不等于. 为什么不

  • 一篇文章带你了解C语言操作符

    目录 一.操作符分类 二.算术操作符 三.移位操作符 1.左移操作符 2.右移操作符 2.1算术移位 2.2逻辑移位 四.位操作符 1.按位与 2.按位或 3.按位异或 4.一道练习题 五.赋值操作符 1.赋值操作符(=)是一个很棒的操作符,他可以让你得到一个你之前不满意的值.也就是你可以对其重新赋值. 2.赋值操作符可以连续使用 3.复合赋值符 六.单目操作符 1.逻辑反操作 2.取地址 3.sizeof 4.++和--运算符 4.1前置++和-- 4.2后置++和-- 七.关系操作符 八.逻

  • 一篇文章带你了解C语言的一些重要字符串与内存函数

    目录 一.字符串函数 1. 求字符串长度的strlen 2.比较字符串大小的strcmp 3.复制字符串的strcpy 4.追加字符串的strcat 5.查找字符串函数的strstr 二.内存函数 1.复制 memcpy,memmove 2.比较 memcmp 总结 一.字符串函数 1. 求字符串长度的strlen size_t strlen ( const char * str ); 字符串以 '\0' 作为结束标志,strlen函数返回的是在字符串中 '\0' 前面出现的字符个数(不包含 '

  • 一篇文章带你实现C语言中常用库函数的模拟

    目录 前言 函数介绍 strlen(求字符串长度) strcpy(字符串拷贝) strcat(字符串追加) strcmp(字符串比较) strstr(找子字符串) memcpy(内存拷贝) memmove(内存移动) 总结 前言 C语言中对字符和字符串的处理很是频繁,但是C语言本身是没有字符串类型的,字符串通常放在常量字符串中或者字符数组中. 字符串常量适用于那些对它不做修改的字符串函数. 函数介绍 strlen(求字符串长度) size_t strlen ( const char * str

  • 一篇文章带你用C语言玩转结构体

    目录 前言 一.结构体的声明与定义 1.结构体的声明 2.结构成员的类型 3.结构体的定义 二.初始化结构体 三.访问结构体成员 四.结构体嵌套 五.结构体指针 六.结构体传参 总结 前言 C语言提供了不同的数据类型,比如说int.float.double.char等,不同的类型决定了一个变量在内存中应该占据的空间以及表现形式. 但是,当我们定义一个人的时候,人的不同属性就比较难用同一个数据类型来定义了,因为人的身高.年龄.体重等属性往往需要不同数据类型,在这个时候,我们便引入结构体这个概念.

随机推荐