进阶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),又称铁达尼号,是当时世界上体积最庞大、内部设施最豪华的客运轮船,有“永不沉没”的美誉ÿ…...

深入剖析AI大模型:大模型时代的 Prompt 工程全解析
今天聊的内容,我认为是AI开发里面非常重要的内容。它在AI开发里无处不在,当你对 AI 助手说 "用李白的风格写一首关于人工智能的诗",或者让翻译模型 "将这段合同翻译成商务日语" 时,输入的这句话就是 Prompt。…...

阿里云ACP云计算备考笔记 (5)——弹性伸缩
目录 第一章 概述 第二章 弹性伸缩简介 1、弹性伸缩 2、垂直伸缩 3、优势 4、应用场景 ① 无规律的业务量波动 ② 有规律的业务量波动 ③ 无明显业务量波动 ④ 混合型业务 ⑤ 消息通知 ⑥ 生命周期挂钩 ⑦ 自定义方式 ⑧ 滚的升级 5、使用限制 第三章 主要定义 …...

通过Wrangler CLI在worker中创建数据库和表
官方使用文档:Getting started Cloudflare D1 docs 创建数据库 在命令行中执行完成之后,会在本地和远程创建数据库: npx wranglerlatest d1 create prod-d1-tutorial 在cf中就可以看到数据库: 现在,您的Cloudfla…...
深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法
深入浅出:JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中,随机数的生成看似简单,却隐藏着许多玄机。无论是生成密码、加密密钥,还是创建安全令牌,随机数的质量直接关系到系统的安全性。Jav…...
Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务
通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...

【项目实战】通过多模态+LangGraph实现PPT生成助手
PPT自动生成系统 基于LangGraph的PPT自动生成系统,可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析:自动解析Markdown文档结构PPT模板分析:分析PPT模板的布局和风格智能布局决策:匹配内容与合适的PPT布局自动…...

(二)原型模式
原型的功能是将一个已经存在的对象作为源目标,其余对象都是通过这个源目标创建。发挥复制的作用就是原型模式的核心思想。 一、源型模式的定义 原型模式是指第二次创建对象可以通过复制已经存在的原型对象来实现,忽略对象创建过程中的其它细节。 📌 核心特点: 避免重复初…...
Python爬虫(二):爬虫完整流程
爬虫完整流程详解(7大核心步骤实战技巧) 一、爬虫完整工作流程 以下是爬虫开发的完整流程,我将结合具体技术点和实战经验展开说明: 1. 目标分析与前期准备 网站技术分析: 使用浏览器开发者工具(F12&…...

Mac软件卸载指南,简单易懂!
刚和Adobe分手,它却总在Library里给你写"回忆录"?卸载的Final Cut Pro像电子幽灵般阴魂不散?总是会有残留文件,别慌!这份Mac软件卸载指南,将用最硬核的方式教你"数字分手术"࿰…...

Springcloud:Eureka 高可用集群搭建实战(服务注册与发现的底层原理与避坑指南)
引言:为什么 Eureka 依然是存量系统的核心? 尽管 Nacos 等新注册中心崛起,但金融、电力等保守行业仍有大量系统运行在 Eureka 上。理解其高可用设计与自我保护机制,是保障分布式系统稳定的必修课。本文将手把手带你搭建生产级 Eur…...