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

news2024/12/25 12:54:31

系列文章目录

 第一章 “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.coloradmin.cn/o/911918.html

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

相关文章

C++信息学奥赛2049:【例5.19】字符串判等

这段代码的功能是比较两个输入的字符串是否相等&#xff08;忽略大小写和空格&#xff09;&#xff0c;并输出 “YES” 或 “NO”。 解析注释后的代码如下&#xff1a; #include<bits/stdc.h> using namespace std; int main() {string arr; // 定义字符串变量arr&…

Python网络爬虫入门到实战

&#x1f482; 个人网站:【工具大全】【游戏大全】【神级源码资源网】&#x1f91f; 前端学习课程&#xff1a;&#x1f449;【28个案例趣学前端】【400个JS面试题】&#x1f485; 寻找学习交流、摸鱼划水的小伙伴&#xff0c;请点击【摸鱼学习交流群】 网络爬虫&#xff08;We…

C++信息学奥赛1130:找第一个只出现一次的字符

这段代码的功能是找出输入字符串中第一个重复出现的字符&#xff0c;并输出该字符。 解析注释后的代码如下&#xff1a; #include<bits/stdc.h> using namespace std; int main() {string arr;getline(cin, arr); int a0;for(int i0;i<arr.length();i){for(int j0;j…

怎么看电脑型号?5个方法,快速查看!

“弟弟快上大学了&#xff0c;想给他买一部性价比好一点的电脑&#xff0c;但是我对电脑又不太熟悉&#xff0c;请问大家怎么查看电脑型号呢&#xff1f;感谢回答&#xff01;” 每台电脑不一样&#xff0c;它的电脑型号也可能有差别。电脑型号是指一台电脑特定的标识&#xff…

在外SSH远程连接macOS服务器

文章目录 前言1. macOS打开远程登录2. 局域网内测试ssh远程3. 公网ssh远程连接macOS3.1 macOS安装配置cpolar3.2 获取ssh隧道公网地址3.3 测试公网ssh远程连接macOS 4. 配置公网固定TCP地址4.1 保留一个固定TCP端口地址4.2 配置固定TCP端口地址 5. 使用固定TCP端口地址ssh远程 …

(vue)el-table 怎么把表格列中相同的数据 合并为一行

(vue)el-table 怎么把表格列中相同的数据 合并为一行 效果&#xff1a; 文档解释&#xff1a; 写法&#xff1a; <el-table:data"tableData"size"mini"class"table-class"borderstyle"width:100%"max-height"760":span-…

ubuntu查看网速

使用speedomster测试网速 sudo apt-get install speedometer 查询需要测速的网卡 speedometer -r ens33 -t ens33 -r: 指定网卡的接收速度 -t: 指定网卡的发送速度 使用nload测试 sudo apt-get install nload 测速 nload -t 200 -i 1024 -o 128 -U M 参数含义&#xff0…

自组织映射

无监督深度学习&#xff1a; 无监督模型使神经网络能够执行聚类、异常检测、特征选择、特征提取、降维和推荐系统等任务。这些神经网络包括 自组织图、玻尔兹曼机、自动编码器。 什么是 SOM&#xff1f; 简而言之&#xff0c;自组织映射是一种基于竞争学习的人工神经网络&am…

react通过ref获取函数子组件实例方法

在react16之后带来了hooks之后&#xff0c;确实方便了很多组件开发&#xff0c;也加快了函数式编程的速度&#xff0c;但是当你通过useRef获取子组件的时候&#xff0c;又恰好子组件是一个函数组件&#xff0c;那么将会报一个错误&#xff1a;报这个错误的主要原因是函数组件没…

Lnton羚通云算力平台如何在OpenCV-Python 中,来进行图像算术运算?

在 OpenCV-Python 中&#xff0c;可以使用函数 ​​cv2.add()​​、​​cv2.subtract()​​、​​cv2.multiply()​​ 和 ​​cv2.divide()​​ 来进行图像算术运算。这些函数接受两个输入图像&#xff0c;并对其进行逐像素的运算。 1.图像加法&#xff1a; ​​OpenCV​​ 是…

