进阶指针(一)

news2024/9/27 9:23:41

图片来源于网络

✨博客主页:小钱编程成长记
🎈博客专栏:进阶C语言

进阶指针(一)

  • 0.回顾初阶指针
  • 1.字符指针
    • 1.1 相关面试题
  • 2.数组指针
  • 3.指针数组
    • 3.1 数组指针的定义
    • 3.2 &数组名VS数组名
    • 3.3 数组指针的使用
  • 4.数组传参和指针传参
    • 4.1 一维数组传参
    • 4.2 二维数组传参
    • 4.3 一级指针传参
    • 4.4 二级指针传参
  • 5.函数指针
    • 5.1 两段有趣的代码
  • 总结

0.回顾初阶指针

  1. 指针就是个变量,用来存放地址,地址唯一标识一块内存空间。(内存单元数有编号的,编号=地址=指针)
  2. 指针的大小是固定的4/8个字节(32位平台/64位平台)。
  3. 指针是有类型的,指针的类型决定了指针的±整数的步长,指针解引用操作的时候的权限。
  4. 指针的运算。

1.字符指针

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

它有两种使用方式:

  1. 指向字符
  2. 指向字符串(实际上指向的是首字符,但因为字符串中的字符都是连续的,所以也可以说是指向字符串)

指针指向字符:

int main()
{
  char ch = 'w';
  char *pc = &ch;
  *pc = 'w';
  return 0;
}

指针指向字符串:

int main()
{
  const char* pstr = "abcdef";//这里是把一个字符串放到pstr指针变量里了吗?
  printf("%s\n", pstr);
  return 0;
}

在这里插入图片描述

  • const char* pstr = “abcdef”;
    这里不是把一个字符串放到pstr指针变量里,而是将字符串的首字符的地址放到了pstr里。因为当字符串作为一个表达式时,结果是首字符的地址。

  • const *pstr = “abcdef” 和 char arr[] = “abcdef” 在内存中存储的都是abcdef\0 ;

  • 因为"abcdef” 和 arr 表示的都是字符串的首字符地址,所以我们可以将常量字符串想象成数组名,“abcdef” == arr。

    如下所示:
    在这里插入图片描述
    有些朋友可能会发现,为什么指针指向常量字符串时前面要加上const?比如:const char* pstr = “abcdef”;
    原因是:常量字符串在内存中不能被修改,若修改会出现写入错误(这个错误很难被及时发现)。如下所示:
    在这里插入图片描述
    用const修饰指针变量,使其变成常变量,不能被修改,即使不小心修改了,也能在编译期间及时发现错误。如下所示:在这里插入图片描述

1.1 相关面试题

//在《剑指offer》一书中有这样一道题
#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;
}

结果为:
在这里插入图片描述

为什么呢?
  1. 把常量字符串放到字符数组中,字符数组中存放的是字符。
    每个数组创建时在内存中开辟的空间并不同,所以用相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。(几个 数组的内容可能会相同,但每个数组在内存中的地址一定不同)
    所以str1和str2代表的首字符地址不同。
  2. C/C++会把常量字符串存储到一个单独的内存区域。因为常量字符串不能被修改,没必要保存多份,所以在内存中只存储一份。
    当几个指针指向同一个字符串时,实际上它们指向的也是同一个地址。

2.数组指针

在初阶指针中我们也学了指针数组。指针数组是一个存放指针的数组,存放在数组中的元素都是指针类型的。
我们来复习一下下面的指针数组是什么意思?

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

指针数组的使用场景举例:

//模拟二维数组
#include <stdio.h>
int main()
{
	int arr1[] = { 1,2,3,4,5 };
	int arr2[] = { 2,3,4,5,6 };
	int arr3[] = { 3,4,5,6,7 };
	int* arr[] = { arr1, arr2, arr3 };
	int i = 0;
	for (i = 0; i < 3; i++)
	{
		int j = 0;
		for (j = 0; j < 5; j++)
		{
			printf("%d ", arr[i][j]);
		}
		printf("\n");
	}
	return 0;
}

//一次性打印多个字符串
#include <stdio.h>
int main()
{
	char* arr[3] = { "hello", "hello", "C++" };
	int i = 0;
	for (i = 0; i < 3; i++)
	{
		printf("%s ", arr[i]);
	}
	return 0;
}

3.指针数组

3.1 数组指针的定义

数组指针是指针?还是数组?
答案是:指针。
我们已经知道:
整型指针: int * pint; 能够指向整型数据的指针。
浮点型指针: float * pf; 能够指向浮点型数据的指针。
那数组指针应该是:能够指向数组的指针。

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

int *p1[10];//p1是指针数组名
int (*p2)[10];//p2是数组指针

