进阶指针(一)

图片来源于网络

✨博客主页:小钱编程成长记
🎈博客专栏:进阶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");elseprintf("str1 and str2 are not same\n");if (str3 == str4)printf("str3 and str4 are same\n");elseprintf("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的地址赋值给数组指针变量preturn 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.mzph.cn/news/83682.shtml

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

相关文章

C# Onnx Yolov8 Cls 分类

效果 项目 代码 using Microsoft.ML.OnnxRuntime; using Microsoft.ML.OnnxRuntime.Tensors; using OpenCvSharp; using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System…

Jenkins学习笔记1

CI 服务器&#xff1a; 认识Jenkins&#xff1a; Jenkins是一个可扩展的持续集成&#xff08;CI&#xff09;引擎&#xff0c;是一个开源项目&#xff0c;旨在提供一个开放易用的软件平台&#xff0c;使得软件持续集成变成可能。Jenkins非常易于安装和配置&#xff0c;简单易…

什么是SVG(可缩放矢量图形)?它与普通图像格式有何不同?

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ 什么是SVG&#xff1f;⭐ 与普通图像格式的不同⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 欢迎来到前端入门之旅&#xff01;感兴趣的可以订阅本专栏哦&#xff01;这个专栏是为那些对Web开发感兴趣、刚刚…

Tomcat多实例+Nginx动静分离、负载均衡

这里写目录标题 Tomcat多实例动静分离、负载均衡一、Tomcat多实例部署1、安装JDK2、安装启动tomcat 二、NginxTomcat负载均衡、动静分离1、Nginx负载均衡实现原理1.1 原理1.2 Nginx配置反向代理的主要参数 2、Nginx动静分离实现原理2.1 原理2.2 Nginx静态处理优势 3、动静分离配…

jvm-sandbox-repeater源码解析-配置管理

一、配置初见 源码里提供的控制台截图如下&#xff1a;&#xff08;怎么搭建自己去百度&#xff09; 从中取出对应的配置如下&#xff1a; { "degrade": false, //阻断能力 "exceptionThreshold": 1000, //异常采样率 "httpEntrancePatterns&qu…

工业检测 ocr

