初始C语言(7)——详细讲解有关初阶指针的内容

系列文章目录

 第一章 “C“浒传——初识C语言(1)(更适合初学者体质哦!)

 第二章 初始C语言(2)——详细认识分支语句和循环语句以及他们的易错点 

 第三章 初阶C语言(3)——特别详细地介绍函数 

 第四章 初始C语言(4)——详细地讲解数组的内容以及易错点 

 第五章 初始C语言(5)——详细讲解操作符以及操作符的易错点  

 第六章 初始C语言(6)——详细讲解表达式求值以及其易错点 

 第七章 初始C语言(7)——详细讲解有关初阶指针的内容


目录

系列文章目录

前言

一、指针是什么?

1.1 指针 

总结:指针就是地址,口语中受到指针通常是指针变量。 

1.2 指针变量

1.3 内存

1.2.1 什么是内存: 

1.2.2 内存的抽象模型:

二、指针和指针类型

2.1 指针的解引用

2.2 指针 +- 整数

2.3 指针类型的应用

三、野指针

3.1 野指针成因 

3.2 如何规避野指针 

3.2.1 指针初始化要注意两个点 

3.2.2 指针使用之前要检查有效性

悬空指针

四、指针运算

4.1 指针 +- 整数

4.2 指针 - 指针

4.3 指针的关系运算

五、指针和数组

六、二级指针

七、指针数组

总结


前言

       在上一章内,小编带领大家详细学习了有关表达式求值的相关内容,学习了隐式类型转换和显式类型转换的相关内容介绍了操作符的一些属性

       而在这一章内,小编要带领大家进行学习初阶指针的一些内容,不要害怕,这一部分只是为了后面讲述进阶指针做一个铺垫,所以不要担心指针这一节很难,当你刚开始就害怕的话,你将永远地活在这个阴影之下,会永远学不会这一章的内容,希望大家能够有耐心地将这一章看完!


一、指针是什么?

       说起指针,小编当年也是比较害怕的,因为指针算的上是C语言中很难的一个知识点,但是一般写代码的时候也不常用,导致很多人不是非常重视,所以请认真学习一下这一节,刚开始先讲一下之前讲过的东西。

指针理解的两个要点

  1. 指针是内存中一个最小单元的编号,也就是地址
  2. 平时口语中说的指针,通常指的是指针变量,是用来存放地址的变量

1.1 指针 

  • 指针是C语言非常重要的特征,指针也是一种变量,只不过它所表示的不是数据的值,而是内存的地址。
  • 通过使用指针,可以对任意(非绝对)内存地址的数据进行读写。
  • 在了解指针读写的过程前,我们先需要了解如何定义一个指针,和普通的变量不同,在定义指针时,我们通常会在变量名前加一个*号。

       我们以32位计算机为例,32位计算机的内存地址是4个字节,在这种情况下,指针的长度也是32位。下面会讲解32位计算机的地址线。

总结:指针就是地址,口语中受到指针通常是指针变量。 

       在上面总结中有两个名词:内存指针变量。 我们先来认识一下指针变量,再来认识一下内存。

1.2 指针变量

       在前面初始C语言(5)——详细讲解操作符以及操作符的易错点我们学过一个操作符 & ,我们可以通过 & (取地址操作符)取出变量的内存其实就是地址,把地址可以存放在一个变量中,这个变量就是指针变量。在了解指针读写的过程前,我们先需要了解如何定义一个指针,和普通的变量不同,在定义指针时,我们通常会在变量名前加一个 *

int main()
{int a = 0x11223344; //在内存中开辟一个空间int* p = &a;//这里我们对变量a,取出他的地址//a变量占用4个字节的空间,这里是将a的4个字节的第一个字节的地址存放在p变量中,p是一个指针变量。return 0;
}

总结:指针变量就是用来存放地址的变量。(存放在指针中的值都被当成地址处理) 

1.3 内存

       可能大家对内存的理解都不是很到位,在这里小编将要带领大家进行深一步学习,大家先记住一句话就是:内存和存储空间不是一回事。(更加详细地请看内存)

