C语言进阶学习之指针

目录
  • 1.指针概念回顾
  • 2.字符指针
  • 3.数组指针和指针数组
    • 3.1数组指针的含义
    • 3.2&数组名vs数组名
    • 3.3数组指针
  • 4.数组传参和指针传参
    • 4.1一维数组传参
    • 4.2二维数组传参
    • 4.3一级指针传参
    • 4.4二级指针传参
  • 5.函数指针
  • 6.函数指针数组
  • 7.指向函数指针数组的指针
  • 8.回调函数
  • 总结

1.指针概念回顾

指针的基本概念:

指针是一个变量,用来存放地址,地址唯一标识一块内存空间。指针的大小是固定的4/8个字节(32位平台/64位平台)。指针是有类型,指针的类型决定了指针的±整数的步长,指针解引用操作的时候的权限。不能对没有初始化的指针或者是空指针进行间接访问

2.字符指针

指针类型为 char* 的指针就是字符指针

字符指针指向的对象的类型为字符型

一般使用方法:

    char ch = 'a';
	char* p = &ch; //取ch的地址放到指针变量p中

另外一种常见使用方法:

char* p2 = "bcdef";

对于第二种使用方法我们需要注意的是

p2实际存放的只是字符串首字符的地址,即p2存放的是字符b的地址。

我们来看一道经典的面试题:

#include <stdio.h>
int main()
{
	char str1[] = "hello yang.";
	char str2[] = "hello yang.";
	char* str3 = "hello yang.";
	char* str4 = "hello yang.";
	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指向的都是字符串H的地址

当指针指向同一个字符串的时候,它们实际会指向同一块内存。但是用相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。

因此它的答案为

3.数组指针和指针数组

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

例如

int* arr1[10]; //整形指针的数组
char *arr2[4]; //一级字符指针的数组
char **arr3[5]; //二级字符指针的数组

3.1数组指针的含义

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

答案是:指针。

int *p1[10];
int (*p2)[10];
//p1, p2分别是什么?

这里就不得不谈论到优先级了,优先级从大到小的顺序为: “( )” > “[ ]” > “ * ”

首先看int *p1[10]; , 由于[ ]的优先级大于*,因此p1先与“[” 结合,故而p1首先它是一个数组,它是一个存放了10个指针变量的数组。

再来看int (*p2)[10];, 由于()的优先级大于 [ ], 先看()里的,p2与 * 结合,故而篇是一个指针,它指向了一个存放了10个整型的数组

3.2&数组名vs数组名

对于下面这个代码

#include <stdio.h>
int main()
{
    int arr[5] = {0};
    printf("%p\n", arr);
    printf("%p\n", &arr);
    return 0;
}

arr 和 &arr 有什么区别呢?

其运行结果如下:

如果你认为它们代表的含义是正确的,不妨试着运行下面这个代码

#include <stdio.h>
int main()
{
	int arr[5] = { 0 };
	printf("arr = %p\n", arr);
	printf("&arr= %p\n", &arr);
	printf("arr+1 = %p\n", arr + 1);
	printf("&arr+1= %p\n", &arr + 1);
	return 0;
}

我的编译器下运行结果如下

如果 arr 和 &arr 代表的含义一样,那么后面两个的结果应该也是一致的,但是却不一样。

实际上: &arr 表示的是整个数组的地址,而不是数组首元素的地址。

数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于 &arr 的差值是40

3.3数组指针

数组指针就是一个指向数组的指针,意味着,这个指针内存放的是数组的地址

#include <stdio.h>
int main()
{
    int arr[10] = {1,2,3,4,5,6,7,8,9,0};
    int (*p)[10] = &arr;//把数组arr的地址赋值给数组指针变量p
    return 0;
}

我们来使用下指向一维数组的数组指针

//使用函数打印数组内的内容
# include <stdio.h>
void print_arr(int(*p)[10], int sz) //传过来的是数组的地址,用指针来接收
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d ", (*p)[i]); //先解引用找到数组
	}
}
int main(void)
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,0 };
	int sz = sizeof(arr) / sizeof(arr[0]); //求出数组的元素个数
	print_arr(&arr, sz);
	return 0;
}

4.数组传参和指针传参

