- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我了解什么是可变长度数组以及它们是如何实现的。这个问题是关于它们为什么存在。
我们知道 VLA 只允许在功能块(或原型(prototype))中使用,并且它们基本上不能在堆栈上的任何地方(假设正常实现):C11,6.7.6.2-2:
If an identifier is declared as having a variably modified type, it shall be an ordinary identifier (as defined in 6.2.3), have no linkage, and have either block scope or function prototype scope. If an identifier is declared to be an object with static or thread storage duration, it shall not have a variable length array type.
void f(int n)
{
int array[n];
/* etc */
}
n <= 0
:f
必须防止这种情况,否则行为未定义:C11,6.7.6.2-5(强调我的):If the size is an expression that is not an integer constant expression: if it occurs in a declaration at function prototype scope, it is treated as if it were replaced by
*
; otherwise, each time it is evaluated it shall have a value greater than zero. The size of each instance of a variable length array type does not change during its lifetime. Where a size expression is part of the operand of asizeof
operator and changing the value of the size expression would not affect the result of the operator, it is unspecified whether or not the size expression is evaluated.
n > stack_space_left / element_size
: 没有找到剩余多少堆栈空间的标准方法(因为只要涉及标准,就没有堆栈这样的东西)。所以这个测试是不可能的。唯一明智的解决方案是为 n
设置一个预定义的最大可能大小,说 N
, 以确保不会发生堆栈溢出。 0 < n <= N
一些
N
选择。但是,该程序应该适用于
n == N
无论如何,因此不妨声明一个大小恒定的数组
N
而不是可变长度
n
.
alloca
(正如在
this answer 中提到的),但实际上它们是一样的(在堆栈上分配可变大小的内存)。
alloca
因此 VLA 存在,为什么它们没有被弃用?在我看来,使用 VLA 的唯一安全方法是使用有界大小,在这种情况下,采用具有最大大小的普通数组始终是一个可行的解决方案。
最佳答案
出于我不太清楚的原因,几乎每次讨论中出现 C99 VLA 的话题时,人们都开始主要谈论将运行时大小的数组声明为本地对象的可能性(即在堆栈上创建它们) ”)。这是相当令人惊讶和误导的,因为 VLA 功能的这一方面 - 支持本地数组声明 - 恰好是 VLA 提供的一个相当辅助的辅助功能。它在 VLA 可以做什么方面并没有真正发挥任何重要作用。大多数时候,地方 VLA 声明及其伴随的潜在陷阱的问题被 VLA 批评者强行推到前台,他们将其用作“稻草人”,旨在破坏讨论并将其陷入几乎不相关的细节中。
C 语言中 VLA 支持的本质首先是语言类型概念的革命性定性扩展。它涉及引入这种全新的类型,如 可变修改类型 .几乎每个与 VLA 相关的重要实现细节实际上都附加到它的类型,而不是附加到 VLA 对象本身。正是在语言中引入了可变修改的类型,才构成了众所周知的 VLA 蛋糕的大部分,而在本地内存中声明此类类型对象的能力只不过是在蛋糕上的一个微不足道且相当无关紧要的糖霜。
考虑一下:每次在自己的代码中声明类似的东西时
/* Block scope */
int n = 10;
...
typedef int A[n];
...
n = 5; /* <- Does not affect `A` */
A
(例如
n
的值)在控制通过上述 typedef 声明的确切时刻完成。
n
的值的任何变化更进一步(低于
A
的声明)不会影响
A
的大小.停下来思考一下这意味着什么。这意味着该实现应该与
A
相关联。一个隐藏的内部变量,它将存储数组类型的大小。这个隐藏的内部变量是从
n
初始化的在运行时控制越过
A
的声明.
/* Block scope */
int n = 10;
goto skip; /* Error: invalid goto */
typedef int A[n];
skip:;
typedef
,
typedef
需要运行时初始化与
typedef
有很大不同。是“经典”语言。 (它也恰好对在 C++ 中采用 VLA 的方式构成了重大障碍。)
void foo(unsigned n, unsigned m, unsigned k, int a[n][m][k]) {}
void bar(int a[5][5][5]) {}
int main(void)
{
unsigned n = 5;
int vla_a[n][n][n];
bar(a);
int classic_a[5][6][7];
foo(5, 6, 7, classic_a);
}
a
实际上是具有类型
int (*)[m][k]
。此类型不受
n
的值的影响。我特意向数组添加了一些额外的维度以保持其对运行时值的依赖。)
n
,
m
和
k
进入函数参数列表。不声明
n
,
m
和
k
首先,用户将无法声明
a
(另请参阅上面关于
n
的注释)。这些由用户显式传递给函数的参数将带入有关
a
的实际大小的信息。 .
#include <stdio.h>
#include <stdlib.h>
void init(unsigned n, unsigned m, int a[n][m])
{
for (unsigned i = 0; i < n; ++i)
for (unsigned j = 0; j < m; ++j)
a[i][j] = rand() % 100;
}
void display(unsigned n, unsigned m, int a[n][m])
{
for (unsigned i = 0; i < n; ++i)
for (unsigned j = 0; j < m; ++j)
printf("%2d%s", a[i][j], j + 1 < m ? " " : "\n");
printf("\n");
}
int main(void)
{
int a1[5][5] = { 42 };
display(5, 5, a1);
init(5, 5, a1);
display(5, 5, a1);
unsigned n = rand() % 10 + 5, m = rand() % 10 + 5;
int (*a2)[n][m] = malloc(sizeof *a2);
init(n, m, *a2);
display(n, m, *a2);
free(a2);
}
关于c - 无论如何,VLA 的意义何在?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22530363/
我想制作一个主要功能取决于发送短信的应用程序。在我开发 android(native) 之前,但现在我使用 React-Native 使其适用于 IOS 和 Android。 在android中,如果
我想重新运行一个测试类,包括它的 @BeforeMethod当其中任何一个是 @Test失败。我已经实现了 TestNG 重试逻辑来重新运行失败的测试用例,但我想运行整个类。 最佳答案 可以这样做。
前几天我遇到了一个接线员,-),而我找到的一个名称“arrow application”通常链接到 the other kind of Arrow present in Haskell ,这似乎无关。
我希望能够在我的sql语句中获取记录的ID: $result = $db->query("select id,FQDN, ip_address, ....."); 但是,我不希望它使用标题显示在导出中
我刚刚在Windows XP中激活了主题(通常使用经典的Win9x外观进行工作),并且看到两个面板是纯黑色的。其他面板也可以(颜色= clBtnFace)。 这两个面板的共同点是它们的父面板。两者都直
我正在研究使用数据库存储到该数据库条目的生成的链接,该链接包含有关该数据库条目的更多信息。因此,您会看到一些数据库,然后单击该条目并打开一个新页面,其中包含有关该条目的更多信息。 我一直在寻找一种可以
这里有几篇与此主题相关的帖子,但我已经应用了与我读到的所有内容相关的内容: CSS .infoWindow { width: 90px; height: 90px; } 创建信息窗口并
我目前正在编写使用sqlite3的脚本。最近,由于我的代码因错误而提前退出,我遇到了另一个程序正在使用该数据库的问题。 遇到类似问题,通常使用: conn = sqlite3.connect(...)
我只想在Redis数据库中使用1个键值对。并且该值将每60秒减少1。可能吗? 最佳答案 一个有趣的问题:)是的,您可以花一些技巧。 众所周知,Redis TTL会随着时间自动降低。因此,您可以将TTL
我尝试了数十种不同的方式来播放来自YouTube或服务器的视频,而没有使其进入全屏模式。我知道这是有可能的,因为YouTube应用程序允许这种情况发生,但似乎无法弄清楚该怎么做。请指教! 最佳答案 看
是否有办法在jade文件或其他模板引擎中编写服务器端nodejs代码(如down代码)? (或没有模板引擎): extends layout block content p Welcome to
Closed. This question needs to be more focused。它当前不接受答案。 想改善这个问题吗?更新问题,使其仅关注editing this post的一个问题。
您好,我最近下载并打开了适用于everyplay android的unity软件包,想知道我是否还能抓取玩游戏的用户上传到youtube的视频的URL?看来,实际共享功能的大部分代码是外部的,我看不到
我正在尝试在 Google 表单的网址中传递查询参数,并将该表单提交给其他人来填写,因此我希望 onSubmit 事件检查该查询参数并获取它。 我的 onSubmit 事件有以下代码: functio
我有一个这样的功能,我想将单元格放在网格列的左侧和右侧,现在它只是一个一个地放置,对于左侧网格列,它应该是 ,对于正确的,可以是 ,有没有根据索引区别对待,如果是奇数则属于左Grid Column
我有以下索引模板 { "index_patterns": "notificationtiles*", "order": 1, "version": 1, "aliases": {
我有一个Docker容器,该容器将具有大量(大约100个)自定义设置。与将默认值全部编码到DockerFile中相比,我希望具有更大的灵活性。我注意到docker run命令支持此选项: --env-
我有一个带有指针的谷歌地图。我已经对其进行了设置,因此每当缩放发生变化或视口(viewport)发生变化时,它都会相应地加载一组新的指针: google.maps.event.addListener(
我有一个php容器,每次启动容器时都需要启动php-fpm。现在,由于php-fpm config文件中的配置错误,fpm无法启动,因此,容器无法启动。无论如何,我可以在没有php-fpm的情况下启动
我正在运行没有sudo访问权限的docker容器 docker run -it --user 739000:8500 blabla... 无论如何,我可以在没有sudo访问的情况下在此docker容器
我是一名优秀的程序员,十分优秀!