C 语言—— 数组
【C 语言】数组
- 1. 概念
- 2. 声明
- 3. 分类
- 4. 初始化
- 5. 赋值
- 6. 附加语法
- 7. VLA 的一些补充
1. 概念
数组是存放一组 相同类型 的 有序 数据的一段 连续 空间。
2. 声明
TYPE identifier[static(optional) qualifiers(optional) expression(optional)]
TYPE identifier[qualifiers(optional) static(optional) expression(optional)]
// farr 是一个包含 20 个元素的数组,元素的类型是 float
// pfarr 是一个包含 10 个元素的数组,数组的类型是 指向 float 的指针
float farr[20], *pfarr[10];
3. 分类
数组分为已知常量长度数组,变长度数组,以及未知大小数组。
1)expression 为整数常量表达式,则声明为已知常量长度数组:
// 声明一个包含 20 个类型为 float 的元素的数组,
// 整数常量 20 是常量表达式
float farr[20];// sizeof 是常量表达式
char text[sizeof(double)];// 枚举常量也是常量表达式
enum { MAX_SIZE = 100 };
int narr[MAX_SIZE];
2)expression 不是整数常量表达式,则声明为可变长度数组(VLA):
int n = 0;while(n++ < 10) {// 每次控制流经过该声明时会重新声明数组int a[n * 2];printf("The array has %zu elements\n", sizeof(a)/sizeof(*a));// 离开作用域 VLA 结束其生命周期
}
使用 * 作为 expression 时,声明为未指定长度的数组。这种声明只能出现在函数原型声明中。
void foo(size_t x, int a[*]);
void foo(size_t x, int a[x]) {// 这里 sizeof(a) 的大小等同于 sizeof(int*)printf("%zu\n", sizeof(a));
}
- 若忽略 expression ,则声明为未知大小数组。
未知大小数组 区别于 可变长度数组 的地方在于:可变长度数组 在其生命周期内,数组大小是不变的。
// 未知长度
extern int xarr[];
// 长度为 3
int iarr[] = {0, 1, 2};
未知大小数组可以作为 struct 的最后一个成员
struct s {int n;double d[];
};void func() {struct s *s1 = malloc(sizeof(struct s) + sizeof(double) * 8);//...
}
4. 初始化
- 列表初始化
int iarr0[] = {1, 2, 3}; // 3 个元素:1,2,3
int iarr1[5] = {1, 2, 3}; // 5 个元素:1,2,3,0,0
int iarr2[3] = {1};// 3 个元素:1, 0, 0
int iarr3[3] = {1, 2, 3, 4, 5}; // 3 个元素:1,2,3.
字符数组还可以使用字符串字面值来初始化。
// 10 个元素:
// 'H', 'e', 'l', 'l', 'o',
// '\0', '\0', '\0', '\0', '\0'
char str0[10] = "Hello";
- 使用指派表达式
使用指派表达式时,会先依次初始化指派表达式之前的元素(如果有的话),然后再初始化指派表达式指定的位置,最后再依次初始化指派表达式之后的位置。
有点绕,看下面例子:
// 初始化后的 iarr0:
// 1, 0, 0, 0, 0,
// 7, 0, 0, 11, 3,
// 9, 0, 0, 0, 0
int iarr0[15] = {1, [5] = 7, [8] = 11, 3, 9};
5. 赋值
通过下标可以对数组元素进行赋值。
int iarr0[] = {1, 2, 3};
// 现在是:1,10,3
iarr0[1] = 10;
注意,尽管数组可以进行取值,但数组类型的对象并不是可修改类型,不可以作为左值。
int iarr1[3] = {1, 2, 3};
int iarr2[3] = {4, 5, 6};
// 取地址,可。
int (*parr)[3] = &iarr1;
// 对数组赋值,不可。
iarr1 = iarr2;
6. 附加语法
数组类型声明可以使用 const、volatile 或者 restrict 限定符来进行修饰,在 C23 之前此时数据类型无限定,但数组元素类型有限定,C23 开始数组类型与其元素类型有等同限定:
typedef int A[2][3];
const A a = {{4, 5, 6}, {7, 8, 9}};
// int const[3] 转 int *,不可。
int *pi = a[0];
// int const[2][3] 转 void *,
// 在 gcc 12.1 可以通过编译,
// 在 clang 13.1.6 无法通过编译。
void *unqual_ptr = a;
7. VLA 的一些补充
可变长数组类型(Variable Length Array, VLA)和指向可变长数组的指针类型称为可变修改类型(Variably Modified, VM)。
- 任何可变修改类型的对象只能声明于 块作用域 或 函数原型作用域 中。
#define MAX_SIZE 100
extern int n;// ...// VLA 声明在文件作用域,不可
int A[n];
// 可
int B[MAX_SIZE];
// 指向 VLA 的指针,不可
extern int (*p2)[n];
// VLA 声明于函数原型作用域,可
void fvla(int m, int C[m]);
- VLA 必须拥有自动或分配存储期。指向 VLA 的指针可以有静态存储期。 VM 类型不能拥有链接。
(自动即定义在函数内并且不带auto以外存储类标识符修饰的对象;分配即通过 malloc、calloc、realloc 获得内存)
// int C[m] :指向 VLA 的指针(自动存储期);块作用域。可。
void func(int m, int C[m]) {// 块作用域,可。typedef int I_VLA[m][m];I_VLA ivla;// 块作用域;自动存储期。可。int D[m];// 块作用域;但静态。不可。static int E[m];// VLA 链接,不可。extern int F[m];// 块作用域;分配存储期。可。int (*s)[m];s = malloc(m * sizeof(int));// VM 链接,不可。extern int (*r)[m];// 指向 VLA 的静态指针。可。static int (*q)[m] = &B;
}
- VM 不能作为联合(union)或者结构(struct)的成员。
void func(void) {int n = 5;// 错误的例子struct tag {// VLA 作为结构体成员,不可。int z[n];// VM 作为结构体成员,不可。int (*y)[n];};
}
相关文章:
C 语言—— 数组
【C 语言】数组1. 概念2. 声明3. 分类4. 初始化5. 赋值6. 附加语法7. VLA 的一些补充1. 概念 数组是存放一组 相同类型 的 有序 数据的一段 连续 空间。 2. 声明 TYPE identifier[static(optional) qualifiers(optional) expression(optional)] TYPE identifier[qualifiers(o…...
Oracle-RAC集群主机重启问题分析
问题背景: 在对一套两节点Oracle RAC19.18集群进行部署时,出现启动数据库实例就会出现主机出现重启的情况,检查发现主机重启是由于节点集群被驱逐导致。 问题: 两节点Oracle RAC19.18集群,启动数据库实例会导致主机出现重启。 问题分析: 主机多次出现…...
Python每日一练(20230227)
目录 1. 路径交叉 ★★★ 2. 缺失的第一个正数 ★★★ 3. 寻找两个正序数组的中位数 ★★★ 附录 散列表 基本概念 常用方法 1. 路径交叉 给你一个整数数组 distance 。 从 X-Y 平面上的点 (0,0) 开始,先向北移动 distance[0] 米,然后向西移…...
Scratch少儿编程案例-算法练习-存款收益计算
专栏分享 点击跳转=>Unity3D特效百例点击跳转=>案例项目实战源码点击跳转=>游戏脚本-辅助自动化点击跳转=>Android控件全解手册点击跳转=>Scratch编程案例👉关于作者...
【Linux驱动开发100问】Linux驱动开发工程师在面试中常被问到的问题汇总
🥇今日学习目标:什么是Kconfig?如何使用Kconfig? 🤵♂️ 创作者:JamesBin ⏰预计时间:10分钟 🎉个人主页:嵌入式悦翔园个人主页 🍁专栏介绍:Lin…...
每日学术速递2.27
CV - 计算机视觉 | ML - 机器学习 | RL - 强化学习 | NLP 自然语言处理 Subjects: cs.CL 1.FiTs: Fine-grained Two-stage Training for Knowledge-aware Question Answering 标题:FiTs:用于知识感知问答的细粒度两阶段训练 作者:Qichen…...
【数据库系统概论】基础知识总结
🌹作者:云小逸 📝个人主页:云小逸的主页 📝Github:云小逸的Github 🤟motto:要敢于一个人默默的面对自己,强大自己才是核心。不要等到什么都没有了,才下定决心去做。种一颗树,最好的时间是十年前…...
简单移动平均在量化中的应用(附Python实战代码)
在大多数金融产品的投资过程中,均线系统都是很重要的投资参考。一般来说,均线可以近似理解为某段时间内成交筹码的均价,它往往能帮助我们找到合适的支撑位和压力位。随着各种技术流派以及统计学的发展,从简单移动平均中逐渐衍生出了更多的均线计算方式,比如指数移动平均、…...
ChatGPT提高你日常工作的五个特点,以及如何使用它来提高代码质量
ChatGPT已经完全改变了代码开发模式。然而,大多数软件开发者和数据专家们仍然不使用ChatGPT来完善——并简化他们的工作。 这就是我们在这里列出提升日常工作效率和质量的5个不同的特点的原因。 让我们一起来看看在日常工作中如何使用他们。 警告:不要…...
spark datasourceV1和v2
datasourceV2 一文理解 Apache Spark DataSource V2 诞生背景及入门实战 https://zhuanlan.zhihu.com/p/83006243 2.3 Data source API v2 https://issues.apache.org/jira/browse/SPARK-15689 Because of the above limitations/issues, the built-in data source impleme…...
10种聚类算法的完整python操作示例
大家好,聚类或聚类分析是无监督学习问题。它通常被用作数据分析技术,用于发现数据中的有趣模式,例如基于其行为的客户群。有许多聚类算法可供选择,对于所有情况,没有单一的最佳聚类算法。相反,最好探索一系…...
构建合作伙伴生态系统刻不容缓
合作伙伴关系管理(PRM)系统是否已死?向合作伙伴生态系统的转变将如何改变我们未来管理合作伙伴计划的方式? 自PC革命以来,间接销售和渠道营销一直普遍存在于技术领域,通过其他公司的销售团队和人脉来增加销售,是一种明…...
剑指 Offer 55 - I. 二叉树的深度(java解题)
剑指 Offer 55 - I. 二叉树的深度(java解题)1. 题目2. 解题思路3. 数据类型功能函数总结4. java代码1. 题目 输入一棵二叉树的根节点,求该树的深度。从根节点到叶节点依次经过的节点(含根、叶节点)形成树的一条路径&a…...
威胁行为者将旧漏洞武器化以发起勒索软件攻击
勒索软件运营商比以往任何时候都更加依赖未打补丁的系统来获得对受害者网络的初始访问权限。 一份新报告显示,攻击者正在互联网和暗网中积极搜索可用于勒索软件攻击的旧漏洞和已知漏洞。 其中许多缺陷已存在多年,对尚未修补或更新易受攻击系统的组织构…...
2023北京健博会/第十届中国国际大健康产博览会
China-DJK北京健博会,立足北京打造国内外大健康产业快速融合发展平台; 大健康时代:20年前没有健康产业,如今健康产业成了全球经济中唯“不缩水”的行业,早已被国际经济学界确定为“无限广阔的兆亿产业”。据机构数据&…...
Python学习笔记之环境搭建
Python学习笔记之环境搭建1. 下载Python2. Windows 安装最新Python3. Linux 安装最新PythonPython是一种编程语言,可以让您更快地工作并更有效地集成系统。 您可以学习使用Python,并立即看到生产力的提高和维护成本的降低。 Python是荷兰程序员吉多范罗苏…...
死锁的总结
哲学家死锁造成的原因:我有你需要的,但你已经有了 饥饿与死锁的区别 死锁一旦发生一定又饥饿现象,但是饥饿现象产生不一定是死锁 历史上对于死锁的声音 死锁的方案 前面两个都是不允许死锁出现 前面都是概念性的东西 后面我们研究如何破坏…...
强化学习RL 01~ 数学基础
目录 RL理解要点 1. RL数学基础 1.1 Random Variable 随机变量 1.2 概率密度函数 Probability Density Function(PDF) 1.3 期望 Expectation 1.4 随机抽样 Random Sampling 2. RL术语 Terminologies 2.1 agent、state 和 action 2.2 策略 policy π 2.3 奖励 reward …...
Java的运算符
目录 一、什么是运算符 二、算术运算符 1. 基本四则运算符:加减乘除模( - * / %) 2、增量运算符 - * % 3. 自增/自减运算符 -- 三、关系运算符 四、 逻辑运算符(重点) 1. 逻辑与 && 2. 逻辑或 || 3. 逻辑非 ! 4. 短路求值…...
扫地机器人(蓝桥杯C/C++)
题目描述 小明公司的办公区有一条长长的走廊,由 NN 个方格区域组成,如下图所示。 走廊内部署了 KK 台扫地机器人,其中第 ii 台在第 A_iAi 个方格区域中。已知扫地机器人每分钟可以移动到左右相邻的方格中,并将该区域清扫干净。…...
<6>-MySQL表的增删查改
目录 一,create(创建表) 二,retrieve(查询表) 1,select列 2,where条件 三,update(更新表) 四,delete(删除表…...
iview框架主题色的应用
1.下载 less要使用3.0.0以下的版本 npm install less2.7.3 npm install less-loader4.0.52./src/config/theme.js文件 module.exports {yellow: {theme-color: #FDCE04},blue: {theme-color: #547CE7} }在sass中使用theme配置的颜色主题,无需引入,直接可…...
日常一水C
多态 言简意赅:就是一个对象面对同一事件时做出的不同反应 而之前的继承中说过,当子类和父类的函数名相同时,会隐藏父类的同名函数转而调用子类的同名函数,如果要调用父类的同名函数,那么就需要对父类进行引用&#…...
GraphQL 实战篇:Apollo Client 配置与缓存
GraphQL 实战篇:Apollo Client 配置与缓存 上一篇:GraphQL 入门篇:基础查询语法 依旧和上一篇的笔记一样,主实操,没啥过多的细节讲解,代码具体在: https://github.com/GoldenaArcher/graphql…...
2025年低延迟业务DDoS防护全攻略:高可用架构与实战方案
一、延迟敏感行业面临的DDoS攻击新挑战 2025年,金融交易、实时竞技游戏、工业物联网等低延迟业务成为DDoS攻击的首要目标。攻击呈现三大特征: AI驱动的自适应攻击:攻击流量模拟真实用户行为,差异率低至0.5%,传统规则引…...
数据库正常,但后端收不到数据原因及解决
从代码和日志来看,后端SQL查询确实返回了数据,但最终user对象却为null。这表明查询结果没有正确映射到User对象上。 在前后端分离,并且ai辅助开发的时候,很容易出现前后端变量名不一致情况,还不报错,只是单…...
Python 高级应用10:在python 大型项目中 FastAPI 和 Django 的相互配合
无论是python,或者java 的大型项目中,都会涉及到 自身平台微服务之间的相互调用,以及和第三发平台的 接口对接,那在python 中是怎么实现的呢? 在 Python Web 开发中,FastAPI 和 Django 是两个重要但定位不…...
RFID推动新能源汽车零部件生产系统管理应用案例
RFID推动新能源汽车零部件生产系统管理应用案例 一、项目背景 新能源汽车零部件场景 在新能源汽车零部件生产领域,电子冷却水泵等关键部件的装配溯源需求日益增长。传统 RFID 溯源方案采用 “网关 RFID 读写头” 模式,存在单点位单独头溯源、网关布线…...
论文笔记:Large Language Models for Next Point-of-Interest Recommendation
SIGIR 2024 1 intro 传统的基于数值的POI推荐方法在处理上下文信息时存在两个主要限制 需要将异构的LBSN数据转换为数字,这可能导致上下文信息的固有含义丢失仅依赖于统计和人为设计来理解上下文信息,缺乏对上下文信息提供的语义概念的理解 ——>使用…...
leetcode 386. 字典序排数 中等
给你一个整数 n ,按字典序返回范围 [1, n] 内所有整数。 你必须设计一个时间复杂度为 O(n) 且使用 O(1) 额外空间的算法。 示例 1: 输入:n 13 输出:[1,10,11,12,13,2,3,4,5,6,7,8,9]示例 2: 输入:n 2…...
