【理解指针(二)】

文章目录

  • 一、指针的运算
    • (1)指针加整数
    • (2)指针减指针(指针关系运算)
  • 二、野指针
    • (1)野指针的成因
      • (1.1)指针未初始化
      • (1.2)指针的越界访问
      • (1.3)指针指向的空间释放
  • (2)如何避免野指针
      • (2.1)指针需要初始化
      • (2.2)指针变量不再使用时,及时置NULL,指针使用之前检查有效性
  • 三、 strlen的模拟实现
  • 四、总结

一、指针的运算

(1)指针加整数

例子,我们打印一组数字:

#include<stdio.h>
int main()
{int arr[10] = { 0,1,2,3,4,5,6,7,8,9 };int* pa = &arr[0];int sz = sizeof(arr) / sizeof(arr[0]);for (int i = 0; i < sz; i++){printf("%d ", *(pa + i));}return 0;
}

在这里插入图片描述

这里的pa + i就是指针变量加上一个整数,pa + i是地址,*是对这个地址的解引用。

(2)指针减指针(指针关系运算)

#include<stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10};int* pa = &arr[0];printf("%d ", &arr[9] - pa);return 0;
}

在这里插入图片描述
上面的代码我们也可以用指针关系运算来写:(这里我们需要用到循环)

#include<stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int* pa = &arr[0];int sz = sizeof(arr) / sizeof(arr[0]);while (pa < arr+sz){printf("%d ", *pa);pa++;}return 0;
}

这里的pa < arr+sz是指针的比较,pa就是第一个元素开始的地址,而arr+sz就是最后一个元素结束的地址:

这里的&arr[9] - pa也可以写成&arr[9]-&arr[0],而得到的结果是 9 。由此我们可以得到一个结论:
指针的关系运算,得到的是指针和指针之间元素的个数。

二、野指针

(1)野指针的成因

(1.1)指针未初始化

示范一个错误例子:

#include<stdio.h>int main()
{int* pa;*pa = 1;printf("%d ", *pa);return 0;
}

在这里插入图片描述

局部变量指针未初始化,默认为随机值。

(1.2)指针的越界访问

#include<stdio.h>
int main()
{int arr[10] = { 0 };int* pa = &arr[0];for (int i = 0; i <= 11; i++){*(pa++) = i;printf("%d ", *pa);}return 0;
}

指针指向的范围超出数组arr的范围,pa就是野指针(指针指向的范围有11个,而数组arr的范围只有10个)

在这里插入图片描述

(1.3)指针指向的空间释放

#include<stdio.h>
int Print()
{int a = 90;return &a;
}
int main()
{int* pa = Print();printf("%d ", *pa);return 0;
}

我们写一个函数,将a赋值为90,然后把a的空间返回到主函数中,*pa可以接受到a的地址,但是出了Print()函数,空间就被回收了,此时*pa带着地址去访问该空间,不会得到任何数字,这就是空间的释放。

(2)如何避免野指针

(2.1)指针需要初始化

  • 如果我们明确知道指针指向哪里就直接赋值地址:
#include<stdio.h>
int main()
{int a = 20;int* pa = &a;return 0;
}

这里前面就是知道pa指针指向的是a的地址,所以我们直接:int* pa = &a

  • 如果我们不知道指针应该指向哪里,可以给指针赋值NULL。NULL 是C语⾔中定义的⼀个标识符常量,值是0,0也是地址,这个地址是⽆法使⽤的,读写该地址会报错。
#include<stdio.h>
int main()
{int* p = NULL;*p = 20;          //errprintf("%d ", *p);return 0;
}

小心指针的越界:

⼀个程序向内存申请了哪些空间,通过指针也就只能访问哪些空间,不能超出范围访问,超出了就是越界访问。(可以见上面的(1.2))

(2.2)指针变量不再使用时,及时置NULL,指针使用之前检查有效性

#include<stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int* pa = &arr[0];for (int i = 0; i < 10; i++){*(pa++) = i;}pa= NULL;pa = &arr[0];if (pa != NULL){for (int i = 0; i < 10; i++){printf("%d ", *(pa + i));}}return 0;
}

*(pa++) = i;循环结束后,pa是超出了arr的范围的,此时可以把pa重置;当我们要重新用到pa时,可以让pa重新获得地址,重新获得地址后,我们需要判断pa是不是空指针,以确保代码的安全性。

  • 避免返回局部变量的地址(可见(1.3))

三、 strlen的模拟实现

库函数strlen的功能是求字符串长度,统计的是字符串中 \0 之前的字符的个数。

如果要模拟实现只要从起始地址开始向后逐个字符的遍历,只要不是 \0 字符,计数器就+1,这样直到 \0 就停止。

#include<stdio.h>
#include<string.h>
int my_strlen(char* pc)
{int count = 0;while (*pc != '\0'){count++;pc++;}return count;
}
int main()
{char arr[] = { "abcdf" };int len = my_strlen(arr);printf("%d ", len);return 0;
}

上面的代码,我们可以发现有很多的不足之处,不可以确保代码完全的安全性:

  1. 如果用户不小心把数组传成了空指针
    在这里插入图片描述

  2. int是有符号的整形,它可以为负数,但是我们统计数字不可能是负数

  3. char *pc只是用于遍历的,不能对*pc进行修改
    在这里插入图片描述
    所以最后的代码应该改为:

#include<assert.h>
#include<stdio.h>
#include<string.h>
size_t my_strlen( const char* pc)
{assert(pc != NULL);size_t count = 0;while (*pc != '\0'){count++;pc++;}return count;
}
int main()
{char arr[] = { "abcdf" };size_t len = my_strlen(arr);printf("%zd ", len);return 0;
}

四、总结

指针需要理解的东西有很多,希望大家可以自己慢慢去消化,指针这一章的内容还没有结束哟,希望与大家下一次再见。ԅ(¯ㅂ¯ԅ)

在这里插入图片描述

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

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

相关文章

idea2023版使用废弃类没有删除线处理方法

idea2023版使用废弃类没有删除线处理方法 新版Idea使用废弃类时,默认是黄色警告处理方法1. 打开file -> setting2. 编辑(Editor) -> 检查(Inspections) -> 搜索Deprecated API usage 新版Idea使用废弃类时,默认是黄色警告 处理方法 1. 打开file -> setting 2. 编…

微服务---Eureka注册中心

目录 一、服务中的提供者与消费者 二、Eureka工作流程 三、搭建Eureka服务 四、服务拉取 五、总结 1.搭建EurekaServer 2.服务注册 3.服务发现 一、服务中的提供者与消费者 服务提供者&#xff1a;一次业务中&#xff0c;被其他微服务调用的服务。即提供接口给其他微服务。…

<Linux> 初识线程

目录 前言&#xff1a; 一、什么是线程 &#xff08;一&#xff09;基本概念 &#xff08;二&#xff09;线程理解 &#xff08;三&#xff09;线程与进程的关系 &#xff08;四&#xff09;简单实用线程 &#xff08;五&#xff09;重谈虚拟地址空间 1. 页表的大小 2…

编码器-解码器模型(Encoder-Decoder)

注意&#xff1a;本文引用自专业人工智能社区Venus AI 更多AI知识请参考原站 &#xff08;[www.aideeplearning.cn]&#xff09; 编码器-解码器模型简介 Encoder-Decoder算法是一种深度学习模型结构&#xff0c;广泛应用于自然语言处理&#xff08;NLP&#xff09;、图像处理…

数组:初始化,访问某一个,遍历

文章目录 静态初始化数组数组的访问&#xff1a;遍历数组案例 动态初始化数组总结案例 静态初始化数组 定义数组的时候直接给数组赋值。 简化格式&#xff1a; int[] ages {12,52,96}; 完整格式&#xff1a; int[] ages new int[]{12,16,26};数组变量名中存储的是数组在内存…

数据结构奇妙旅程之二叉平衡树

꒰˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好&#xff0c;我是xiaoxie.希望你看完之后,有不足之处请多多谅解&#xff0c;让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN …

Dubbo-记录

1.概念 Apache Dubbo 是一款 RPC 服务开发框架&#xff0c;用于解决微服务架构下的服务治理与通信问题&#xff0c;官方提供了 Java、Golang 等多语言 SDK 实现。使用 Dubbo 开发的微服务原生具备相互之间的远程地址发现与通信能力&#xff0c; 利用 Dubbo 提供的丰富服务治理…

Linux最小系统安装无法查看IP地址

1&#xff0c;出现原因 服务器重启完成之后&#xff0c;我们可以通过linux的指令 ip addr 来查询Linux系统的IP地址&#xff0c;具体信息如下: 从图中我们可以看到&#xff0c;并没有获取到linux系统的IP地址&#xff0c;这是为什么呢&#xff1f;这是由于启动服务器时未加载网…

Decontam去污染:一个尝试

为了程序运行的便利性&#xff0c;不想将Decontam放到windows的Rstudio里面运行&#xff0c;需要直接在Ubuntu中运行&#xff0c;并且为了在Decontam时进行其他操作&#xff0c;使用python去运行R 首先你需要有一个conda环境&#xff0c;安装了R&#xff0c;Decontam&#xff0…

【b站咸虾米】1 Vue介绍 2021最新Vue从基础到实例高级_vue2_vuecli脚手架博客案例

课程地址&#xff1a;【2021最新Vue从基础到实例高级_vue2_vuecli脚手架博客案例】 https://www.bilibili.com/video/BV1pz4y1S7bC/?share_sourcecopy_web&vd_sourceb1cb921b73fe3808550eaf2224d1c155 感觉尚硅谷的Vue看完忘得差不多了&#xff0c;且之前学过咸虾米的unia…

使用css结合js实现html文件中的双行混排

此前写过一个使用flex布局实现html文件中的双行混排&#xff0c;但是感觉效果不佳。经过几天思考&#xff0c;我认为双行混排的要点其实是两个&#xff1a; 1、正文和批注的文字大小不同&#xff1b; 2、正文和批注的行距相互配合进行设定。 正文和批注的文字大小及行距都可…

【Python】进阶学习:OpenCV--一文详解cv2.namedWindow()

【Python】进阶学习&#xff1a;OpenCV–一文详解cv2.namedWindow() &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1f448; 希望…