解释:

int (*p)[10];
//解释:p先和*结合,说明p是一个指针变量,然后指向的是一个大小为10个整型的数组。所以p是一个指针,指向一个数组,叫数组指针。
//注意:必须明确指针指向的数组有几个元素,不写时,默认为0,与真实数组的元素个数不同,会出错。
//指向的数组的元素个数不同,数组指针的类型也不同。

3.2 &数组名VS数组名

arr和&arr分别是什么?

我们知道arr是数组名,数组名表示的是数组的首元素地址。
那&arr数组名是什么呢?
我们先来看一段代码和运行结果:
在这里插入图片描述
数组名和&数组名打印的地址是一样的。
难道两个是一样的吗?
我们再来看一段代码和运行结果:
在这里插入图片描述

根据上面的代码我们发现,其实&arr和arr,虽然值是一样的,但是意义是不一样的。
实际上: &arr 表示的是数组的地址,而不是数组首元素的地址。
本例中 &arr 的类型是: int(*)[10] ,是一种数组指针类型, 数组的地址+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;
}

小提示:

printf("%d\n", (*p)[i]);//(*p) == *(&arr) == arr
printf("%d\n", p[i]);//p[i] == *(p+i) 因为p = &arr,所以p+i等于跳过了i个数组

数组指针主要应用于二维数组的传参:

#include <stdio.h>

void print_arr1(int(*arr)[5], int row, int col)//arr是数组指针
{
    int i = 0;
    for (i = 0; i < row; i++)
    {
        int j = 0;
        for (j = 0; j < col; j++)
        {
            printf("%d ", arr[i][j]);
            //arr[i][j] == (*(arr+i))[j], 
            //arr+i相当于二维数组的第i行的一维数组的地址,
            //*(arr+i)相当于二维数组的第i行的一维数组的首元素地址。
        }
        printf("\n");
    }
}

int main()
{
    int arr[3][5] = { 1,2,3,4,5,6,7,8,9,10 };
    print_arr1(arr, 3, 5);
    //数组名arr,表示首元素的地址
    //但是二维数组的首元素是二维数组的第一行
    //所以这里传递的arr,其实相当于第一行的地址,是一维数组的地址
    //可以数组指针来接收

    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 };
	test(arr);
	test2(arr2);
}

在这里插入图片描述

4.2 二维数组传参

void test(int arr[3][5])
{}
void test(int arr[][5])//行可以省略,但列不能省略,因为若不确定列,数据连续存储后,就无法正确拆开连续存储的数据排成几行输出。
{}

void test(int(*arr)[5])//因为二维数组的首元素是一维数组,所以形参用指针时要用数组指针。
{}

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;
}

思考:

当一个函数的参数部分为一级指针的时候,函数能接收什么参数?

例如:

void test1(int *p)//test1函数能接收什么参数?
{}

int main()
{
	int a = 10;
	test1(&a);//传整型变量的地址
	int* pa = &a;
	test1(pa);//传整型指针
	int arr[10] = { 1,2,3,4,5,6,7,8,9,0 };
	test1(arr);//传整形一维数组的数组名
	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;
}

思考:

当函数的参数为二级指针的时候,可以接收什么参数?

例如:

void test(char** p)
{}


int main()
{
	char c = 'b';
	char* pc = &c;
	char** ppc = &pc;
	char* arr[10];
	test(&pc);//指针的地址
	test(ppc);//二级指针
	test(arr);//指针数组的数组名:首元素(指针)的地址
	return 0;
}

5.函数指针

我们先来看一段代码:

#include <stdio.h>

int Add(int x, int y)
{
	return x + y;
}

int main()
{
	printf("%p\n", &Add);
	printf("%p\n", Add);
	return 0;
}

在这里插入图片描述
由此可见:
&函数名和函数名都是函数的地址

int (*pf1)(int, int) = Add;//pf1就是函数指针变量

pf1先和*结合,说明pf1是指针,指向的是一个函数,指向的函数有两个参数,参数类型都是int,返回类型为int

举例:

#include <stdio.h>

int Add(int x, int y)
{
	return x + y;
}

int main()
{
	int (*pf1)(int, int) = &Add;
	int ret1 = (*pf1)(2, 3);//函数名是地址,地址也是函数名,所以写不写*都行,*几乎是个摆设,写几个都行。
	int ret11 = (pf1)(2, 3);
	//int ret3 = &Add(2, 3);错误,因为 & 取的内容,必须是 = 左边出现过的。
	
	int (*pf2)(int, int) = Add;
	int ret2 = (*pf2)(2, 3);
	int ret22 = (pf2)(2, 3);
	int ret33 = Add(2, 3);

	printf("%d\n", ret1);
	printf("%d\n", ret11);
	printf("%d\n", ret2);
	printf("%d\n", ret22);
	printf("%d\n", ret33);

	return 0;
}