1.2.1 什么是内存: 

  • 内存是计算机中的重要部件,也称内存储器和主存储器,它是程序和CPU进行沟通的桥梁
  • 计算机中所有程序的运行都在内存中进行它用于暂时存放CPU中的运算数据以及与硬盘等外部存储器交换的数据内存性能的强弱影响计算机整体发挥的水平
  • 只要计算机开始运行,操作系统就会把需要运算的数据从内存调到CPU中进行运算,当运算完成,CPU将结果传送出来。

  

1.2.2 内存的抽象模型:

       为了方便大家记住,我们把内存模型映射成为我们现实生活中的模型,内存的模型是一层一层的,在现实生活中,其很像我们生活中的高楼大厦。

       在这个高楼大厦中,一层可以存储一个字节的数据楼层号就是地址,下面是内存和楼层整合的模型图。

  • 我们知道程序中的数据不仅只有数值,还有数据类型的概念 ,从内存上看就是占用内存大小(占用楼层数)的意思。
  • 即使物理上强制以1个字节为单位来逐一读写数据的内存,在程序中,通过指定其数据类型,也能实现以特定字节数为单位来进行读写。

在上面的讲解中也会有一些问题:

  • 一个小的单元到底是多大?(一个字节)
  • 如何编址?

       我们先来回答第一个问题:经过仔细地计算和权衡会发现一个字节给一个对应的地址是比较合适的。因为最小的数据类型是char,其单位为一个字节。如果过小,不好存储数字;如果过大,又很浪费空间。   

       那么回答第二个问题: 对于32位的机器,假设有32根地址线,那么假设每根地址线在寻址的时候产生高电平(高电压)低水平(低电压)就是(1或者0);那么32根地址线产生的地址就会是:

00000000 00000000 00000000 00000000

00000000 00000000 00000000 00000001

00000000 00000000 00000000 00000002

......

11111111   11111111   11111111   11111111

       这里就有2的32次方个地址,每一个地址表示一个字节。那我们就可以给(2^32byte == 2^32/1024KB ==  2^32/1024/1024MB == 2^32/1024/1024/1024GB == 4GB)4G的空间进行编址。同样的方法,那64位机器如果给64根地址线,那能编址多大空间?答案是16G的空间

但这里我们应该明白:

  • 在32位机器上,地址是32个0或者1组成的二进制序列,那地址就要用4个字节(一个字节是8个byte)的空间进行存储,所以一个字节变量的大小就应该是4个字节。
  • 那如果是在64位机器上,如果有64个地址线,那一个字节变量的大小是8个字节,才能存放一个地址。

总结:

  • 指针是用来存放地址的,地址是唯一标识一块地址空间的。
  • 指针的大小在32位平台是4个字节,在64位平台上是8个字节

二、指针和指针类型

       在之前,我们学习变量时,会根据现实生活提供不同的类型:整形,浮点型等……那么指针有没有类型呢?准确地说:有的。

当有这样的代码:

int num = 10;
p = #

       我们要将 &num(num的地址)保存到p中,我们知道p是一个指针变量,那么它的类型是什么呢?我们要给指针变量相应的类型。

char*   pc = NULL;
int*    pc = NULL;
short*  pc = NULL;
long*   pc = NULL;
float*  pc = NULL;
double* pc = NULL;

这里可以看到,指针的定义方式是:type + *

其实:

  • char* 类型的指针是为了存放 char 类型变量的地址;
  • short* 类型的指针是为了存放 short 类型变量的地址;
  • int* 类型的指针是为了存放 int 类型变量的地址。

       那为什么要这么麻烦?指针变量的大小不是都是4或者8个字节吗?那指针类型的意义是什么?请看下面进行讲解: 

2.1 指针的解引用

#include <stdio.h>
int main()
{int n = 0x11223344;char* pc = (char*)&n; //为了消除警告int* pi = &n;*pc = 0;//重点在调试的过程中观察内存的变化*pi = 0;//重点在调试的过程中观察内存的变化return 0;
}

