C语言--- 指针(3)

news2025/2/28 2:59:35

一.字符指针变量 

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

一般使用:

#include<stdio.h>
int main()
{
	char ch = 'a';
	char* p = &ch;
	*p = 'b';
	printf("%c\n",ch);
	return 0;
}

其实还有一种使用方式 :

#include<stdio.h>
int main()
{
	const char* pstr = "hello world";
	printf("%s",pstr);
	return 0;
}

这里是把一个字符串放到了pstr的指针变量里了吗?

其实不是,本质上是将字符串的首字符的地址放到了pstr中。

  1. 我们知道指针变量的大小其实是4/8字节,而字符串却不是,所以是不可能将字符串存放到pstr中的。
  2. #include<stdio.h>
    int main()
    {
    	 char* pstr = "hello world";
    	printf("%c",*pstr);
    	return 0;
    }

    在解引用后我们会发现打印的是h,也就是将字符串中首字符h的地址存放到了指针中

  3. 在我们学习指针运算时,指针+-整数,会根据指针的类型跳过不同的字节,这里其实也可以用于证明:
    #include<stdio.h>
    int main()
    {
    	char* pstr = "hello world";
    	printf("%s",pstr+5);
    	return 0;
    }

    因为pstr的类型是char*,所以+5会跳过五个字节从而只打印了后面的world。这也证明了pstr储存的就是首字符的地址。

#include<stdio.h>
int main()
{
	char* pstr = "hello world";
	*pstr = 'abc';
	printf("%s",pstr);
	return 0;
}

虽然这个代码在编译的时候并不会报错,但是程序运行时会崩溃。为什么呢?
因为“hello world”是常量字符串,常量字符串是不能修改的。所以pstr实际的类型应该是

const char* pstr = "hellon world";
//这里的const可写可不写

 虽然不可以修改但是我们可以使pstr指向另一个常量字符串,起到修改的作用。

#include<stdio.h>
int main()
{
	char* pstr = "hello world";
	pstr = "abcdef";
	printf("%s",pstr);
	return 0;
}

 这里打印出来的结果是abcdef;

看一道题:

#include <stdio.h>
int main()
{
	char str1[] = "hello bit.";
	char str2[] = "hello bit.";
	const char* str3 = "hello bit.";
	const char* str4 = "hello bit.";
	if (str1 == str2)
		printf("str1 and str2 are same\n");//1
	else
		printf("str1 and str2 are not same\n");//2
	if (str3 == str4)
		printf("str3 and str4 are same\n");//3
	else
		printf("str3 and str4 are not same\n");//4

	return 0;
}

这里输出结果是什么呢?

结果是2和3;

  1. str1和str2不相同原因:这是两个不同的数组,在内存中会独立开辟空间,既然是不同的空间,那么相应的地址也就不同的,所以是不一样的。
  2. str3和str4相同的原因:这⾥str3和str4指向的是⼀个同⼀个常量字符串。C/C++会把常量字符串存储到单独的⼀个内存区域,当⼏个指针指向同⼀个字符串的时候,他们实际会指向同⼀块内存,并不会开辟两次空间。 

二.数组指针变量 

指针数组是一种数组,数组中存放的是指针(地址)。

那么数组指针到底是数组?还是指针呢?

答案肯定是一种指针变量。

我们熟悉的指针变量:

  1. 整形指针变量:int* pint;存放的是整形变量的地址,能够指向整形数据的指针。
  2. 浮点数指针变量:float* pf;存放的是浮点型变量的地址,能够指向浮点数据的指针。

那么数组指针变量 :存放的是数组的地址,能够指向数组的指针变量。 

思考一下:p1和p2到底是什么??? 

int *p1[10];
int (*p2)[10];

p1毫无疑问是一个数组 ,*的优先级小于[ ],所以p1先与[10]结合,所以它先是一个数组,数组元素的类型是int*,所以它是一个指针数组。

那么p2就是数组指针了,因为()的优先级很高,可以用于更改其他操作符的优先级,所以p2先和*结合,代表它是一个指针,指针所指向的是一个大小为10的整形数组。

 数组指针如何初始化呢?

数组指针变量是⽤来存放数组地址的,得到数组的地址就需要用到&数组名  。

int arr[10] = { 0 };
int (*p)[10] = &arr;

&arr和p的类型是一模一样的。

int (*p)[10] = &arr; 
//int指的是p所指向的数组的元素的类型。
//*指p是一个指针变量。
//[10]指的是p所指向的数组的大小。

三.二维数组传参的本质

观察下面这个代码:

#include <stdio.h>
void test(int a[3][5], int r, int c)
{
	int i = 0;
	int j = 0;

	for (i = 0; i < r; i++)
	{
		for (j = 0; j < c; j++)
		{
			printf("%d ", a[i][j]);
		}
		printf("\n");
	}
}
int main()
{
	int arr[3][5] = { {1,2,3,4,5}, {2,3,4,5,6},{3,4,5,6,7} };
	test(arr, 3, 5);
	return 0;
}
⾸先我们再次理解⼀下⼆维数组,⼆维数组起始可以看做是每个元素是⼀维数组的数组,也就是⼆维数组的每个元素是⼀个⼀维数组。那么⼆维数组的⾸元素就是第⼀⾏,是个⼀维数组。

所以,根据数组名是数组⾸元素的地址这个规则,⼆维数组的数组名表⽰的就是第⼀个一维数组的地址。根据上⾯的例⼦,第⼀⾏的⼀维数组的类型就是 int [5] ,所以第⼀⾏的地址的类型就是数组指针类型 int(*)[5] 。那就意味着⼆维数组传参本质上也是传递了地址,传递的是第一⾏这个⼀维数组的地址,那么形参也是可以写成指针形式的。

 所以test函数就可以这样写。

void test(int(*p)[5], int r, int c)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < r; i++)
	{
		for (j = 0; j < c; j++)
		{
			printf("%d ", *(*(p + i) + j));
		}
		printf("\n");
	}
}

我们可以发现参数不仅改了,printf函数中的参数也有所变化。

*(*(p + i) + j);
p是第一行的地址,p+i就可以得到第二行或者第三行的地址。
再解引用就是一维数组首元素的地址,加j就可以得到不同位置元素的地址,
再解引用就可以得到该元素。

总结:⼆维数组传参,形参的部分可以写成数组,也可以写成数组指针的形式。 

四.函数指针变量

(1)函数指针变量的创建。

通过前面的学习,我们类比可得知,函数指针变量存放的就是函数的地址,也就是指向一个函数。

函数是否有地址呢?

我们可以运行一下这个代码。

#include<stdio.h>
int Add(int x,int y)
{
	return x + y;
}
int main()
{
	printf("%p\n",Add);
	printf("%p\n",&Add);
	return 0;
}

输出结果:

Add和&Add结果是一样的,也就是说函数名就是函数的地址,当然我们也可以&函数名来获取函数的地址。

那么我们应该如何书写这个指针变量呢?

int (*pf)(int,int) = &Add;
int (*pf)(int x,int y) = Add;//;x和y可写可不写

 如何理解这个呢?其实和数组指针是类似的

int (*pf3) (int x, int y)
|      |    ------------
|      |          |
|      |      pf3指向函数的参数类型和个数的交代
|      函数指针变量名
pf3指向函数的返回类型
int (*) (int x, int y) //pf3函数指针变量的类型
#include <stdio.h>
int Add(int x, int y)
{
	return x + y;
}
int main()
{
	int(*pf3)(int, int) = Add;

	printf("%d\n", (*pf3)(2, 3));//输出结果5
	printf("%d\n", pf3(3, 5));//输出结果8
	return 0;
}

观察两个有趣的代码:

代码1:

(*(void (*)())0)();

代码2:

void (*signal(int , void(*)(int)))(int);

代码1理解: 

  1. (void (*)( )) ,是一个返回值为void,参数为空的函数指针原型。
  2. (void (*)( ))0,把0强制类型转换为一个返回值为void,参数为空的函数指针,指针指向的地址为0.
  3. *(void (*)( ))0,前面加上*表示整个是一个返回值为void的函数的名字
  4. (*(void (*)( ))0)( ),这当然就是一个函数了。

代码2理解:

  1. void(*)(int)就是一个函数指针,这个函数的返回值为void,参数只有一个int.
  2. signal是这个函数名,它的参数就是int和void(*)(int).
  3. 返回值也是一个函数指针void(*)(int),只不过signal作为函数名写在了*后面。

(2)typedef关键字

typedef关键字使用类型重命名的,可以简化复杂的类型。

比如unsigned int 你觉得写起来不方便就可以这样写。

typedef unsigned int u_int;
int main()
{
	u_int a = 10;
}

这里就将unsigned int 重命名为了u_int;

指针也可以重命名。

typedef int* ptr_t;

但是对于函数指针和数组指针有一点不一样。

typedef int(*parr_t)[5]; //新的类型名必须写在*的后面
typedef void(*pfun_t)(int);

代码简化就可以这样写:

typedef void(*pfun_t)(int);
pfun_t signal(int, pfun_t);

五.函数指针数组

什么是函数指针数组呢?

首先,函数指针数组肯定是一个数组,存放的是函数的地址。

那么函数指针的数组如何定义呢?

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

答案是第一个。

