进阶C语言——指针(二)【题目练习】
文章目录
- 1.指针和数组概念的理解
- 2.指针和数组笔试题解析
- 一维数组
- 字符数组
- 二维数组
1.指针和数组概念的理解
指针和数组
- 数组:能够存放一组相同类型的元素,数组的大小取决于数组的元素个数和元素类型
- 指针:也是地址或指针变量,大小是4或8个字节
- 数组是数组,指针是指针,二者不等价
- 数组名是数组首元素的地址,这个地址就可以存放在指针变量中,我们可以使用指针来遍历数组
数组名大部分情况下就是首元素地址,但是有两个例外:
sizeof(数组名) —— 数组名表示整个数组,计算的是整个数组的大小
&数组名 —— 数组名表示整个数组,取出的是数组的地址
2.指针和数组笔试题解析
一维数组
#include <stdio.h>
#include <stdio.h>
int main()
{int a[] = { 1,2,3,4 };printf("%d\n", sizeof(a));//16//sizeof(a)就是数组名单独放在sizeof内部,计算的数组总大小,单位是字节printf("%d\n", sizeof(a + 0));//4/8 个字节//a+0不是单独的数组名//a+0 其实是数组首元素的地址printf("%d\n", sizeof(*a));//4//a是数组首元素的地址 - &a[0]//*a -> *&a[0] -> a[0]printf("%d\n", sizeof(a + 1));//4/8//a是数组首元素的地址 -- int*//a+1 跳过1个整型, 是第二个元素的地址printf("%d\n", sizeof(a[1]));//4//数组第二个元素的大小printf("%d\n", sizeof(&a));//&a - 取出的是数组的地址,但是数组的地址也是地址呀,是地址大小就是4/8字节//int (*pa)[4] = &a;//int(*)[4]printf("%d\n", sizeof(*&a));//16//sizeof(a)//int(*)[4]printf("%d\n", sizeof(&a + 1));//4/8//&a --> int (*)[4]//&a+1 跳过一个数组printf("%d\n", sizeof(&a[0]));//取出首元素的地址 4/8printf("%d\n", sizeof(&a[0] + 1));//第二个元素的地址return 0;
}
运行结果:(x86环境下的运行结果)

a+1与&a+1的区别

字符数组
类型一
数组为arr[] = { ‘a’,‘b’,‘c’,‘d’,‘e’,‘f’ }的字符
#include <string.h>
int main()
{char arr[] = { 'a','b','c','d','e','f' };printf("%d\n", sizeof(arr));//6printf("%d\n", sizeof(arr + 0));//arr+0是数组首元素的地址 4/8printf("%d\n", sizeof(*arr));//*arr是首元素的,计算的是首元素的大小 1printf("%d\n", sizeof(arr[1]));//1printf("%d\n", sizeof(&arr));//&arr是数组的地址 4/8printf("%d\n", sizeof(&arr + 1));//&arr + 1跳过一个数组后的地址,4/8printf("%d\n", sizeof(&arr[0] + 1));//4/8 第二个元素的地址printf("%d\n", strlen(arr));//随机值,因为不知道\0的位置printf("%d\n", strlen(arr + 0));//随机值//传arr与arr+0传的地址一样//printf("%d\n", strlen(*arr));//非法访问//printf("%d\n", strlen(arr[1]));//'b' - 98 当成地址,形参非法访问printf("%d\n", strlen(&arr));//随机值printf("%d\n", strlen(&arr + 1));//随机值-6printf("%d\n", strlen(&arr[0] + 1));//随机值-1return 0;
}
*arr传的是字符'a','a'的本质是97,strlen就会把97当成地址来访问,从而造成非法访问
注:
- sizeof计算的是占用内存空间的大小,单位是字节,不关注内存中到底存放的是什么(上边代码中sizeof(arr)大小为6)
- sizeof不是函数,是操作符
- strlen是函数
- strlen是针对字符串的,求的是字符串的长度,本质上是统计 \0 之前出现的字符个数


类型二
数组为arr[] = "abcdef"的字符
#include <stdio.h>
#include <string.h>
int main()
{char arr[] = "abcdef";//内部存放的的是[a b c d e f \0]printf("%d\n", sizeof(arr));//7printf("%d\n", sizeof(arr + 0));//4/8printf("%d\n", sizeof(*arr));//*arr -是数组首元素 1//表示首元素写法:arr[0] *(arr+0)printf("%d\n", sizeof(arr[1]));//1printf("%d\n", sizeof(&arr));//数组的地址,是地址就是4 / 8printf("%d\n", sizeof(&arr + 1));//4 / 8printf("%d\n", sizeof(&arr[0] + 1));//4 / 8printf("%d\n", strlen(arr));//6printf("%d\n", strlen(arr + 0));//6//printf("%d\n", strlen(*arr));//err//printf("%d\n", strlen(arr[1]));//errprintf("%d\n", strlen(&arr));//6//&arr - char (*)[7]printf("%d\n", strlen(&arr + 1));//随机值printf("%d\n", strlen(&arr[0] + 1));//5return 0;
}
&arr与&arr+1