在调试过程中,我们可以发现,指针类型是有意义的,指针类型决定了指针进行解引用操作时,访问几个字节。

总结:

       指针的类型决定了,对指针解引用的时候有多大的权限(能操作几个字节)。比如:char* 的指针解引用就只能访问一个字节,而 int* 的指针解引用就能访问四个字节

       经过上面的学习,有些人可能要疑惑,为什么不自动识别传来的地址是什么类别的呢? 其实是已经识别了。

  

2.2 指针 +- 整数

#include <stdio.h>
int main()
{int n = 10;char* pc = (char*)&n;int* pi = &n;printf("%p\n", &n);printf("%p\n", pc);printf("%p\n", pc + 1);printf("%p\n", pi);printf("%p\n", pi + 1);return 0;
}

为什么这两个指针跳过的大小不同呢?

 首先,这两个指针的类型是不一样的;

其次,整形类型指针指向的是一个整形的对象;字符类型指针指向的是一个字符的对象。

总结:

       指针的类型决定了指针向前或者向后走一步跳过几个字节(距离)。 比如:一个 char* 的指针加1跳过一个字节,一个 int* 的指针加1跳过4个字节。

2.3 指针类型的应用


三、野指针

       野指针,顾名思义就是不确定的指针,大家可以用野人来类比。概念:野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的) 

说明:指针变量也是变量,在前面我们知道如果变量不初始化,在内存中会存放随机值(全局变量不初始化,存放0,局部变量不初始化,存放随机值)。同理指针变量如果赋给随机值是没有意义的,会成为野指针的。随机值会导致指针无法指向一个有效的内存空间,操作系统不允许操作此指针指向的内存区域

注释:野指针是不会直接引发错误的,而野指针指向的内存空间会出现问题的。 

3.1 野指针成因 

1.指针未初始化:指针变量终究只是一个变量(就和全局变量与局部变量一样),如果不给这个变量进行初始化,那么这个变量将会存放随机值

下面进行代码演示:

    int* p; //局部变量指针为初始化,默认为随机值
//正确写法为:int* p = NULL;*p = 20;

2.指针越界访问:在数组这一节中,我们学习数组越界访问,其实指针越界访问与其类似,指针指向的空间超出了其分配的合理空间

下面进行代码演示:

    int arr[10] = {0};int* p = arr;for(int i = 0; i <= 10; i++)
//正确写法为:for(int i = 0; i < 10; i++){//当指针指向的范围超出数组arr的范围时,p就是野指针*(p++) = i;}

3.指针指向的空间释放:在未学习动态内存开辟时,我们先用函数释放来解释这个成因。建立一个返回值为指针类型的函数,在主函数中用指针变量进行接收,当程序出函数体后,原先函数所指向的空间进行释放,此时指针就为野指针

下面进行代码演示:

int* test()
{int a = 10;return &a;
}
int main()
{int* p = test();printf("%d\n", *p);return 0;
}

3.2 如何规避野指针 

  1. 指针初始化
  2. 小心指针越界
  3. 指针指向的空间释放,及时置NULL
  4. 避免放回局部变量的地址
  5. 指针使用之前检查有效性

3.2.1 指针初始化要注意两个点 

先看代码,进行举例:

int a = 10;
int* p = &a;
  • 如果明确指针应该指向哪里的话,就应该初始化正确的地址
  • 如果不能明确指针应该指向哪里的话,安全起见要将指针初始化为NULL (空指针,就是0)

3.2.2 指针使用之前要检查有效性

#include <stdio.h>
int main()
{int* p = NULL; //不管三七二十一,我们要进行初始化//……int a = 10;p = &a;if(p != NULL)*p = 20;return 0;
}

       在这里,我们要先说一句,NULL是0,其实它也是一个地址,但是在我们用户手中是无法访问的,一旦我们去访问,程序就会崩溃。 