采用OpenCV和深度学习的钢印识别_菲斯奇的博客-CSDN博客采用OpenCV和深度学习的钢印识别[这个帖子标题党了很久&#xff0c;大概9月初立贴&#xff0c;本来以为比较好做&#xff0c;后来有事情耽搁了&#xff0c;直到现在才有了一些拿得出手的东西。肯定不会太监的。好&#xf…

2023.9.19 关于 数据链路层 和 DNS 协议 基本知识

目录 数据链路层 MTU DNS 协议 补充 DHCP协议 数据链路层 基本概念&#xff1a; 考虑相邻两个节点之间的传输&#xff08;通过 网线 / 光纤 / 无线 直接相连的两个设备&#xff09;以太网协议 规定了 数据链路层 和 物理层 的内容 IP地址 与 mac地址 的相互配合 IP地址 描…

vue的模板语法(下篇)

目录 一.事件处理 二.表单的综合案例 三.组件通信⭐⭐ 3.1 自定义组件 3.2 组件通信之父传子 3.3组件通信之子传父 一.事件处理 Vue通过由点(.)表示的指令后缀来调用修饰符&#xff0c; .stop .prevent .capture .self .once 如下&#xff1a; 阻止单击事件冒泡 <a v-on…

踩坑:Invalid character found in method name. HTTP method names must be tokens

一、原因 在进行本地小程序与服务端请求时,由于加了签名认证,访问接口时报错 Spring boot端 小程序端 二、解决方案 2.1 更改访问路径 将https:更换成http: 示例:https://localhost:8080 改为 http://localhost:8080 2.2其他原因 ssl证书到期了Tomcat的header缓冲区大小不…

2023数学建模国赛游记

第一参加数学建模国赛&#xff0c;大概也是最后一次参加了&#xff0c;记录一下这几天的历程吧。 我们队的情况是计算机电气数统&#xff0c;计算机负责编程&#xff0c;电气学院的负责论文部分&#xff0c;数统的同学负责建模&#xff0c;数据处理部分我们是共同承担。 第一天…

秦丝9周年 | 各行业实体生意如何实现数字化转型?

近期&#xff0c;北京、深圳、天津、重庆等全国27个省都在推进“一刻钟便民生活圈”——以社区居民为服务对象&#xff0c;在步行15分钟左右的范围内&#xff0c;满足居民日常生活基本消费和品质消费。 而各行业的实体店是这个“圈”中的重要组成部分&#xff0c;很多入驻的实…

【使用Cpolar将Tomcat网页传输到公共互联网上】

文章目录 1.前言2.本地Tomcat网页搭建2.1 Tomcat安装2.2 配置环境变量2.3 环境配置2.4 Tomcat运行测试2.5 Cpolar安装和注册 3.本地网页发布3.1.Cpolar云端设置3.2 Cpolar本地设置 4.公网访问测试5.结语 1.前言 Tomcat作为一个轻量级的服务器&#xff0c;不仅名字很有趣&#…

pdfjs解决ie浏览器预览pdf问题

pdfjs是一个js库&#xff0c;可以将pdf文件用canvas重新绘制&#xff0c;从而无需借助pdf读取插件就可以直接预览。 目前chrome内核的浏览器已内置pdf读取插件&#xff0c;但ie浏览器还没有。而我们最近在做的一个项目使用对象是医院&#xff0c;使用的浏览器竟然还是ie。所以我…

Python 数据分析学习路线

Python 数据分析学习路线 第一阶段&#xff1a;Python语言基础第二阶段&#xff1a;数据采集和持久化第三阶段&#xff1a;数据分析第四阶段&#xff1a;数据挖掘与机器学习书籍介绍参与方式 第一阶段&#xff1a;Python语言基础 在学习数据分析之前&#xff0c;首先需要掌握P…

iOS加固保护技术:保护你的iOS应用免受恶意篡改

目录 转载&#xff1a;开始使用ipaguard 前言 下载ipa代码混淆保护工具 获取ipaguard登录码 代码混淆 文件混淆 IPA重签名与安装测试 转载&#xff1a;开始使用ipaguard 前言 iOS加固保护是直接针对ios ipa二进制文件的保护技术&#xff0c;可以对iOS APP中的可执行文件…

【操作系统笔记】内存分配

内存对齐 问题&#xff1a;为什么需要内存对齐呢&#xff1f; 主要原因是为了兼容&#xff0c;为了让程序可以运行在不同的处理器中&#xff0c;有很多处理器在访问内存的时候&#xff0c;只能从特定的内存地址读取数据。换个说法就是处理器每次只能从内存取出特定个数字节的数…

ClickHouse与Elasticsearch比较总结

目录 背景 分布式架构 存储架构 写入链路设计 Elasticsearch 再谈Schemaless 查询架构 计算引擎 数据扫描 再谈高并发 性能测试 日志分析场景 access_log&#xff08;数据量197921836&#xff09; trace_log&#xff08;数据量569816761&#xff09; 官方Ontime测…

云原生的简单理解

一、何谓云原生&#xff1f; 一种构建和运行应用软件的方法 应用程序从设计之初即考虑到云的环境&#xff0c;原生为云而设计&#xff0c;在云上以最佳姿势运行&#xff0c;充分利用和发挥云平台的弹性分布式优势。 二、包括以下四个要素 采用容器化部署&#xff1a;实现云平…

el-table表格中加入输入框

<template><div class"box"><div class"btn"><el-button type"primary">发送评委</el-button><el-button type"primary" click"flag true" v-if"!flag">编辑</el-button…

win系统环境搭建(九)——Windows安装chatGPT

windows环境搭建专栏&#x1f517;点击跳转 win系统环境搭建&#xff08;九&#xff09;——Windows安装chatGPT 本系列windows环境搭建开始讲解如何给win系统搭建环境&#xff0c;本人所用系统是腾讯云服务器的Windows Server 2022&#xff0c;你可以理解成就是你用的windows…