【C语言】走进指针世界(下卷)
前言
在“走进指针世界(上卷)”中,我们已经说过:什么是指针、内存和地址,指针的使用、声明、初始化,取地址运算符、解引用运算符以及这两者关系,还有指针赋值。
在正式使用指针进行各种代码的编写之前,在这篇“走进指针世界(下卷)”里,让我们再了解一些指针的重要的前置知识吧。
指针变量的大小
指针变量有大小吗? 答案是有的。我们知道,sizeof是一个操作符, 其作用是返回一个对象或类型所占的内存字节数。就像整型变量、字符型变量等可以用sizeof计算大小一样,指针变量也可以这样计算大小:
#include<stdio.h>int main()
{int a = 10;int* p = &a;printf("%zd\n", sizeof(p));return 0;
}
在vs2022的x86(32位环境)下我们可以得出这样的运行结果:

如果改成x64(64位环境),就是这样的运行结果:

所以我们知道指针变量p的大小就是4或8个字节。
指针变量大小与指向类型的关系:
p是指向整型变量的指针变量,那指向字符型变量的指针变量有多大呢?我们可以再用同样的方式进行观察:
char a = 'a';
char* p = &a;
printf("%zd\n", sizeof(p));
return 0;
x86(32位环境)下结果:

x64(64位环境)下结果:

可以看到指向字符型变量的指针变量的大小同样是4或8个字节。我们可以继续去观察指向其他类型变量的指针变量的大小,最终会发现都是4或8个字节(取决于环境)。
为什么呢?

明白这个问题我们必须知道地址是怎么产生的。
计算机中的编址,是通过硬件设计完成的。
计算机中有很多的硬件单元,硬件单元之间互相协同工作,而协同工作需要相互之间能够进行数据传递。硬件之间相互独立,它们的通信方式是用“线”连起来。CPU和内存之间有大量的数据交互,所以这两者也用线连起来。
32位机器有32根地址总线,每根线只有两态,表示为0或1(电脉冲有无),那么一根线可以表示两种含义,32根线就可以表示2^32种含义,每种含义可以表示一个地址。
既然把32根地址线产生的2进制序列当作一个地址,那么一个地址就是32个bit位,需要4个字节来存储。所以,指针变量的大小就是4个字节。
在这个分析的过程中,我们可以看出,指针变量的大小与该指针指向的数据类型确实是无关的,只与是32位环境还是64位环境有关。
指针变量类型的意义
既然指针变量的大小都是4或8个字节,那么你可能会好奇指针变量类型的意义在哪呢?现在我们就来了解一下指针变量类型的意义。
对于解引用的意义:
第一个方面的意义在于解引用,指针类型的意义决定了在对指针解引用时有多大的权限,或者说一次能操作几个字节:

*pa=0之后:

从内存窗口我们可以看出,在对指向int类型的指针变量pa解引用并赋值0时,我们把4个字节(00为一个字节)都该为了0。
如果将同样的动作施加于一个指向char类型的指针变量,会发生什么?

*pa=0之后:

可以看到,我们只能操作一个字节。
所以当指针变量指向的类型不同,即指针变量的类型也不同时,解引用的权限可能是不同的。
对于指针加减整数的意义:
指针变量的类型还决定了指针加减一个整数时,实际移动的字节数。或者通俗点说,指针向前或向后走一步有多大距离。

我们在vs的x86下,用%p打印观察地址。发现同一个整型变量,我们用两个不同类型的指针存储它的地址,在加1后地址的变化不相同。我们知道一个内存单元为1个字节,每个字节有自己的编号即地址,所以pa+1加了4个字节,而pc+1加了1个字节。
其实,pa+1时,加的并不是整数1,而是1*sizeof(int),pc+1时,加的是1*sizeof(char)。
void*指针
有一种特殊的指针类型,void*指针,也就是无具体类型的指针,也叫泛型指针。
这种指针有其优缺点。
优点:
它的优点就是既然是泛型指针,在接收一个地址的时候无需局限于某个类型的地址,可以把任意类型数据的地址交给它。比如下面这个代码是合法的:
int a = 10;
void* p = &a;
当我们遇到不知道具体类型的地址又需要用指针存储时,void*指针就可以派上用场。
缺点:
但是它也有自己的缺点,就是无法直接进行指针的加减整
数运算和解引用运算,因为它不知道要操作几个字节。
int a = 10;
void* p = &a;
printf("%d\n", *p);//会报错
const修饰指针
有时,我们不希望指针指向的内容被修改,那么我们就可以使用const对其进行修饰。
还记得const的用法吗?
int a = 100;
a = 200;
在这个代码中,我们把a的值改变了。
const int a = 100;
a = 200;//会报错
而我们用const修饰变量a时,a变成了“常变量”,本质上还是个变量,但是不能被修改。
那么现在我们再看一个代码:

我们用const修饰变量n的声明时,我们本意是希望n的值不会被修改,而通过指针我们却间接改掉了变量n的值,那么我们怎样才能达到我们的预期呢?
当我们在指针变量p的声明时用cconst进行修饰,就无法再通过刚才的方式修改n的值,而是会报错了。
const与*的前后关系
在声明一个指针变量并想要用const进行修饰时,我们需要注意的一点是const和*的先后关系。因为当const放在*前面时,无法被修改的是指针指向的内容,当const放在*后面时,无法改变的是指针的指向:

注意,const放在*前面时,在int前还是后都是一样的效果:

而当const放在*后面时,才会有根本的改变:

还有一种情况,当我们在*的前后都放上const,那就变成了指向和指向的内容都无法修改:

总之,可以根据我们不希望被修改的内容来决定const的用法,并且在需要使用const的时候不要吝啬使用,因为它可以在我们不小心修改了不应该修改的内容时及时给出警告。
那么,到此为止,“走进指针世界”就结束了,后面我还会持续更新指针相关的更多内容,希望大家发现错误可以向我指正^_^
相关文章:
【C语言】走进指针世界(下卷)
前言 在“走进指针世界(上卷)”中,我们已经说过:什么是指针、内存和地址,指针的使用、声明、初始化,取地址运算符、解引用运算符以及这两者关系,还有指针赋值。 在正式使用指针进行各种代码的…...
【Spring】SSM整合_入门代码实现
1. Maven依赖 在pom.xml中添加SSM框架的依赖 <!-- Spring Core --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.3.x</version> </dependency>…...
C++代码错误解决1(函数模板)
1、代码如下 //示例函数模板的使用 #include <iostream> #include <string> using namespace std; template <typename T>//函数模板 T max(T a,T b) {return a>b?a:b; } int main() {int a,b;cout<<"input two integers to a&b:"…...
idea configuration 配置 方便本地启动环境切换
idea 再项目启动的时候避免切换环境导致上线的时候出现环境配置问题 可以再idea 的 configuration 中配置项目的 vm options 虚拟机的内容占用 -Xmx256m -Xms256m -Xmn100m -Xss256k program arguments properties 文件中需要修改的配置参数 active profiles 指定启动的本…...
win10配置wsl的深度学习环境
# 1、一步完成wsl:开启虚拟机、linux子系统、并下载ubuntu # 官方文档: https://learn.microsoft.com/zh-cn/windows/wsl/install wsl --install# 2、打开windows terminal,选ubuntu交互环境 # 第一次需要配置用户名和密码 # 接下来正常使用即可# 3、cud…...
如何处理时间序列的缺失数据
您是否应该删除、插入或估算? 世界上没有完美的数据集。每个数据科学家在数据探索过程中都会有这样的感觉: df.info()看到类似这样的内容: 大多数 ML 模型无法处理 NaN 或空值,因此如果您的特征或目标包含这些值,则在…...
fastapi中实现多个路由请求
大家伙,我是雄雄,欢迎关注微信公众号:雄雄的小课堂。 前言 最近在写机器人相关的接口,顺手学了学python,发现这是个好东西,写代码效率比java要高很多,比如写个词云呀,写个回调呀&am…...
前端框架选择指南:React vs Vue vs Angular
选择前端框架时,React、Vue 和 Angular 都是流行的选择,各有优缺点。我们可以从各个维度进行比较和选择: React 核心理念: 组件化开发,专注于视图层。学习曲线: 相对平缓,因为重点在于JSX和组…...
猫头虎 解析:为什么AIGC在国内适合做TOB,在国外适合做TOC?
猫头虎 解析:为什么AIGC在国内适合做TOB,在国外适合做TOC? 博主 猫头虎 的技术世界 🌟 欢迎来到猫头虎的博客 — 探索技术的无限可能! 专栏链接: 🔗 精选专栏: 《面试题大全》 — 面…...
并发编程笔记8--ThreadLocal结构详解
ThreadLocal,即线程变量,是一个以ThreadLocal对象为键,任意对象为值的存储结构。这个结构被附带在线程上,也就是说一个线程可以根据一个ThreadLocal对象查询到绑定在这个线程上的值。可以通过set(T)方法来设置一个值,在…...
强烈推荐 20.7k Star!企业级商城开源项目强烈推荐!基于DDD领域驱动设计模型,助您快速掌握技术奥秘,实现业务快速增长
更多资源请关注纽扣编程微信公众号 1 项目简介 商城是个从零到一的C端商城项目,包含商城核心业务和基础架构两大模块,推出用户、消息、商品、订单、优惠券、支付、网关、购物车等业务模块,通过商城系统中复杂场景,给出对应解决方案。使用 …...
【C++STL详解(四)------vector的模拟实现】
文章目录 vector各函数接口总览vector当中的成员变量介绍默认成员函数构造函数1构造函数2构造函数3拷贝构造函数赋值运算符重载函数析构函数 迭代器相关函数begin和end 容量和大小相关函数size和capacityreserveresizeempty 修改容器内容相关函数push_backpop_backinserterases…...
租赁系统|北京租赁系统|租赁软件开发流程
在数字化时代的浪潮下,小程序成为了各行各业争相探索的新领域。租赁行业亦不例外,租赁小程序的开发不仅提升了用户体验,更为商家带来了更多商业机会。本文将详细解析租赁小程序的开发流程,为有志于进军小程序领域的租赁行业从业者…...
JAVA面试题大全(十四)
1、Kafka 可以脱离 Zookeeper 单独使用吗?为什么? kafka不能脱离zookper单独使用,因为kafka使用zookper管理和协调kafka的节点服务器。 2、Kafka 有几种数据保留的策略? Kafka提供了多种数据保留策略,这些策略用于定…...
Web Accessibility基础:构建无障碍的前端应用
Web Accessibility(网络无障碍)是确保所有人都能平等访问和使用网站和应用程序的关键。这包括视觉、听觉、运动和认知能力有限的用户。以下是一些构建无障碍前端应用的基础原则和代码示例: 2500G计算机入门到高级架构师开发资料超级大礼包免…...
谈谈你对 SPA 的理解?
1 理解基本概念 SPA(single-page application)单页应用,默认情况下我们编写 Vue、React 都只有一个html 页面,并且提供一个挂载点,最终打包后会再此页面中引入对应的资源。(页面的渲染全部是由 JS 动态进行…...
JAVA给一个JSON数组添加对象
操作Mysql表的json字段,查询json字段的内容,将新增的内容添加到查询的json数组中 String a "[{\"name\": \"张三\", \"age\": 10, \"gender\": \"男\", \"email\": \"123qq.co…...
设计一个完美的用户角色权限表
设计一个完美的用户角色权限表需要考虑系统的安全性、灵活性和可扩展性。以下是一个详细的用户角色权限管理表设计方案,包含多个表结构和字段描述。 目录 1. 用户表(Users Table)2. 角色表(Roles Table)3. 权限表&…...
Git 基本使用
目录 Git 安装与设置 在 Windows上安装 Git git 的配置 Git 原理 git 的四个区域 git 工作流程 git 文件的状态 Git 操作 创建仓库 免密登录 基本操作 版本回退 本地仓库整理 分支命令 合并分支 解决冲突 Git 安装与设置 在 Windows上安装 Git 在 Windows上使…...
LabVIEW使用PID 控制器有哪些应用场景?
如何在LabVIEW中创建PID控制器? LabVIEW为各种控制工程任务提供了内置函数和库,包括PID控制器编程。这些功能位于控制设计和仿真调色板中,其中有用于不同类型控制器的子调色板。要在LabVIEW中创建PID控制器,需要将PID函数从PID子调色板拖放…...
vscode里如何用git
打开vs终端执行如下: 1 初始化 Git 仓库(如果尚未初始化) git init 2 添加文件到 Git 仓库 git add . 3 使用 git commit 命令来提交你的更改。确保在提交时加上一个有用的消息。 git commit -m "备注信息" 4 …...
Unity3D中Gfx.WaitForPresent优化方案
前言 在Unity中,Gfx.WaitForPresent占用CPU过高通常表示主线程在等待GPU完成渲染(即CPU被阻塞),这表明存在GPU瓶颈或垂直同步/帧率设置问题。以下是系统的优化方案: 对惹,这里有一个游戏开发交流小组&…...
如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...
DAY 47
三、通道注意力 3.1 通道注意力的定义 # 新增:通道注意力模块(SE模块) class ChannelAttention(nn.Module):"""通道注意力模块(Squeeze-and-Excitation)"""def __init__(self, in_channels, reduction_rat…...
数据链路层的主要功能是什么
数据链路层(OSI模型第2层)的核心功能是在相邻网络节点(如交换机、主机)间提供可靠的数据帧传输服务,主要职责包括: 🔑 核心功能详解: 帧封装与解封装 封装: 将网络层下发…...
(转)什么是DockerCompose?它有什么作用?
一、什么是DockerCompose? DockerCompose可以基于Compose文件帮我们快速的部署分布式应用,而无需手动一个个创建和运行容器。 Compose文件是一个文本文件,通过指令定义集群中的每个容器如何运行。 DockerCompose就是把DockerFile转换成指令去运行。 …...
实现弹窗随键盘上移居中
实现弹窗随键盘上移的核心思路 在Android中,可以通过监听键盘的显示和隐藏事件,动态调整弹窗的位置。关键点在于获取键盘高度,并计算剩余屏幕空间以重新定位弹窗。 // 在Activity或Fragment中设置键盘监听 val rootView findViewById<V…...
select、poll、epoll 与 Reactor 模式
在高并发网络编程领域,高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表,以及基于它们实现的 Reactor 模式,为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。 一、I…...
在web-view 加载的本地及远程HTML中调用uniapp的API及网页和vue页面是如何通讯的?
uni-app 中 Web-view 与 Vue 页面的通讯机制详解 一、Web-view 简介 Web-view 是 uni-app 提供的一个重要组件,用于在原生应用中加载 HTML 页面: 支持加载本地 HTML 文件支持加载远程 HTML 页面实现 Web 与原生的双向通讯可用于嵌入第三方网页或 H5 应…...
Reasoning over Uncertain Text by Generative Large Language Models
https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829 1. 概述 文本中的不确定性在许多语境中传达,从日常对话到特定领域的文档(例如医学文档)(Heritage 2013;Landmark、Gulbrandsen 和 Svenevei…...