parr1首先和[ ]结合,证明它是一个数组,然后是类型为int (*)()的函数指针。虽然第三个也是这样的,但是parr3[3]的位置不对,应该放在*后面。

六.转移表

函数指针数组的⽤途:转移表 。
利用函数指针数组实现计算器基本用法。
​​​​​​​
#include <stdio.h>
int add(int a, int b)//加法
{
	return a + b;
}
int sub(int a, int b)//减法
{
	return a - b;
}
int mul(int a, int b)//乘法
{
	return a * b;
}
int div(int a, int b)//除法
{
	return a / b;
}
int main()
{
	int x, y;
	int input = 1;
	int ret = 0;
	int(*p[5])(int x, int y) = { 0, add, sub, mul, div }; //转移表 函数指针数组
	do
	{
		printf("*************************\n");
		printf(" 1:add 2:sub \n");
		printf(" 3:mul 4:div \n");
		printf(" 0:exit \n");
		printf("*************************\n");
		printf("请选择:");
		scanf("%d", &input);
		if ((input <= 4 && input >= 1))
		{
			printf("输⼊操作数:");
			scanf("%d %d", &x, &y);
			ret = (*p[input])(x, y);
			printf("ret = %d\n", ret);
		}
		else if (input == 0)
		{
			printf("退出计算器\n");
		}
		else
		{
			printf("输⼊有误\n");
		}
	} while (input);
	return 0;
}

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

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

相关文章

vue2、vue3各自的响应式原理

查看本专栏目录 关于作者 还是大剑师兰特&#xff1a;曾是美国某知名大学计算机专业研究生&#xff0c;现为航空航海领域高级前端工程师&#xff1b;CSDN知名博主&#xff0c;GIS领域优质创作者&#xff0c;深耕openlayers、leaflet、mapbox、cesium&#xff0c;canvas&#x…

数学建模【相关性模型】

一、相关性模型简介 相关性模型并不是指一个具体的模型&#xff0c;而是一类模型&#xff0c;这一类模型用来判断变量之间是否具有相关性。一般来说&#xff0c;分析两个变量之间是否具有相关性&#xff0c;我们根据数据服从的分布和数据所具有的特点选择使用pearson&#xff…

《论文阅读》一个基于情感原因的在线共情聊天机器人 SIGIR 2021

《论文阅读》一个基于情感原因的在线共情聊天机器人 前言简介数据集构建模型架构损失函数实验结果咨询策略总结前言 亲身阅读感受分享,细节画图解释,再也不用担心看不懂论文啦~ 无抄袭,无复制,纯手工敲击键盘~ 今天为大家带来的是《Towards an Online Empathetic Chatbot…

MySQL之中文转拼音(全拼音、简拼)

