百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

C语言重点——指针篇(一篇让你完全搞懂指针)

haoteby 2025-02-21 13:27 94 浏览

一. 前言

C语言是比较偏底层的语言,为什么它比较偏底层,就是因为他的很多操作都是直接针对内存操作的。

这篇我们就来讲解C语言的一大特点,也是难点,指针和指针操作。

这篇文章我会先从基本类型的存储过程和原理讲起,然后再讲解指针int *p,再举一反三,搞懂int **p和int ***p,学会指针。

搞懂int *p,int **p和int ***p,完全学会指针!!!!

二. 理解一个变量的存储过程和原理(必须清楚掌握)

2.1 直接'='赋值

int a = 5;
printf("a = %d",a);

结果: a = 5

这一句话完成了两个操作,我们先了解c语言在计算机内部干了什么?

两个操作:

(1)int a;

在栈中定义了一个变量a,并且在内存中开辟了一个int类型大小的空间, 即4个字节 ,然后让a指向这篇空间,也就是这篇空间,计算机分配给了a, a以后就有了一片属于自己的空间;

(2) a = 5;

在a的自己的那片空间,里面存放数值5 ,把5转换成二进制,存到a的4个字节的空间 。

2.2 利用输入流,手动赋值

scanf("%d",&a);

我们还有过输入赋值操作,刚好可以证明上述观点:

用户输入了一个int类型的数值,比如输入5,然后&a,先找到a的那片地址空间,最后把5转成二进制,存入a的那片地址空间,即完成了对a的赋值,也就是在a的那片4字节的空间填入了二进制的5;

2.3 总结

从上述讲述我们可以了解,一个变量的存储,先从内存开辟一个类型大小的空间(int类型4个字节大小),在让变量指向这篇空间,即就是这片空间属于这个变量,再在这片空间中存储你要存储的数值。

三. 指针类型(int *)的存储过程和原理

3.1 指针类型的赋值规范

(1) 第一种先定义后赋值

int *p;
p = &a;	  //这种方式正确
printf("p = %d\n",p);

结果:p = 6618636

变量p存放的a的地址

重点:

先了解,指针类型,int *p,虽然是*p在一起写着,但是变量名叫p,类型为int *,也就是整型的指针类型,当你理清变量名和类型之后,你对指针的理解程度已经懂了大半了 。

(2)第二种定义赋值一步完成

int *r = &a;
printf("r = %d\n",r);

结果:r = 6618636存放的是a的地址

还有一种常用的错误赋值方法:

// 	p = a;    //这种赋值方式错误

错误的操作,不能把一个具体的数字赋给指针(类型不匹配),

一个指针类型,一个int类型


3.2 指针存储过程和原理

前面列举了两种常用的指针的赋值,下来具体讲解计算机都干了什么?

  • 可以把存放一个int类型变量的地址赋给一个int *指针类型的变量
  • '='左边是一个int *指针类型的变量 ,可以存放放置着int类型数值的地址
  • '='右边是&a,a是int类型的变量数值5,&是取地址符,&a就是拿到int类型a的数值的地址

总的来说,就是把a的那片空间,给了p一个钥匙,让p也可以对a的那片空间操作,这个已经属于指针操作了,后面我们会讲到。

由上述可以证明,c语言的赋值,必须是类型对应

总结:int *p; 变量名叫p,类型为int *,可存放一个int数据的地址 。

注意:这块的可存放一个int数据的地址,不是存放一个地址,是int类型

例如:

 int a = 5;
 int *p;
 p = &a;
  • 这里a是一个int类型的变量,存放的int类型的数值5

&a 取到了存放int类型a的地址

p = &a; 把int类型a的地址赋给了int *类型的p

即就是int *类型的变量可存放一个int数据的地址

四. 指针类型(int **)的存储过程和原理

int **q;
q = &p;
printf("q = %d\n",q);

结果:q = 6618624存放的p的地址

int *p明白了,那么int **q呢?

首先: 先对数据类型和变量划分开

int **q; 变量名为q,数据类型为int **

int *中存放的是int类型数据的地址

int **中存放的是int类型数据的地址的地址

上述我们明白了,一个*是指一个int数值的地址,

p中存放的是个int数值的地址,p = 6618636,为a的地址。

那么我们可以推到,两个*q就是存放p的地址。

p存放a的地址,p本身也是一个变量,他的值为a的地址 ,

而内存也给他自己开辟了一片空间,让他存放而他的数值

q存放p的地址,q也是一个变量,


他们的指向关系如下 :

a<----p<----q

五. 指针类型(int ***)的存储过程和原理

int ***m;
m = &q;
printf("m = %d\n",m);

结果:m = 6618616存放的q的地址

既然,int *和int **都懂了,那么int ***就迎刃而解了

同理,int*** 存放的是int **类型数据的地址

六. 指针操作(*操作)

这块我们这说属于指针自己的操作

        printf("p = %d\n",p);
	printf("*p = %d\n",*p);
	printf("q = %d\n",q);
	printf("*q = %d\n",*q);
	printf("**q = %d\n",**q);
	printf("m = %d\n",m);
	printf("*m = %d\n",*m);
	printf("**m = %d\n",**m);
	printf("***m = %d\n",***m);

结果:

p = 6618636
*p = 5
q = 6618624
*q = 6618636
**q = 5
m = 6618616
*m = 6618624
**m = 6618636
***m = 5

p、q和m都是上述例子中的变量

首先除了定义指针变量的时候,变量前面有*为定义类型,其他时候均为指针的取值操作,注意是取值,不是取址,拿的是指针变量中存放的值。