悬空指针


四、指针运算

4.1 指针 +- 整数

int main()
{int arr[] = { 0,1,2,3,4,5,6,7,8,9 };//            0 1 2 3 4 5 6 7 8 9 //使用指针打印数组的内容int* p = arr;for (int i = 0; i < 10; i++){printf("%d ", *(p + i));//p指向的是数组首元素//p+i 是数组中下标为i的元素的地址//p+i 起始时跳过了i*sizeof(int)个字节}printf("\n");return 0;
}

 扩展:arr == p

            arr+i == p+i 

            *(arr+i) == *(p+i) == arr[i]

            *(i+arr) == i[arr]

4.2 指针 - 指针

  • 指针 - 指针的前提是:两个指针指向同一块区域,指针类型相同的
  • 指针 - 指针差值的绝对值是:指针和指针之间的元素个数
int arr[10];
printf("%d\n", &arr[0] - &arr[9]);  //-9
printf("%d\n", &arr[9] - &arr[0]);  //9

  

模拟实现一下strlen()函数

size_t my_strlen(char* ptr)
{char* s = ptr;while(*s)
//还可以这样写:while(*s != '\0')s++;return s - ptr;
}

4.3 指针的关系运算

#define N_VALUES 5
for(vp = &values[N_VALUES];vp > &values[0];)
{*--vp = 0;
}

代码简化,将这个代码进行修改:

for(vp = &values[N_VALUES];vp > &values[0]; vp--)
{*vp = 0;
}

       实际上在绝大部分的编译器上是可以顺利完成任务的,然而我们还是要避免这样写,因为标准并不能保证这个方法可行。 

标准规定: 

       在C语言中,允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与指向第一个元素之前的那个内存位置的指针进行比较。


五、指针和数组

在一小部分中,我们要知道:

  • 指针就是指针,指针变量就是变量,是用来存放地址的,指针变量的大小是4/8个字节
  • 数组就是数组,可以存放一组数,数组的大小是取决于元素的类型和个数
  • 数组的数组名是数组首元素的地址(两种情况除外),指针变量是可以访问地址的

在绝大多数情况下,数组名和数组首元素的地址是一样的,看下图:

总结:

数组名表示数组首元素的地址。 

但是有两个例外:

  1. sizeof(数组名),数组名单独放在sizeof内部,数组名表示整个数组,计算的是数组的大小,单位是字节。
  2. &数组名,数组名表示整个数组,取出的是数组的地址。数组的地址和数组首元素的地址,值是一样的,但是类型和意义是不一样的。

       既然可以把数组名当成地址存放到一个指针中,我们使用指针来访问就成为可能,那我们可以直接通过指针来访问数组。看下下面代码:

int main()
{int arr[] = { 1,2,3,4,5,6,7,8,9,0 };int* p = arr; //指针存放数组首元素的地址int sz = sizeof(arr) / sizeof(arr[0]);for (int i = 0; i < sz; i++){printf("%d ", *(p + i));}return 0;
}

六、二级指针

       指针变量也是变量(一级指针变量),是变量就有地址,那指针变量的地址存放在哪里?这个就是二级指针。下面就是代码解释:

int a = 10;
int* p = &a; //p是指针变量,一级指针变量
int* * pp = &p; //pp是指针变量,二级指针变量
//你还可以继续写,有几个*,就是几级指针变量
//int** * ppp = &pp; //ppp是指针变量,三级指针变量


七、指针数组

       指针数组是指针还是数组呢?答案是:是数组,是存放指针的数组。我们可以使用指针数组来模拟实现一个二维数组。

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 };for (int i = 0; i < 5; i++){for (int j = 0; j < 5; j++){printf("%d ", arr[i][j]);}printf("\n");}return 0;
}

  

总结

       在这一部分,小编详细地编写了有关初阶指针的一篇博客。希望大家看完以后,进行点评,谢谢大家!

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

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

相关文章

NestJS 中的 gRPC 微服务通信