中文转换拼音全称 1、创建拼音对照表 CREATE TABLE IF NOT EXISTS t_base_pinyin (pin_yin_ VARCHAR(255) CHARACTER SET gbk NOT NULL,code_ INT(11) NOT NULL,PRIMARY KEY (code_) ) ENGINEINNODB DEFAULT CHARSETutf8mb4;2、插入对照数据 INSERT INTO t_base_pinyin (pin…

提高办公效率:Excel在文秘与行政办公中的应用技巧

&#x1f482; 个人网站:【 海拥】【神级代码资源网站】【办公神器】&#x1f91f; 基于Web端打造的&#xff1a;&#x1f449;轻量化工具创作平台&#x1f485; 想寻找共同学习交流的小伙伴&#xff0c;请点击【全栈技术交流群】 在当今信息化时代&#xff0c;Excel作为一款常…

STM32程序(移植)中头文件的路径

例:#include "./BSP/LCD/lcd.h"为什么有的头文件加了路径? 先看AI的回答: 在C和C编程中&#xff0c;当我们在源文件中包含&#xff08;或称为“引入”或“导入”&#xff09;一个头文件时&#xff0c;编译器需要知道这个头文件的位置。通常&#xff0c;编译器会在…

hcia datacom课程学习(1):通信基础

1.总体框架 上图为发送方通过互联网传递信息给接收方的过程。 家用路由器会直接集成上图中的四层&#xff08;vlan&#xff0c;DHCP&#xff0c;静态路由&#xff0c;NAT&#xff0c;PPPoE&#xff09;。 2.网络性能指标 &#xff08;1&#xff09;带宽 单位时间内传输的数…

【linux进程信号(一)】信号的概念以及产生信号的方式

&#x1f493;博主CSDN主页:杭电码农-NEO&#x1f493;   ⏩专栏分类:Linux从入门到精通⏪   &#x1f69a;代码仓库:NEO的学习日记&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学更多操作系统知识   &#x1f51d;&#x1f51d; 进程信号 1. 前言2. 信号的基…

零基础C++开发上位机--基于QT5.15的串口助手(三)

本系列教程本着实践的目的&#xff0c;争取每一节课都带大家做一个小项目&#xff0c;让大家多实践多试验&#xff0c;这样才能知道自己学会与否。 接下来我们这节课&#xff0c;主要学习一下QT的串口编程。做一款自己的串口助手&#xff0c;那么这里默认大家都是具备串口通信…

keil的首次尝试,芯片为stm32F103C6T6

已经试了一下&#xff0c;吐槽这个软件的使用好麻烦啊 安装 然后先去安装对应的pack 这个鬼玩意里找对应的芯片&#xff0c;或者去官网上下载 我是在这里搜到芯片&#xff0c;再去官网下载一个驱动 https://www.keil.arm.com/packs/stm32f1xx_dfp-keil/boards/ 会有一个安装…

Laravel04 eloquent

eloquent 1. eloquent2. 创建eloquent model 以及 取数据 1. eloquent 文档地址&#xff1a; https://learnku.com/docs/laravel/8.x/eloquent/9406 下面是我们&#xff0c;通过laravel的DB类从数据库中获取了post记录&#xff0c;那么有没有可能我们直接获取一个post对象&am…

音频常用测试参数

一、总谐波失真&#xff08;THDN&#xff09; 总谐波失真指音频信号源通过功率放大器时&#xff0c;由于非线性元件所引起的输出信号比输入信号多出的额外谐波成份。谐波失真是由于系统不是完全线性造成的&#xff0c;我们用新增加总谐波成份的均方根与原来信号有效值的百分比来…

如何远程访问内网数据库?

天联是一种专门为远程访问内网数据库而设计的组网解决方案。由于其操作简单、跨平台应用、无网络要求以及独创的安全加速方案等原因&#xff0c;天联在几十万用户中广泛应用&#xff0c;解决了各行业客户的远程连接需求。通过采用穿透技术&#xff0c;天联实现了简单易用的远程…

基于雷达影像的洪水监测技术方法详解

洪水发生时候大多数是阴雨天气&#xff0c;光学影像基本上拍不到有效影像。雷达影像这时候就能发挥其不受天气影像的优点。现在星载的雷达卫星非常多&#xff0c;如高分三号、陆探一号、海丝一号&#xff08;巢湖一号&#xff09;、哨兵1号等。本文以哨兵1号L1地距(GRD)产品来介…

在Pycharm中运行Django项目如何指定运行的端口

方法步骤&#xff1a; 打开 PyCharm&#xff0c;选择你的 Django 项目。在菜单栏中&#xff0c;选择 “Run” -> “Edit Configurations...”。在打开的 “Run/Debug Configurations” 对话框中&#xff0c;选择你的 Django server 配置&#xff08;如果没有&#xff0c;你…

nginx(三)重写功能 防盗链 方向代理 等

return 可以写在location server if 里面 return用于完成对请求的处理&#xff0c;并直接向客户端返回响应状态码&#xff0c;比如:可以指定重定向URL(对于特殊重定向状态码&#xff0c;301/302等) 或者是指定提示文本内容(对于特殊状态码403/500等)&#xff0c;处于此指令后…

simple-pytest 框架使用指南

simple-pytest 框架使用指南 一、框架介绍简介框架理念&#xff1a;框架地址 二、实现功能三、目录结构四、依赖库五、启动方式六、使用教程1、快速开始1.1、创建用例&#xff1a;1.2、生成py文件1.3、运行脚本1.3.1 单个脚本运行1.3.2 全部运行 1.4 报告查看 2、功能介绍2.1、…

JAVA高并发——CompletableFuture

CompletableFuture是Java 8新增的一个超大型工具类。为什么说它大呢&#xff1f;因为它实现了Future接口&#xff0c;而更重要的是&#xff0c;它也实现了CompletionStage接口。CompletionStage接口也是Java 8中新增的&#xff0c;它拥有多达40种方法&#xff01;是的&#xff…

大数据职业技术培训包含哪些

技能提升认证考试&#xff0c;旨在通过优化整合涵盖学历教育、职业资格、技术水平和高新技术培训等各种教育培训资源&#xff0c;通过大数据行业政府引导&#xff0c;推进教育培训的社会化&#xff0c;开辟教育培训新途径&#xff0c;围绕大数据技术人才创新能力建设&#xff0…

k8s pv与pvc理解与实践

参考文章&#xff1a; https://blog.csdn.net/qq_41337034/article/details/117220475 一、 pv/pvc简述 Pv是指PersistentVolume&#xff0c;中文含义是持久化存储卷是对底层的共享存储的一种抽象&#xff0c;Pv由管理员进行配置和创建&#xff0c;只要包含存储能力&#xff…