- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我有一段极其微不足道的 C 代码:
static int arr[];
int main(void) {
*arr = 4;
return 0;
}
我知道第一条语句是非法的(我已经声明了一个具有静态存储持续时间和文件链接但没有指定大小的文件范围数组),但为什么会导致链接器错误? :
/usr/bin/ld: /tmp/cch9lPwA.o: in function `main':
unit.c:(.text+0xd): undefined reference to `arr'
collect2: error: ld returned 1 exit status
编译器难道不能在链接器之前捕获它吗?
令我感到奇怪的是,如果我省略 static
存储类,编译器会简单地假设数组的长度为 1
并且不会产生超过该长度的错误:
int arr[];
int main(void) {
*arr = 4;
return 0;
}
结果:
unit.c:5:5: warning: array 'arr' assumed to have one element
int arr[];
为什么在这里省略存储类会导致不同的行为,为什么第一段代码会产生链接器错误?谢谢。
最佳答案
空数组 static int arr[];
和零长度数组 static int arr[0];
是 gcc non-standard extensions .
这些扩展的目的是修复旧的“struct hack”。回到 C90 时代,人们编写了如下代码:
typedef struct
{
header stuff;
...
int data[1]; // the "struct hack"
} protocol;
where data
将被使用,就好像它具有超出数组的可变大小,具体取决于标题部分中的内容。这样的代码有问题,通常将数据写入填充字节并调用数组越界未定义行为。
gcc 通过添加空/零数组作为编译器扩展解决了这个问题,使代码的行为没有错误,尽管它不再是可移植的。
C 标准委员会认识到这个 gcc 特性很有用,因此他们在 1999 年将灵活的数组成员 添加到 C 语言中。从那时起,gcc 特性被视为过时的,因为使用C 标准灵活数组成员是首选。
如链接的 gcc 文档所识别:
Declaring zero-length arrays in other contexts, including as interior members of structure objects or as non-member objects, is discouraged.
这就是您的代码所做的。
请注意,没有编译器选项的 gcc 默认传递给 -std=gnu90
(gcc < 5.0) 或 -std=gnu11
(gcc > 5.0)。这为您提供了所有启用的非标准扩展,因此程序编译但不链接。
如果你想要符合标准的行为,你必须编译为
gcc -std=c11 -pedantic-errors
-pedantic
标志禁用 gcc 扩展,链接器错误按预期切换为编译器错误。对于您的情况下的空数组,您会得到:
error: array size missing in 'arr'
对于一个零长度数组,你得到:
error: ISO C forbids zero-size array 'arr' [-Wpedantic]
int arr[]
起作用的原因是因为这是 tentative definition 的数组声明具有外部链接(参见 C17 6.9.2)。它是有效的 C,可以被视为前向声明。这意味着在代码的其他地方,编译器(或者更确切地说是链接器)应该期望找到例如 int arr[10]
,然后它引用相同的变量。这样,可以在知道大小之前在代码中使用 arr
。 (我不推荐使用这种语言特性,因为它是一种“意大利面式编程”形式。)
当您使用 static
时,您可以通过强制变量具有内部链接来阻止在其他地方指定数组大小的可能性。
关于c - 为什么此语句会在 gcc 中产生链接器错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52067353/
我之前让 dll 注入(inject)器变得简单,但我有 Windows 7,我用 C# 和 C++ 做了它,它工作得很好!但是现在当我在 Windows 8 中尝试相同的代码时,它似乎没有以正确的方
我正在尝试制作一个名为 core-splitter 的元素,该元素在 1.0 中已弃用,因为它在我们的项目中起着关键作用。 如果您不知道 core-splitter 的作用,我可以提供一个简短的描述。
我有几个不同的蜘蛛,想一次运行所有它们。基于 this和 this ,我可以在同一个进程中运行多个蜘蛛。但是,我不知道如何设计一个信号系统来在所有蜘蛛都完成后停止 react 器。 我试过了: cra
有没有办法在达到特定条件时停止扭曲 react 器。例如,如果一个变量被设置为某个值,那么 react 器应该停止吗? 最佳答案 理想情况下,您不会将变量设置为一个值并停止 react 器,而是调用
https://code.angularjs.org/1.0.0rc9/angular-1.0.0rc9.js 上面的链接定义了外部js文件,我不知道Angular-1.0.0rc9.js的注入(in
我正在尝试运行一个函数并将服务注入(inject)其中。我认为这可以使用 $injector 轻松完成.所以我尝试了以下(简化示例): angular.injector().invoke( [ "$q
在 google Guice 中,我可以使用函数 createInjector 创建基于多个模块的注入(inject)器。 因为我使用 GWT.create 在 GoogleGin 中实例化注入(in
我在 ASP.NET Core 1.1 解决方案中使用配置绑定(bind)。基本上,我在“ConfigureServices Startup”部分中有一些用于绑定(bind)的简单代码,如下所示: s
我在 Spring MVC 中设置 initBinder 时遇到一些问题。我有一个 ModelAttribute,它有一个有时会显示的字段。 public class Model { privat
我正在尝试通过jquery post发布knockoutjs View 模型 var $form = $('#barcodeTemplate form'); var data = ko.toJS(vm
如何为包含多态对象集合的复杂模型编写自定义模型绑定(bind)程序? 我有下一个模型结构: public class CustomAttributeValueViewModel { publi
您好,我正在尝试实现我在 this article 中找到的扩展方法对于简单的注入(inject)器,因为它不支持开箱即用的特定构造函数的注册。 根据这篇文章,我需要用一个假的委托(delegate)
你好,我想自动注册我的依赖项。 我现在拥有的是: public interface IRepository where T : class public interface IFolderReposi
我正在使用 Jasmine 测试一些 Angular.js 代码。为此,我需要一个 Angular 注入(inject)器: var injector = angular.injector(['ng'
我正在使用 Matlab 代码生成器。不可能包含代码风格指南。这就是为什么我正在寻找一个工具来“ reshape ”、重命名和重新格式化生成的代码,根据我的: 功能横幅约定 文件横幅约定 命名约定 等
这个问题在这里已经有了答案: Where and why do I have to put the "template" and "typename" keywords? (8 个答案) 关闭 8
我开发了一种工具,可以更改某些程序的外观。为此,我需要在某些进程中注入(inject)一个 dll。 现在我基本上使用这个 approach .问题通常是人们无法注入(inject) dll,因为他们
我想使用 swing、spring 和 hibernate 编写一个 java 应用程序。 我想使用数据绑定(bind)器用 bean 的值填充 gui,并且我还希望它反射(reflect) gui
我有这段代码,当两个蜘蛛完成后,程序仍在运行。 #!C:\Python27\python.exe from twisted.internet import reactor from scrapy.cr
要点是 Spring Batch (v2) 测试框架具有带有 @Autowired 注释的 JobLauncherTestUtils.setJob。我们的测试套件有多个 Job 类提供者。因为这个类不
我是一名优秀的程序员,十分优秀!