想象一下&#xff0c;你回家过节&#xff0c;你的家人决定聚会。而不是让一个人做所有的烹饪&#xff0c;每个人都同意带上他们擅长制作的特色菜。目标是通过组合所有这些菜肴来制作一顿完整的饭菜。你同意做鸡肉炒饭&#xff0c;你哥哥做甜点蛋糕&#xff0c;妹妹做沙拉。 每…

2022年工作架构分析

mpmw自动化流程工具 schema动态数据 Schema 本身是一个JSON &#xff0c;Schema 通过一些特定字段描述和定义 JSON的数据结构。 最常见的表单通过类XML语法定义。一些库支持通过一些特定结构的 JSON (Schema)来生成类XML标签。 formily 是其中实现之一。 表单设计器通过可视…

excel统计函数篇2之count系列

1、COUNT(value1,[value2],…):计算参数列表中数字的个数 2、COUNTA(value1,[value2],…)&#xff1a;计算参数列表中值的个数 联想在excel之数学函数、excel中的通配符一文中提到求和函数&#xff1a; SUMIF(range,ceriteria,[sum_range])&#xff1a;对范围内符合指定条件的…

【广州华锐互动】牲畜养殖VR模拟实操系统为传统教育注入新的生命力

随着科技的不断发展&#xff0c;虚拟现实(VR)技术已经逐渐走进我们的生活。在农业领域&#xff0c;VR技术的应用也日益广泛&#xff0c;为现代农业人才培养提供了新的途径。 由广州华锐互动开发的“牲畜养殖VR模拟实操系统”引起了广泛关注&#xff0c;系统包含了鸡、猪、牛、马…

MFC140.dll缺失的修复方法,安装MFC140.dll文件

大家好&#xff0c;今天我要和大家分享的是如何正确安装和使用MFC140.dll。MFC140.dll是一种常见的动态链接库文件&#xff0c;它是Microsoft Foundation Classes(MFC)的一部分&#xff0c;被广泛应用于Windows操作系统中的各种应用程序中。在本文中&#xff0c;我们将详细介绍…

「Vue|网页开发|前端开发」01 快速入门:用vue-cli快速写一个Vue的HelloWorld项目

本文主要介绍如何用vue开发的标准化工具vue-cli快速搭建一个符合实际业务项目结构的hello world网页项目并理解vue的代码文件结构以及页面渲染流程。 文章目录 一、准备工作&#xff1a;安装node.js二、项目搭建创建项目目录全局安装vue-cli使用Webpack初始化项目启动项目学会…

STM32 printf函数

printf函数输出流程 用户调用printf()函数到C标准库调用printf函数相关部分&#xff0c;printf函数由编译器提供的stdio.h解析。包含在usart.h文件中。fputc()最终实现输出。用户需要根据最终输出的硬件重新定义该函数&#xff0c;此过程为&#xff1a;printf重定向。 printf的…

html动态爱心代码【四】(附源码)

目录 前言 特效 完整代码 总结 前言 情人节马上就要到了&#xff0c;为了帮助大家高效表白&#xff0c;下面再给大家带来了实用的HTML浪漫表白代码(附源码)背景音乐&#xff0c;可用于520&#xff0c;情人节&#xff0c;生日&#xff0c;表白等场景&#xff0c;可直接使用。…

pytorch中的register_buffer

今天在一个模型的init中遇到了self.register_buffer(‘running_mean’, torch.zeros(num_features)) register_buffer(self, name, tensor)是一个PyTorch中的方法&#xff0c;它的作用是向模块&#xff08;module&#xff09;中添加一个持久的缓冲区&#xff08;buffer&#xf…

校企合作谋发展 合作共赢谱新篇|云畅科技与湖南民族职业学院签订校企合作协议

产业是经济发展的重要引擎&#xff0c;人才是产业发展的重要资源。为积极探索软件人才培育新路径&#xff0c;共商政产学研协同新机制&#xff0c;8月8日&#xff0c;云畅科技与湖南省民族职业学院教育技术学院软件技术专业签订校企合作协议。 会上&#xff0c;学院副校长王志平…