在这里插入图片描述

小知识:

int ret3 = &Add(2, 3); 错误,因为 & 取的内容,必须是 = 左边出现过的

5.1 两段有趣的代码

《C陷阱和缺陷》一书中提及这两个代码:

代码1:
(*(void (*)())0)();
void (*)()是一个函数指针类型,(void ( * )())0 是把0强转成这种函数指针类型的数据。这个代码是用来调用0地址处的函数。这个函数没有参数,返回类型是void。( *函数地址0的操作可写可不写,因为函数地址也就相当于函数名)

代码2:
void (*signal(int, void(*)(int)))(int);

  • 这个代码是一次函数声明,声明的是signal函数,signal函数的参数有2个;第一个是int类型,第二个是函数指针类型,该类型是void ( * )(int);该函数指针指向的函数,参数是int类型的,返回类型是void。
  • signal函数的返回类型也是函数指针类型,该类型是void (*)(int),该函数指针指向的函数,参数是int,返回类型是void。
疑问:

代码2太复杂了,能否简化呢?
可以用typedef类型重命名来解决:

》》与函数指针类型相关的内容,不能写在类型的左/右边,只能写在类型中 * 的后面
typedef void (*pfun_t)(int);//类型重命名
pfun_t signal(int, pfun_t);//函数调用
然而:
typedef void (*)(int) pfun_t;//错误
void (*)(int) signed(int, void (*)(int))//错误

总结

本片文章我们回顾了初阶指针,又学习了两种字符指针、指针数组、数组指针及其使用、一维二维的数组传参和指针传参、函数指针。感谢大家的阅读!大家一起进步。如果文章有错误的地方,欢迎大家在评论区指正。

点赞收藏加关注,C语言学习不迷路!
图片来源于网络

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1026423.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

linux下解决tomcat错误问题

错误一&#xff1a; Linux下Tomcat启动报错&#xff1a;Neither the JAVA_HOME nor the JRE_HOME environment variable is defined 原因&#xff1a;可能是Linux环境变了&#xff0c;需要在catalina.sh文件里指定JDK路径 解决方式&#xff1a; 在/bin/catalina.sh配置文件中加…