类型三
char* p = "abcdef"类型的指针变量
#include <string.h>
#include <stdio.h>
int main()
{char* p = "abcdef";printf("%d\n", sizeof(p));//4 / 8printf("%d\n", sizeof(p + 1));//'b'的地址,4/8printf("%d\n", sizeof(*p));//1printf("%d\n", sizeof(p[0]));//*(p+0)--'a' 1printf("%d\n", sizeof(&p));//4/8printf("%d\n", sizeof(&p + 1));//4/8printf("%d\n", sizeof(&p[0] + 1));//&p[0]+1是'b'的地址printf("%d\n", strlen(p));//6printf("%d\n", strlen(p + 1));//p+1是'b'的地址 5//printf("%d\n", strlen(*p));//err// *p得到的是a,传的是a的地址,无法访问//printf("%d\n", strlen(p[0]));//errprintf("%d\n", strlen(&p));//随机值printf("%d\n", strlen(&p + 1));//随机值printf("%d\n", strlen(&p[0] + 1));//从b向后传 5return 0;
}
&p与&p+1

二维数组
二维数组在内存中的实际存放形式

#include <stdio.h>
int main()
{int a[3][4] = { 0 };printf("%d\n", sizeof(a)); //48 - a这个二维数组的数组名单独放在sizeof内部,计算整个数组的大小printf("%d\n", sizeof(a[0][0]));//第一行第一个元素,4个字节printf("%d\n", sizeof(a[0]));//16//a[0] 第一行的数组名,这时数组名单独放在sizeof内部了//计算的是数组的大小,单位是字节,16printf("%d\n", sizeof(a[0] + 1));//a[0]不是单独放在sizeof内部,a[0]表示的首元素的地址,即第一行第一个元素的地址 - &a[0][0]//a[0] + 1 是第一行第2个元素的地址 &a[0][1]printf("%d\n", sizeof(*(a[0] + 1)));//a[0][1] 大小是:4个字节printf("%d\n", sizeof(a + 1));//a作为二维数组的数组名并非单独放在sizeof内部,所以表示首元素的地址//二维数组的首元素是第一行,这里的a就是第一行的地址--- int (*)[4]//a+1是跳过第一行,指向了第二行printf("%d\n", sizeof(*(a + 1)));//16//*(a+1)-->a[1]printf("%d\n", sizeof(&a[0] + 1));//4/8//&a[0]是第一行的地址//&a[0]+1是第二行的地址printf("%d\n", sizeof(*(&a[0] + 1)));//16 a[1]printf("%d\n", sizeof(*a));// 16 *a - 就是第一行//*a -- *(a+0) -- a[0]printf("%d\n", sizeof(a[3]));//16return 0;
}
打印结果:

sizeof(a[3])访问时不会存在数组访问越界吗?
答案:不会
我们用代码举例说明:
int main()
{int a = 5;short s = 11;printf("%d\n", sizeof(s = a + 2));//2printf("%d\n", s);//11return 0;
}
打印结果为:

代码运行的流程图