【jsthreeJS】入门three,并实现地月绕行,附带全码

效果图&#xff1a; 序幕准备&#xff1a; 需要自己准备地球跟月球的纹理贴图&#xff0c;如下&#xff1a; 图片格式转换网址&#xff0c;如下&#xff1a; Zamzar - File Conversion progress 开始编码&#xff1a; 实现步骤如下&#xff1a; 在JavaScript部分&#xff0c;…

【Vue】使用Hbuilder x开发vue前端项目

原文作者&#xff1a;我辈李想 版权声明&#xff1a;文章原创&#xff0c;转载时请务必加上原文超链接、作者信息和本声明。 文章目录 一、Hbuilder x 代码风格调整1.在工具中找到设置&#xff0c;点击插件设置2.代开jsbeautify.js配置2.uni-app项目icon的使用 一、Hbuilder x …

解读逻辑输出光隔离器TLP2355(TPL,E(T 在工业自动化应用技巧

逻辑输出光隔离器是一种用于隔离电路中的逻辑信号和光信号的器件。它通常由光电耦合器和逻辑输出接口组成。 光电耦合器是一种能够将光信号转换为电信号的器件&#xff0c;它由一个发光二极管和一个光敏二极管组成。当发光二极管接收到电路中的逻辑信号时&#xff0c;它会发出…

智能在线客服系统,客户问题接待更及时

优质的客户服务可以提高客户的满意度和忠诚度&#xff0c;从而为企业带来更多的商业机会。智能在线客服系统&#xff0c;是指通过人工智能技术和自然语言处理技术&#xff0c;将客户的问题自动化地分配给相应的客服人员&#xff0c;从而实现客户问题的快速解答。相比传统的客服…

七夕节海外网红营销:助力品牌赢得全球消费者

随着全球化的进程不断加速&#xff0c;七夕节作为一个传统的华人节日&#xff0c;正逐渐走向国际舞台&#xff0c;成为了海外品牌开拓市场的营销新热点。近年来&#xff0c;海外网红已成为品牌宣传的重要工具&#xff0c;他们的影响力和粉丝基础成为了出海品牌开拓海外市场的利…

WindowsServer安装IIS以及搭建Asp网页

简介 Internet Information Services (IIS) 是一个由微软公司创建和维护的&#xff0c;用于在Windows系统上托管网站和Web应用程序的服务器。IIS是Windows Server的一个组成部分&#xff0c;并且也可以在某些版本的Windows上运行。 IIS支持多种Web开发技术&#xff0c;包括AS…

第六章,创作文章

6.1添加创作页面 <template><div class="blog-container"><div class="blog-pages"><div class="col-md-12 panel"><div class="panel-body"><h2 class="text-center">创作文章&l…

220V转5V芯片三脚芯片-AH8652

220V转5V芯片三脚芯片是一种非常常见的电源管理芯片&#xff0c;它通常被用于将高压交流输入转为稳定的直流5V输出。芯片型号AH8652是一款支持交流40V-265V输入范围的芯片&#xff0c;采用了SOT23-3三脚封装。该芯片内部集成了650V高压MOS管&#xff0c;能够稳定地将输入电压转…

中小型企业人事考勤管理系统的设计(论文+源码)_kaic

摘要 中小企业是我国经济的重要组成部分&#xff0c;人事管理对于中小企业的发展和运营至关重要。传统的人事考勤管理方式往往存在着效率低、准确度不高以及易于被操控等问题&#xff0c;给企业的管理带来了一定的困扰。为了解决中小企业在人事考勤管理方面面临的问题&#xff…

Databend 开源周报第 107 期

Databend 是一款现代云数仓。专为弹性和高效设计&#xff0c;为您的大规模分析需求保驾护航。自由且开源。即刻体验云服务&#xff1a;https://app.databend.cn 。 Whats On In Databend 探索 Databend 本周新进展&#xff0c;遇到更贴近你心意的 Databend 。 理解连接参数 …