小象课堂在线授课教育系统

此项目包含后端全部代码&#xff0c;前端包括后台和web界面的源码&#xff0c;数据库用的mysql,可当作课设或者毕设&#xff0c;还可写入自己的简历中 web界面展示&#xff1a; 前端后台界面展示&#xff1a; 用户管理 课程管理 内容配置 订单管理 系统管理 系统监控

STM32 F103C8T6学习笔记12:红外遥控—红外解码-位带操作

今日学习一下红外遥控的解码使用&#xff0c;红外遥控在日常生活必不可少&#xff0c;它的解码与使用也是学习单片机的一个小过程&#xff0c;我们将通过实践来实现它。 文章提供源码、测试工程下载、测试效果图。 目录 红外遥控原理&#xff1a; 红外遥控特点&#xff1a; …

AI让儿童绘画动起来-程序员带娃必备

项目效果演示 项目描述 很多小朋友在学习绘画的过程中&#xff0c;创作出来很多比较有创意的作品&#xff0c;那么怎么让这些作品&#xff0c;动起来&#xff0c;甚至是和拍摄的视频进行互动呢&#xff0c;今天分享的这个项目&#xff0c;能够完美解决这个问题。 项目地址http:…

爬虫ip带你探索无限可能

各位程序猿大佬们&#xff0c;今天我要为大家带来一个备受关注的话题&#xff1a;爬虫ip的应用范围&#xff01;你可能会好奇&#xff0c;什么是爬虫ip&#xff1f;它在我们的日常生活中有哪些神奇的应用呢&#xff1f;让我们一起来揭开这个神秘的面纱&#xff0c;探索无限可能…

2023-腾讯QQ客户端远程代码执行漏洞

2023-腾讯QQ客户端远程代码执行漏洞 一、漏洞详情二、威胁程度三、影响范围四、漏洞复现五、修复建议 免责声明&#xff1a;请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失&#xff0c;均由使用…

macOS上编译obs-studio

前言 最近基于obs的1个二开程序&#xff0c;需要移植到macOS平台上&#xff0c;由于遇到些问题&#xff0c;本文记录下如何在macOS上配置&编译&运行obs程序完整过程。 下载 首先下载cmake-gui工具&#xff0c;下载CMAKE&#xff0c;选择对应macOS平台的cmake版本&…

PostgreSQL父子建表查询所有的子数据-利用自定义函数查询

pgsql 函数查询代码 select find_space_tree_list_by_nodeid(1,1) 查询结果示意图 获取子集函数代码 CREATE OR REPLACE FUNCTION "public"."find_space_tree_list_by_nodeid"("nodeid" varchar, "viewid" varchar)RETURNS "…

特斯拉Model 3的七年狂飙

‍ 作者 | 张祥威 编辑 | 德新 发布一周拿下32万张订单&#xff0c;之后用时五年&#xff0c;交付量突破100万辆。粗略计算&#xff0c;自2016年发布至今&#xff0c;特斯拉Model 3已交付超150万辆。 放眼新能源赛道&#xff0c;如此战绩 别无二家。 Model 3踩中纯电动车的…

常用的电参数

电参数根据电流的特点可以分为直流电参数和交流电参数&#xff0c;在电参数中有些是可以通过电参数表测得&#xff0c;有些参数则为通过测得的参数计算而来。 一、电参数 1.1 直接可测电参数 ——瞬时电压值 ——瞬时电流值 n——采样点数 f——频率 time——时间 其中&…

基于web的服装商城系统java网上购物商店jsp源代码mysql

本项目为前几天收费帮学妹做的一个项目&#xff0c;Java EE JSP项目&#xff0c;在工作环境中基本使用不到&#xff0c;但是很多学校把这个当作编程入门的项目来做&#xff0c;故分享出本项目供初学者参考。 一、项目描述 基于web的服装商城系统 系统有1权限&#xff1a;前台…