注:
任何一个表达式或值都有两个属性
值属性和类型属性
举例说明:
上文代码中表达式的类型属性是2,而编译期间根据类型属性就直接确定了大小为2,所以就不会再把a+2进行计算
好了,今天七七的分享就到这里了,如果这篇文章对大家有帮助,请佬佬们点个赞再走吧!如果发现什么问题,欢迎评论区留言!
相关文章:
进阶C语言——指针(二)【题目练习】
文章目录1.指针和数组概念的理解2.指针和数组笔试题解析一维数组字符数组二维数组1.指针和数组概念的理解 指针和数组 数组:能够存放一组相同类型的元素,数组的大小取决于数组的元素个数和元素类型指针:也是地址或指针变量,大小是…...
Ajax简介
Ajax简介和使用 1.简介 AJAX Asynchronous JavaScript and XML(异步的 JavaScript 和 XML)。 AJAX 是一种在无需重新加载整个网页的情况下,能够更新部分网页的技术。 Ajax 不是一种新的编程语言,而是一种用于创建更好更快以及…...
ChatGPT 4 测试 两数比较大小问题。
按: 上次用3.5 测试了ChatGPT的两数比较大小问题,结果失败了。我要求不能用if语句,它避免不了。这次终于成功了,看来是进步很大。对话记录如下(英文) MaraSun Compare two 2 numbers in C# , but IF is no…...
SSM-CRUD整合视频教程:Spring、SpringMVC、MyBatis、bootstrap、pagehelper、JSR303后端校验
1、项目说明 1.1、业务说明 SSM:SpringMVCSpringMyBatisCRUD: Create(创建)Retrieve(查询)Update(更新)Delete(删除) 总结:通过SSM框架来完成一个CRUD的操作。 1.2、功…...
Linux常用命令——基于Ubuntu22.04
本文介绍了一些Linux的常用命令。为了便于快速检索命令位置,文章二级标题都以“命令:命令的作用”展示,有些命令会先介绍命令的几个常用参数,然后结合具体的操作展示命令的使用。为了便于记忆,也会提到命令是由哪些短语…...
Sentinel
SentinelSentinel介绍什么是Sentinel?为什么需要流量控制?为什么需要熔断降级?一些普遍的使用场景本文介绍参考:Sentinel官网《Spring Cloud Alibaba 从入门到实战.pdf》Sentinel下载/安装项目演示构建项目控制台概览演示之前需先明确&#…...
再也不想去字节跳动面试了,6年测开面试遭到这样打击.....
前几天我朋友跟我吐苦水,这波面试又把他打击到了,做了快6年软件测试员。。。为了进大厂,也花了很多时间和精力在面试准备上,也刷了很多题。但题刷多了之后有点怀疑人生,不知道刷的这些题在之后的工作中能不能用到&…...
【深度解刨C语言】符号篇(全)
文章目录一.注释二.续行符与转义符1.续行符2.转义符三.回车与换行四.逻辑操作符五.位操作符和移位操作符六.前置与后置七.字符与字符串八./和%1.四种取整方式2.取模与取余的区别和联系3./两边异号的情况1.左正右负2.左负右正九.运算符的优先级一.注释 注释的两种符号ÿ…...
VS Code 将推出更多 AI 功能给 Java 开发者
大家好,欢迎来到我们的二月更新!我们将为您带来与 JUnit 5 并行测试相关的新功能以及用于 Spring Boot Dashboard 的过滤功能。另外,OpenAI 和 ChatGPT 是最近的热点,所以在 GitHub Copilot 方面也有一些令人激动的消息࿰…...
关于利用FFT分析时域信号幅相的思考与验证
引言 利用FFT分析/估计时域信号的幅度和相位,属于传统估计的范畴。估计的准确程度受频率分辨率的影响较大。如果被估计的目标频率等于频率分辨率的整数倍,信号的幅相估计都是最准确的。一旦目标频率不等于频率分辨率的整数倍,幅度估计值将会…...
基于java中的Springboot框架实现餐厅点餐系统展示
基于java中的Springboot框架实现餐厅点餐系统开发语言和工具 开发语言:Java 框架:springboot JDK版本:JDK1.8 服务器:tomcat7 数据库:mysql 5.7 21世纪的今天,随着社会的不断发展与进步,人们对…...
案例07-在线人员列表逻辑混乱
一、背景介绍 在线人员列表涉及到的问题: 类中写了公共变量最后导致数据混乱现象 保存数据没有考虑业务的隔夜覆盖导致的逻辑漏洞 涉及到继承,对于this,如果父类有同样的成员最终使用哪一个? 参数不一致导致后续维护混乱 mysql由…...
Java集合框架
Java集合框架是Java编程语言所提供的一种便捷的数据结构的实现。Java集合框架提供了一种统一的接口和机制来访问和操作集合中的元素,这些元素可以是对象、基本数据类型或其他集合。Java集合框架是Java应用程序中最常用的特性之一,它为开发人员提供了许多…...
奇异值分解(SVD)原理与在降维中的应用
奇异值分解(SVD)原理与在降维中的应用 奇异值分解(Singular Value Decomposition,以下简称SVD)是在机器学习领域广泛应用的算法,它不光可以用于降维算法中的特征分解,还可以用于推荐系统,以及自然语言处理等领域。是很多机器学习算…...
GDB调试程序
1.GDB 调试程序 GDB是GNU开源组织发布的一个强大的UNIX下的程序调试工具。在UNIX平台下做软件,GDB这个调试工具有比VC的图形化调试器更强大的功能。所谓“寸有所长,尺有所短”就是这个道理。 一般来说,GDB主要帮忙你完成下面四个方面的功能…...
五种IO模型
用户空间与内核空间 操作系统把内存空间划分成了两个部分:内核空间和用户空间。 为了保护内核空间的安全,操作系统一般都限制用户进程直接操作内核。 所以,当我们使用TCP发送数据的时候,需要先将数据从用户空间拷贝到内核空间&a…...
5 全面认识java的控制流程
全面认识java控制流程1.块作用域2.条件语句3.迭代语句3.1while语句3.2do-while语句3.3for语句3.4 for-in语法4.中断控制流程的语句4.1 return4.2 break和continue4.2.1 不带标签的break语句4.2.2 带标签的break语句4.2.3 continue语句4.3 goto()5.多重选择:switch语句1.块作用域…...
第二章 测验【嵌入式系统】
第二章 测验【嵌入式系统】前言推荐第二章 测验【嵌入式系统】最后前言 以下内容源自《嵌入式系统》 仅供学习交流使用 推荐 第一章 测验【嵌入式系统】 第二章 测验【嵌入式系统】 1单选题 32bit宽的数据0x12345678 在小端模式(Little-endian)模式…...
排序算法之插入排序
要考数据结构了,赶紧来复习一波排序算法 文章目录一、直接插入排序二、希尔排序一、直接插入排序 直接上主题 插排,揪出一个数,插入到原本已经有序的数组里面,如数组有n个数据,从0~n下标依次排列,先从左往…...
Kaggle实战入门:泰坦尼克号生生还预测
Kaggle实战入门:泰坦尼克号生生还预测1. 加载数据2. 特征工程3. 模型训练4. 模型部署泰坦尼克号(Titanic),又称铁达尼号,是当时世界上体积最庞大、内部设施最豪华的客运轮船,有“永不沉没”的美誉ÿ…...
【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型
摘要 拍照搜题系统采用“三层管道(多模态 OCR → 语义检索 → 答案渲染)、两级检索(倒排 BM25 向量 HNSW)并以大语言模型兜底”的整体框架: 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后,分别用…...
深度学习在微纳光子学中的应用
深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向: 逆向设计 通过神经网络快速预测微纳结构的光学响应,替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…...
从WWDC看苹果产品发展的规律
WWDC 是苹果公司一年一度面向全球开发者的盛会,其主题演讲展现了苹果在产品设计、技术路线、用户体验和生态系统构建上的核心理念与演进脉络。我们借助 ChatGPT Deep Research 工具,对过去十年 WWDC 主题演讲内容进行了系统化分析,形成了这份…...
Spring Boot 实现流式响应(兼容 2.7.x)
在实际开发中,我们可能会遇到一些流式数据处理的场景,比如接收来自上游接口的 Server-Sent Events(SSE) 或 流式 JSON 内容,并将其原样中转给前端页面或客户端。这种情况下,传统的 RestTemplate 缓存机制会…...
华为云Flexus+DeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建
华为云FlexusDeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建 前言 如今大模型其性能出色,华为云 ModelArts Studio_MaaS大模型即服务平台华为云内置了大模型,能助力我们轻松驾驭 DeepSeek-V3/R1,本文中将分享如何…...
大学生职业发展与就业创业指导教学评价
这里是引用 作为软工2203/2204班的学生,我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要,而您认真负责的教学态度,让课程的每一部分都充满了实用价值。 尤其让我…...
大语言模型(LLM)中的KV缓存压缩与动态稀疏注意力机制设计
随着大语言模型(LLM)参数规模的增长,推理阶段的内存占用和计算复杂度成为核心挑战。传统注意力机制的计算复杂度随序列长度呈二次方增长,而KV缓存的内存消耗可能高达数十GB(例如Llama2-7B处理100K token时需50GB内存&a…...
【生成模型】视频生成论文调研
工作清单 上游应用方向:控制、速度、时长、高动态、多主体驱动 类型工作基础模型WAN / WAN-VACE / HunyuanVideo控制条件轨迹控制ATI~镜头控制ReCamMaster~多主体驱动Phantom~音频驱动Let Them Talk: Audio-Driven Multi-Person Conversational Video Generation速…...
DingDing机器人群消息推送
文章目录 1 新建机器人2 API文档说明3 代码编写 1 新建机器人 点击群设置 下滑到群管理的机器人,点击进入 添加机器人 选择自定义Webhook服务 点击添加 设置安全设置,详见说明文档 成功后,记录Webhook 2 API文档说明 点击设置说明 查看自…...
Caliper 负载(Workload)详细解析
Caliper 负载(Workload)详细解析 负载(Workload)是 Caliper 性能测试的核心部分,它定义了测试期间要执行的具体合约调用行为和交易模式。下面我将全面深入地讲解负载的各个方面。 一、负载模块基本结构 一个典型的负载模块(如 workload.js)包含以下基本结构: use strict;/…...