在写代码的过程中,为了更好地利用模块化的特点,我们会使用大量的函数。但是在将一个功能封装成函数的过程中,我们就不可避免的会传过去数组和指针,那么函数是如何接受的呢?

4.1一维数组传参

以下是常见的一维数组传参

#include <stdio.h>
void test(int arr[]) //直接用数组接收,空间大小可以省略
{}
void test(int arr[10]) //直接用数组接收,空间大小可以指明
{}
void test(int *arr) //接收数组首元素的地址
{}
void test2(int *arr[20]) //直接用数组接收
{}
void test2(int **arr) //传过来一级指针的地址,使用二级指针接收
{}
int main()
{
 int arr[10] = {0};
 int *arr2[20] = {0}; //存放了20个int*的数组
 test(arr); //传过去的是首元素的地址
 test2(arr2);
}

4.2二维数组传参

当我们在谈论二维数组首元素的时候指的是第一行!
数组名是首元素的地址,在二维数组中指的是第一行的地址

void test(int arr[3][5])//直接用二维数组接收
{}
void test(int arr[][])//错误的接收!列不能省略
{}
void test(int arr[][5])//行是可以省略的,但是列一定不能省略
{}
//总结:二维数组传参,函数形参的设计只能省略第一个[]的数字。
//因为对一个二维数组,可以不知道有多少行,但是必须知道一行多少元素。
//这样才方便运算。
void test(int *arr)//错误接收!传过来的是第一行一维数组的地址
{}
void test(int* arr[5])//错误接收!这是一个存放int*类型的数组,而数组内的元素是int
{}

//由于二维数组的首元素是第一行,每一行是一个一维数组
//因此可以写成指向一维数组的指针
void test(int (*arr)[5])
{}
void test(int **arr) //错误!
{}
int main()
{
 int arr[3][5] = {0};
 test(arr);
}

4.3一级指针传参

#include <stdio.h>
void print(int *p, int sz) {
 int i = 0;
 for(i=0; i<sz; i++)
 {
 printf("%d\n", *(p+i));
 }
}
int main()
{
 int arr[10] = {1,2,3,4,5,6,7,8,9};
 int *p = arr;
 int sz = sizeof(arr)/sizeof(arr[0]);
 //一级指针p,传给函数
 print(p, sz);
 return 0; }

4.4二级指针传参

#include <stdio.h>
void test(int** ptr)
{
	printf("num = %d\n", **ptr);
}
int main()
{
	int n = 10;
	int* p = &n;
	int** pp = &p;
	test(pp);
	test(&p);
	return 0;
}

5.函数指针

在之前的学习当中不会每天使用函数指针,但是请不要忘记它的存在。
什么是函数指针?顾名思义,函数指针就是指向一个函数的指针。既然函数指针是指针,那么它也具有指针的一些特性。比如:在对指针进行间接访问之前必须要先初始化

//代码一
int f(int);//f函数
int (*pf)(int) = &f;

这是函数初始化的一种方法。

//代码二
int f(int);
int (*pf)(int) = f;

代码一和代码二是等效的。我们用下面一个代码查看,发现其打印出来的地址是一样的。

在数组中,数组名是数组首元素的地址。而函数名则是函数的首地址。在函数名使用的时候我们所用的编译器会将函数名转化为一个函数指针,&操作符只是显示地说明了函数将隐式执行的任务。

如果你需要将函数的地址存储起来,这个时候你就可能会需要用到函数指针了

void test()
{
	printf("hehe\n");
}
//下面的哪一个语句有能力存放函数test的地址呢?
void (*pfun1)(); //语句1
void *pfun2(); //语句2

存储地址应该需要用到的是指针

语句1:

pfun1首先和 * 操作符结合,那么它是一个指针。后面有一个函数调用操作符,表明它是一个函数指针。它所指向的函数

返回值类型是void

语句2:

pfun2首先和()结合,那么它是一个函数。 *和前面的void结合,表明这个函数的返回值类型是void*

所以选择语句1。

6.函数指针数组

我们知道数组是存储一组相同类型的数据,在此之前我们再来回顾一下指针数组

int *arr[10];
//数组的每个元素是int*

这是一个数组,数组存放了10个int*类型的数据。

按照这个思路,我们把很多函数的地址存放在数组中,那么它就是一个函数指针数组。

