C语言指针之 进阶

news2024/10/6 8:38:35

前言

今天来较为深入的介绍一下指针,希望大家能有所收获~

那么,先进行一些简单的基础知识复习吧。
在这里插入图片描述

字符指针

格式:char *

补充:

表达式“abcdef”的值是首字符a的地址
所以当像下面这么使用时,它的含义是:

char *pc = “abcde”

pc存储的是a的地址

printf(%s”, p)

打印出的abcde

提示:

此处可以理解为将abcde存储在一个数组里面,数组名就是首元素地址

那么由此可以引出一个比较少见的写法:


输出的是c

注意

因为字符串“abcde”是常量,不可改变,但是,p作为指针变量是可以改变的,这时候改变p指向的对象程序就会报错。

所以最后在*左边加上const,使p指向的对象不可改变

一道题:

看下面这段代码:
问:输出结果是什么

#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");
	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;
}

运行结果:

在这里插入图片描述

知识点:
当两个常量数组相同时,只会储存一份,
所以,虽然创建了两个变量,但存储的是同一个地址(同一个值)

图解:
在这里插入图片描述

指针数组

定义:存放指针的数组

用途:
可以用指针数组模拟一个二维数组

如下:

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

数组指针

定义:指向数组的指针
格式(唯一):
存储元素的类型 (*p)[数组大小] = &arr

注意:要明确指定出数组的大小,如果不写就默认为0,程序会报错。

但数组指针的使用不常见,
因为当对数组指针进行解引用后,就是数组名,多此一举。
就算使用,也大多用于二维数组,下面举个例子

void print(int (*p)[5], int r, int c)
{
	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");
	}
}
 
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);
 
	return 0;
}

此处数组名指向的是二维数组的第一行
这里传递的 arr,也就是第一行的地址,是一维数组的地址
此处用数组指针来接收。

旧识新知

数组名指的是首元素的地址,
但有两个例外
1.sizeof(数组名)此处指的是整个数组

2.&数组名
取出的是数组的地址

那么创建一个变量存储数组的地址,他就是数组的指针,简称数组指针

但是 此处需要注意区分数组名和取地址数组名,虽然在打印时二者是一样的,但在运算时,因为 &arr 表示的是 数组的地址 ,所以它是以整个数组大小去运算的,而不是数组中首元素的大小

示例如下:

#include <stdio.h>
int main()
{
 int arr[10] = { 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;
}

运行结果:
在这里插入图片描述
这样就能直观的看出二者的区别

一组辨析

大家可以思考一下下面几行的代码的含义是什么

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

解释:
在这里插入图片描述
在这里插入图片描述

数组传参和指针传参

接下来,我会为大家介绍三种传参情况
一维数组传参、二维数组传参、一级指针传参、二级指针传参

一维数组传参

看看下面这段代码,思考一下那种传参方式是可行的。

#include <stdio.h>
void test(int arr[])//ok?
{}
void test(int arr[10])//ok?
{}
void test(int* arr)//ok?
{}
void test2(int* arr[20])//ok?
{}
void test2(int** arr)//ok?
{}
 
int main()
{
 int arr[10] = {0};
 int* arr2[20] = { 0 };
 test(arr);
 test2(arr2);
}

答案是:这些写法都可以。

数组传参传的是数组首元素的地址。

二维数组传参

我们先看这段代码:

void test(int arr[3][5])//ok?
{}
void test(int arr[][])//ok?
{}
void test(int arr[][5])//ok?
{}
void test(int *arr)//ok?  这是接收一维数组的写法
{}
void test(int* arr[5])//ok?这是指针数组,也不行
{}
void test(int (*arr)[5])//ok?这才是对的,用数组指针
{}
void test(int **arr)//ok?二级指针是用来接收一级指针的,所以也不行
{}
int main()
{
 int arr[3][5] = {0};
 test(arr);
}


知识点:

总结:二维数组传参,函数形参的设计只能省略第一个[]的数字。
因为对一个二维数组,可以不知道有多少行,但是必须知道一行多少元素,这样才可以运算。

一级指针传参

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

用一级指针来实现打印数组。
一级指针传参,形参也要写成一级指针的形式来接收。

思考:

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

有如下三种方式:
在这里插入图片描述
传入数组名(数组首元素的地址)
传入整型变量的地址
传入一级指针

二级指针传参

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

同样,二级指针传参要用同样是二级指针的形参来接收

思考;

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

有如下三种方式:
在这里插入图片描述

传进去一级指针的地址
传进去二级指针
传进去指针数组的首元素地址(也就是指针的地址)

函数指针

定义:指向函数的指针,在内存空间中存放的是函数的地址。

函数地址:&函数名
或者 直接写函数名
函数名就是函数的地址

代码示例:

#include <stdio.h>
void test()
{
	printf("hehe\n");
}
int main()
{
	printf("%p\n", test);
	printf("%p\n", &test);
	return 0;
}

程序输出的结果相同。

函数指针的存储

观察下面这段代码,看看那个才是函数指针

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

答案:第一个
因为pfun1先与8结合,是一个指针,然后才指向这个函数。
这个函数不传参,并且返回类型为void

格式:
指针的类型 (*pf)(函数参数的类型) = 函数名

运用:

想通过函数指针来实现两个数的求和,可以怎么写呢?

代码如下;

int Add(int x, int y)
{
	return x + y;
}
int main()
{
	int (* pf2)(int, int) = &Add;//这里需要把类型强转换成整型
	int ret = (* pf2)(2, 3);
	printf("%d\n", ret);
 
	return 0;
}

所以函数调用可以写成:
pf(参数)

*个数随便写,但要用括号括起来,

(*pf)(参数)

提示:

在使用函数指针时,是不用写*的,因为pf本就是地址,想要调用函数,直接解引用就行,不用多写星号,并且此处写多少个星号都行,
写星号只是为了提醒其他人这个变量是函数指针,便于理解。

题目:

下面介绍两段有趣的代码,大家可以思考一下他们的含义是什么

1.

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

拆开来看:
void(*)()是一个函数指针类型,

对(void (*)())0,就是强制类型转换0,把0转换成可以接收函数地址的指针变量,

那么0地址处就有void(*)()这么一个函数,
所以这段代码的意思就是:调用0地址处的函数,这个函数没有参数,返回类型是void。

2.

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

这行代码我们从里往外看

图解如下:
在这里插入图片描述

可以通过使用typedef,简化成两行代码:

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

结语

指针的知识暂时介绍到这里,下篇文章会继续介绍关于指针的知识,比如函数指针数组、指向函数指针数组的指针、回调函数等,我们下次见。

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

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

相关文章

2023软件测试岗必问的100个面试题【含答案】

一、测试理论 1.什么是软件测试&#xff1f; 答&#xff1a;软件测试是通过执行预定的步骤和使用指定的数据&#xff0c;以确定软件系统在特定条件下是否满足预期的行为。 2.测试驱动开发&#xff08;TDD&#xff09;是什么&#xff1f; 答&#xff1a;测试驱动开发是一种开…

LeetCode150道面试经典题--最后一个单词的长度(简单)

1.题目 给你一个字符串 s&#xff0c;由若干单词组成&#xff0c;单词前后用一些空格字符隔开。返回字符串中 最后一个 单词的长度。 单词 是指仅由字母组成、不包含任何空格字符的最大子字符串。 2.示例 3.思路 通过对字符串的反转&#xff0c;转为数组开始遍历&#xff0c…

【Git】版本控制器详解之git的概念和基本使用

版本控制器git 初始Gitgit的安装git的基本使用初始化本地仓库配置本地仓库三区协作添加---add修改文件--status|diff版本回退--reset撤销修改删除文件 初始Git 为了能够更⽅便我们管理不同版本的⽂件&#xff0c;便有了版本控制器。所谓的版本控制器&#xff0c;就是⼀个可以记…

二分查找(详解)

目录 介绍 思路 循环实现 详解 递归实现1 详解 注意 递归实现2 两个递归代码之间的区别 总结 介绍 二分查找法&#xff0c;也称为折半查找法&#xff0c;是一种在有序数组中查找特定元素的高效算法。其基本思路是将目标元素与数组中间的元素进行比较&#xff0c…

跳跃游戏 II——力扣45

文章目录 题目描述解法一 贪心题目描述 解法一 贪心 int jump(vector<int>& nums){in

【剑指offer】栈与队列4题 全刷(详解)

目录 目录 目录 [简单]剑指 Offer 09. 用两个栈实现队列 题目 方法 [简单]剑指 Offer 30. 包含min函数的栈 题目 方法1&#xff1a;笨办法 方法2&#xff1a;辅助栈 [困难]剑指 Offer 59 - I. 滑动窗口的最大值 题目 方法&#xff1a;单调队列 [中等]剑指 Offer 5…

固态硬盘对游戏性能的影响及优势解析

固态硬盘的作用在于提高电脑的读取速度&#xff0c;这对于游戏性能的提升有着重要的影响。在一台电脑中&#xff0c;CPU和显卡是核心硬件&#xff0c;而游戏的流畅度则主要取决于显卡的性能&#xff0c;显卡的性能直接影响游戏的帧数高低。如果我们的电脑配置与朋友的电脑相似&…

异常支出的真实成本意想不到!管理采购异常支出有实招

采购中的异常支出是指合同外支出&#xff0c;或在预先制定的采购政策之外从非首选供应商处购买的支出。不同组织中异常支出的差异会很大&#xff0c;异常支出的比例取决于管理下的支出、采购政策实施的成功程度和采购成熟度。 异常支出会给企业带来多少损失&#xff1f; 曾有…

springboot+mybatis实现简单的增、删、查、改

这篇文章主要针对java初学者&#xff0c;详细介绍怎么创建一个基本的springboot项目来对数据库进行crud操作。 目录 第一步&#xff1a;准备数据库 第二步&#xff1a;创建springboot项目 方法1&#xff1a;通过spring官网的spring initilizer创建springboot项目 方法2&am…

qt creater运行按钮灰色,问题记录

第一次安装还没运行就出了三个错误&#xff1a; 1.F:\wei\Qt\Tools\CMake_64\share\cmake-3.24\Modules\CMakeTestCXXCompiler.cmake:62: error: The C compiler "C:/Program Files (x86)/Microsoft Visual Studio 14.0/VC/BIN/amd64/cl.exe" is not able to compil…

列队 Queue 接口概述

在Java中&#xff0c;Queue&#xff08;队列&#xff09;是一种基本的数据结构&#xff0c;用于按照先进先出&#xff08;FIFO&#xff09;的顺序存储元素。Java提供了多种实现Queue接口的类&#xff0c;以下是几种常见的实现方式&#xff1a; LinkedList&#xff1a;LinkedLis…

linux环形缓冲区kfifo实践4:异步通知fasync

基础知识 异步通知在内核中使用struct fasync_struct数据结构来描述。 <include/linux/fs.h> struct fasync_struct {spinlock_t fa_lock;int magic;int fa_fd;struct fasync_struct *fa_next; /* singly linked list */struct file *fa_file;struct rcu_head fa…

CTF竞赛密码学之 LFSR

概述: 线性反馈移位寄存器&#xff08;LFSR&#xff09;归属于移位寄存器&#xff08;FSR&#xff09;,除此之外还有非线性移位寄存器&#xff08;NFSR&#xff09;。移位寄存器是流密码产生密钥流的一个主要组成部分。 G F ( 2 ) GF(2) GF(2)上一个n级反馈移位寄存器由n个二元…

matlab使用教程(12)—随机数种子和随机数流

1.生成可重复的随机数 1.1指定种子 本示例显示如何通过首先指定种子来重复生成随机数数组。每次使用相同种子初始化生成器时&#xff0c;始终都可以获得相同的结果。首先&#xff0c;初始化随机数生成器&#xff0c;以使本示例中的结果具备可重复性。 rng( default ); 现在…

django实现登录和登录的鉴权

1、创建数据库的管理员表 在models.py 中定义admin表&#xff0c;为了简单&#xff0c;表里只有用户名和密码还有默认加的id 三个字段 from django.db import models# Create your models here.class Admin(models.Model):username models.CharField(verbose_name"用户…

新利好带动 POSE 持续上扬,月内几近翻倍

PoseiSwap 是 Nautilus Chain 上的首个 DEX&#xff0c;得益于 Nautilus Chain 的模块化 Layer3 构架&#xff0c;PoseiSwap 正在基于 zk-Rollup 方案构建全新的应用层&#xff0c;并基于此构建隐私、合规等全新的特性&#xff0c;为未来其布局 RWA 领域推动 Web2、Web3 世界的…

布谷鸟配音:一站式配音软件

这是一款智能语音合成软件&#xff0c;可以快速将文字转换成语音&#xff0c;拥有多种真人模拟发音&#xff0c;可以选择不同男声、女声、童声&#xff0c;以及四川话、粤语等中文方言和外语配音&#xff0c;并且可对语速、语调、节奏、数字读法、多音字、背景音等进行全方位设…

【gridsample】地平线如何支持gridsample算子

文章目录 1. grid_sample算子功能解析1.1 理论介绍1.2 代码分析1.2.1 x,y取值范围[-1,1]1.2.2 x,y取值范围超出[-1,1] 2. 使用grid_sample算子构建一个网络3. 走PTQ进行模型转换与编译 实操以J5 OE1.1.60对应的docker为例 1. grid_sample算子功能解析 该段主要参考&#xff1a;…

最大子数组和——力扣53

文章目录 题目描述解法一 动态规划题目描述 解法一 动态规划 int maxSubArray(vector<int>& nums){int pre=0, res=nums

spring boot策略模式实用: 告警模块为例

spring boot策略模式实用: 告警模块 0 涉及知识点 策略模式, 模板方法, 代理, 多态, 反射 1 需求概括 场景: 每隔一段时间, 会获取设备运行数据, 如通过温湿度计获取到当前环境温湿度;需求: 对获取回来的进行分析, 超过配置的阈值需要产生对应的告警 2 方案设计 告警的类…