基于springboot+vue的爱心助农网站(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战 主要内容&#xff1a;毕业设计(Javaweb项目|小程序等)、简历模板、学习资料、面试题库、技术咨询 文末联系获取 项目介绍…

Linux离线安装telnet

TELNET rmp地址 链接: https://pan.baidu.com/s/1Yl3zt1p3zjp2fYTwByf7dQ?pwdgdcg 提取码: gdcg 复制这段内容后打开百度网盘手机App&#xff0c;操作更方便哦 命令 rpm -ivh xinetd-2.3.15-14.el7.x86_64.rpm rpm -ivh telnet-0.17-65.el7_8.x86_64.rpm rpm -ivh telnet-…

autoware.ai docker安装

1.进行docker安装 sudo apt-get update sudo apt-get install apt-transport-https ca-certificates curl software-properties-common curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo apt-key add - sudo apt-key fingerprint 0EBFCD88sudo add-apt-reposi…

Unity实战(11):项目非启动状态下使用代码批量替换材质

目录 前言 配置环境 一、场景准备 二、代码演示 三、效果呈现 四、关于Resources.Load()的说明 前言 本文内容为unity在编辑状态&#xff08;非启动状态&#xff09;下使用代码批量替换材质&#xff0c;该方法也适用于其他在编辑状态下对物体的操作需求。 配置环境 win1…

Mysql003:用户管理

前言&#xff1a;本章节讲解的是mysql中的用户管理&#xff0c;包括&#xff08;管理数据用户&#xff09;、&#xff08;控制数据库的访问权限&#xff09;。 目录 1. 查询用户 2. 创建用户 3. 修改用户密码 4. 删除用户 5. 权限控制 1. 查询用户 在mysql数据库中&#xff0…

华为手机如何开启设置健康使用手机模式限制孩子玩手机时间?

华为手机如何开启设置健康使用手机模式限制孩子玩手机时间&#xff1f; 1、在手机上找到「设置」并点击打开&#xff1b; 2、在设置内找到「健康使用手机」并点击进入&#xff1b; 3、开启健康使用手机后&#xff0c;选择孩子使用&#xff1b; 4、在健康使用手机内&#xff0c…

已解决 Java Error: Exception in thread ‘main‘ java.lang.ClassNotFoundException

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页: &#x1f405;&#x1f43e;猫头虎的博客&#x1f390;《面试题大全专栏》 &#x1f995; 文章图文并茂&#x1f996…

Godot配置C#语言编写脚本(使用VSCode作为外部编辑器)

文章目录 Godot部分查看VSCode的所在位置配置外部编辑器 配置VSCode编写脚本中文注释 其他文章字符编码 Godot部分 打开编辑器-编辑器设置&#xff1b; 查看VSCode的所在位置 右键单击你的VScode快捷方式&#xff0c;选择属性。 这里的目标就是你的VSCode所在的位置。 配…

并发编程——synchronized

文章目录 原子性、有序性、可见性原子性有序性可见性 synchronized使用synchronized锁升级synchronized-ObjectMonitor 原子性、有序性、可见性 原子性 数据库事务的原子性&#xff1a;是一个最小的执行的单位&#xff0c;一次事务的多次操作要么都成功&#xff0c;要么都失败…

【探索Linux】—— 强大的命令行工具 P.9(进程地址空间)

阅读导航 前言一、内存空间分布二、什么是进程地址空间1. 概念2. 进程地址空间的组成 三、进程地址空间的设计原理1. 基本原理2. 虚拟地址空间 概念 大小和范围 作用 虚拟地址空间的优点 3. 页表 四、为什么要有地址空间五、总结温馨提示 前言 前面我们讲了C语言的基础知识&am…

【智慧工地源码】智慧工地助力数字建造、智慧建造、安全建造、绿色建造

智慧工地围绕建设过程管理&#xff0c;建设项目与智能生产、科学管理建设项目信息生态系统集成在一起&#xff0c;该数据在虚拟现实环境中&#xff0c;将物联网收集的工程信息用于数据挖掘和分析&#xff0c;提供过程趋势预测和专家计划&#xff0c;实现工程建设的智能化管理&a…

Python实验一

1.计算圆椎体体系积。 要求: 交互式输入圆椎体的底面半径和高。 提示&#xff1a;&#xff08;1&#xff09;使用两个函数 input()和 eval()&#xff0c;其中 input()函数用于接收用户的输入&#xff0c;接收的值 是字符串&#xff1b;eval()函数用来执行一个字符串表达式&…

2023护网行动面试题目汇总

目录 一、常用的外围打点工具有哪些&#xff1f; 二、描述一下外围打点的基本流程&#xff1f; 三、怎么识别CDN? 四、怎么判断靶标站点是windows系统还是Linux系统&#xff1f; 五、举常见的FOFA在外网打点过程中的查询语句&#xff1f; 六、常见的未授权访问漏洞有哪些…

【官宣】游戏革命刚刚开始!

正如标题所言&#xff0c;随着官方 Aavegotchi dApp 游戏中心的推出&#xff0c;我们的 Gotchi 游戏革命今天正式开始。 游戏中心代表着 Aavegotchi.com 向类似于 Steam 和 Epic Games 等完整游戏平台的重大转变。 游戏中心是当今所有 Gotchi 主题游戏的综合目录&#xff0c;…

深度解剖数据在队列的应用

> 作者简介&#xff1a;დ旧言~&#xff0c;目前大一&#xff0c;现在学习Java&#xff0c;c&#xff0c;c&#xff0c;Python等 > 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 > 望小伙伴们点赞&#x1f44d;收藏✨加关注哟&#x1f495;&#x1…

基于微信小程序+Springboot线上租房平台设计和实现【三端实现小程序+WEB响应式用户前端+后端管理】

博主介绍&#xff1a;✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专…

【JAVA-Day28】数组下标越界问题:最佳解决方法

数组下标越界问题&#xff1a;最佳解决方法 数组下标越界问题&#xff1a;最佳解决方法引言一、什么是下标越界问题下标越界的表现 1.1 数组访问异常数组越界异常概述常见情况 1.2 内存访问错误内存访问错误概述常见情况 1.3 未定义行为未定义行为概述 二、下标越界问题如何产生…

Hive 的函数介绍

目录 ​编辑 一、内置运算符 1.1 关系运算符 1.2算术运算符 1.3逻辑运算符 1.4复杂类型函数 1.5对复杂类型函数操作 二、内置函数 2.1数学函数 2.2收集函数 2.3类型转换函数 2.4日期函数 2.5条件函数 2.6字符函数 三、内置的聚合函数 四、内置表生成函数 五、…

msvcp120.dll丢失怎么办?(五种方法快速解决)

首先&#xff0c;让我们来了解一下msvcp120.dll这个文件。msvcp120.dll是一个动态链接库文件&#xff0c;它是Microsoft Visual C 2012 Redistributable Package的一部分。这个文件的作用是支持一些应用程序的运行&#xff0c;例如游戏、办公软件等。当我们在使用这些软件时&am…