下面我们一起来实现它

首先它是一个数组 [ ], 存放的数据类型为函数指针,int (*)()类型的是函数指针

int (*parr1[10])();
int *parr2[10]();
int (*)() parr3[10];

7.指向函数指针数组的指针

经过我们上面这么多的推导,这个也就很容易就能表示出来

指向函数指针数组的指针是一个 指针, 指针指向一个 数组 ,数组的元素都是 函数指针

void test(const char* str)
{
 printf("%s\n", str);
}
int main()
{
 //函数指针pfun
 void (*pfun)(const char*) = test;
 //函数指针的数组pfunArr
 void (*pfunArr[5])(const char* str);
 pfunArr[0] = test;
 //指向函数指针数组pfunArr的指针ppfunArr
 void (*(*ppfunArr)[10])(const char*) = &pfunArr;
 return 0;
}

8.回调函数

回调函数:回调函数就是将一个函数的函数指针作为参数传递给另一个函数,而这个函数就是回调函数。

回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

库函数中的qsort函数就是一个典型的回调函数

总结

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

(0)

相关推荐

  • C语言中的指针新手初阶指南

    目录 1.指针是什么 2.指针和指针类型 3.野指针 3.1野指针成因 3.2如何规避野指针 4.指针的运算 4.1指针±整数 4.2指针-指针 4.3指针的关系运算 5.指针和数组 6.二级指针 7.指针数组 总结 1.指针是什么 ​ 初学者都有一个疑问,那就是指针是什么?简单的说,就是通过它能找到以它为地址的内存单元. 地址指向了一个确定的内存空间,所以地址形象的被称为指针. int main() { int a = 10; int* pa = &a; return 0; } //pa是用来

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

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

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

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

  • 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语言中的初阶指针详解

    目录 1.指针是什么 2.指针和指针类型 3.野指针 3.1野指针成因 3.2如何规避野指针 4.指针的运算 4.1指针±整数 4.2指针-指针 4.3指针的关系运算 5.指针和数组 6.二级指针 7.指针数组 ​ 总结 1.指针是什么 ​ 初学者都有一个疑问,那就是指针是什么?简单的说,就是通过它能找到以它为地址的内存单元. 地址指向了一个确定的内存空间,所以地址形象的被称为指针. int main() { int a = 10; int* pa = &a; return 0; } //pa是

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

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

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

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

  • C语言进阶学习之指针

    目录 1.指针概念回顾 2.字符指针 3.数组指针和指针数组 3.1数组指针的含义 3.2&数组名vs数组名 3.3数组指针 4.数组传参和指针传参 4.1一维数组传参 4.2二维数组传参 4.3一级指针传参 4.4二级指针传参 5.函数指针 6.函数指针数组 7.指向函数指针数组的指针 8.回调函数 总结 1.指针概念回顾 指针的基本概念: 指针是一个变量,用来存放地址,地址唯一标识一块内存空间.指针的大小是固定的4/8个字节(32位平台/64位平台).指针是有类型,指针的类型决定了指针的±整

  • Go语言基础学习之指针详解

    目录 1. 什么是指针 2. 指针地址 & 指针类型 3. 指针取值 4. 空指针 5. make 6. new 7. make 和 new 的区别 8. 问题 今天来说说 Go 语言基础中的指针. Go 语言中指针是很容易学习的,Go 语言中使用指针可以更简单的执行一些任务. 1. 什么是指针 Go 语言中,一个指针变量指向了一个值的内存地址.和 C.C++ 中的指针不同,Go 语言中的指针不能进行计算和偏移操作. Go 语言中的函数传参都是值拷贝,当我们想要修改某个变量的时候,我们可以创建一

  • 深入学习C语言中的函数指针和左右法则

    通常的函数调用     一个通常的函数调用的例子: //自行包含头文件 void MyFun(int x); //此处的申明也可写成:void MyFun( int ); int main(int argc, char* argv[]) { MyFun(10); //这里是调用MyFun(10);函数 return 0; } void MyFun(int x) //这里定义一个MyFun函数 { printf("%d\n",x); } 这个MyFun函数是一个无返回值的函数,它并不完成

  • C语言进阶教程之函数指针详解

    目录 一.函数指针 1.概念 1.2函数指针的使用方法 1.3练习巩固 1.4小结一下 二.阅读两段有趣的代码 1.( *(void( *)( ))0 )( ) 2.void (* signal(int,void( * )( int ) ) )(int) 附:函数指针的应用——函数回调 总结 一.函数指针 1.概念 函数指针:首先它是一个指针,一个指向函数的指针,在内存空间中存放的是函数的地址: 请看示例: int main(){ int a = 10; int*pa = &a; char ch

  • Go语言学习之指针的用法详解

    目录 引言 一.定义结构体 1. 语法格式 2. 示例 二.访问结构体成员 三.结构体作为函数参数 四.结构体指针 总结 引言 Go 语言中数组可以存储同一类型的数据,但在结构体中我们可以为不同项定义不同的数据类型 结构体是由一系列具有相同类型或不同类型的数据构成的数据集合 结构体表示一项记录,比如保存图书馆的书籍记录,每本书有以下属性: Title :标题 Author : 作者 Subject:学科 ID:书籍ID 一.定义结构体 1. 语法格式 结构体定义需要使用 type 和 struc

  • C语言学习之指针知识总结

    目录 一.地址 二.指针与指针变量 三.指针的作用 四.初学指针时常见的错误 五.通过调用函数修改主调函数中的值 六.指针与一维数组 七.使用函数操作一维数组 八.指针变量所占字节数 九.静态数组的缺陷 十.malloc函数 十一.动态数组的构造 十二.静态内存与动态内存的对比 十三.多级指针 十四.跨函数使用内存 一.地址 内存中的最小单元是字节,一个字节对应一个编号,这里的编号就是对应字节的地址.换句话说,地址就是内存单元的编号. 二.指针与指针变量 指针与指针变量是两个不同的概念,指针是某

  • C语言学习之指针的使用详解

    目录 一.指针概念 1.指针变量 2.指针类型 3.二级指针 二.野指针 1.野指针成因 2.规避野指针 三.指针运算 1.指针±整数 2.指针-指针 3.指针关系运算 四.指针数组 1.指针和数组 2.指针数组的概念 五.字符指针 六.数组指针 七.数组传参和指针传参 1.一维数组传参 2.二维数组传参 3.一级指针传参 4.二级指针传参 八.函数指针 九.函数指针数组 十.回调函数 一.指针概念 在学习指针之前我们先要了解一下内存,内存是存储区域,我们可以把内存划分成一个一个的内存单元,最小

  • C语言进阶教程之循环语句缺陷详析

    目录 前言 1 循环语句的三要素 2 使用不同循环语句实现六种排列组合 2.1 第一种排列(ABC) 2.2 第二种排列(ACB) 2.3 第三种排列(BCA) 2.4 第四种排列(CBA) 2.5 第五种排列(BAC) 2.6 第六种排列(CAB) 3 什么时候用for循环语句 4 什么时候用while循环语句 5 什么时候用do-while循环语句 6 其他情况 7 总结 前言 你是否也有过下面的体会? 为什么刚开始学习C语言时很喜欢用for循环语句,但逐渐发现有经验的工程师都在用while

  • C语言进阶教程之字符串&内存函数

    目录 前言: 一.求字符串长度 strlen strlen函数的模拟实现 二.长度不受限制的字符串函数 strcpy strcpy函数的模拟实现 strcat strcat函数的模拟实现 strcmp strcmp函数的模拟实现 三.长度受限制的字符串函数 strncpy strncpy函数的模拟实现 strncat strncat函数的模拟实现 strncmp strncmp函数的模拟实现 四.字符串查找 strstr strstr函数的模拟实现 strtok strtok函数的模拟实现 五.

  • PHP进阶学习之反射基本概念与用法分析

    本文实例讲述了PHP进阶学习之反射基本概念与用法.分享给大家供大家参考,具体如下: 一.前言 Reflection(反射)是Java程序开发语言的特征之一,它允许运行中的Java程序对自身进行检查,或者说"自审",并能直接操作程序的内部属性.这一特征在实际应用中也许用得不是很多. PHP从5.0开始完美支持反射API.PHP反射可以用于观察并修改程序在运行时的行为.一个面向反射的(reflection-oriented)程序组件可以监测一个范围内的代码执行情况,可以根据期望的目标与此相

随机推荐