6.1 举个现实中栗子

举个现实中的例子,你比如说去银行开保险柜,其中*p操作比如开保险柜这个操作,你得先拿着你的柜子号在银行找到保险柜,然后拿着钥匙再打开保险柜取出里面的钱;就像是p中存放的是一个地址,你先拿着p中存放的地址,在内存中找到那块空间,然后再*p操作,取出那块空间中存放的值。

所以 * 操作就是取值操作,即取出指针变量存放的地址中所存放的数据。


6.2 *操作怎么去分析

上面几个例子都属于指针的取值操作,也就是也就是拿着指针变量中存的地址号去内存中找里面存的东西。

所以看这种连着好几个*后面跟个变量的表达式,需要从右往左依次抛开

即:***m ; 就是* ( * ( * m ))),看的时候需要从最里层一层一层抛开。


  • *p ;
  • 先看成*(p),再从里向外看,首先他有一个变量p,所有直接可以先从内存中拿到p存放的数据6618636(p中的数据),再找到内存中6618636那片内存,最后取出6618636中的存放数据5(具体数据)


  • **q = 5 ;
  • 先看成 *( * (q)),再从里向外看
  • 先从内存中拿到q存放的数据6618624(q中的数据),再从内存中找到6618624那片内存,取出存放的数据6618636(*q中存放的数据),完成了 * (q)操作,再从内存中找到6618636那片内存,取出存放的数据5 ( *( *(q))中存放的数据 ),完成了 *(* (q))操作,再中的存放数据5(具体数据) (几个*查找几层)


  • ***m = 5
  • 先看成 *(*( * (m))),再从里向外看
  • 先从内存中拿到m存放的数据6618616(m中的数据),再从内存中找到6618616那片内存,取出存放的数据6618624(*q中存放的数据),完成了 * (m)操作,再从内存中找到6618624那片内存,取出存放的数据6618636 ( *( *(m)) 中存放的数据),完成了 *(* (m))操作,再从内存中找到6618636 那片内存,取出存放的数据5( *(*( *(m))) 中存放的数据),完成了 *(*( *(m)))操作, (几个*查找几层)

相关推荐

如何为MySQL服务器和客户机启用SSL?

用户想要与MySQL服务器建立一条安全连接时,常常依赖VPN隧道或SSH隧道。不过,获得MySQL连接的另一个办法是,启用MySQL服务器上的SSL封装器(SSLwrapper)。这每一种方法各有其...

OpenVPN客户端配置_openvpn客户端配置文件解析

...

k8s 证书问题排查_k8s dashboard 证书

从去年开始一些老项目上陆陆续续出现一些列的证书问题,(证书原理这里就不说了,官方文档一堆)多数刚开始的表现就是节点的kubelet服务起不来,节点状态NotReady表现日志如下failed...

企业级网络互通方案:云端OpenVPN+爱快路由器+Win11互联实战

企业级网络互通方案:OpenVPN搭建公有云+爱快路由器+Win11三地互联实战指南「安全高效」三地局域网秒变局域网实施环境说明...

OpenV** Server/Client配置文件详解

Server配置详解...

接口基础认知:关键信息与合规前提

1.核心技术参数(必记)...

S交换机通过SSH登录设备配置示例(RADIUS认证+本地认证独立)

说明:●本示例只介绍设备的认证相关配置,请同时确保已在RADIUS服务器上做了相关配置,如设备地址、共享密钥、创建用户等配置。●通过不同的管理域来实现RADIUS认证与本地认证两种方式同时使用,两...

SSL证书如何去除私钥密码保护_ssl证书怎么取消

有时候我们在生成证书的时候可以加入了密码保护。然后申请到证书安装到了web服务器。但是这样可能会带来麻烦。每次重启apache或者nginx的时候,都需要输入密码。那么SSL证书如何去除私钥密码保护。...

SSL证书基础知识与自签名证书生成指南

一、证书文件类型解析...

S交换机通过SSH登录设备配置示例(RADIUS认证)

说明:本示例只介绍设备的认证相关配置,请同时确保已在RADIUS服务器上做了相关配置,如设备地址、共享密钥、创建用户等配置。假设已在RADIUS服务器上创建了用户名yc123,密码test#123。对...

HTTPS是什么?加密原理和证书。SSL/TLS握手过程

秘钥的产生过程非对称加密...

HTTPS TLS握手流程_进行tls握手

1.客户端向服务器发送`ClientHello`消息,包括支持的TLS版本、加密套件、随机数等信息。2.服务器收到`ClientHello`消息后,解析其中的信息,并根据配置选择一个加密套件。3....

Spring Boot 单点登录(SSO)实现_spring boot 单点登录jwt

SpringBoot单点登录(SSO)实现全指南单点登录(SingleSign-On,SSO)是一种身份验证机制,允许用户使用一组凭证登录多个相关但独立的系统。在微服务架构和企业级系统中,SS...

源码分享:在pdf上加盖电子签章_pdf如何加盖电子公章

在pdf上加盖电子签章,并不是只是加个印章图片,。而是要使用一对密钥中的私钥对文件进行签字。为啥要用私钥呢?很简单,因为公钥是公开的,其他人才可以用公钥为你证明,这个文件是你签的。这就是我们常说的:私...

微信支付商户API证书到期 怎么更换

微信支付商户API证书到期更换是一个非常重要的操作,需要仔细按照流程进行。如果证书过期,所有通过API的支付、退款等操作都会失败,将直接影响您的业务。请按照以下详细步骤进行操作:重要前提:分清...