我问,因为在这个帖子中引发了讨论.
尝试在其他人的回复下使用评论进行严肃的来回讨论并不容易或有趣.所以我想听听我们的C专家的想法,一次不限制500个字符.
C标准几乎没有什么可说的NULL
和空指针常量.我只能找到两个相关部分.第一:
3.2.2.3指针
具有值0的整型常量表达式,或者类型为void*的表达式,称为空指针常量.如果为指针的相等性分配或比较空指针常量,则将常量转换为该类型的指针.这种称为空指针的指针保证与指向任何对象或函数的指针不相等.
第二个:
4.1.5通用定义
宏是
Run Code Online (Sandbox Code Playgroud)NULL
它扩展为实现定义的空指针常量;
问题是,可以NULL
扩展为实现定义的空指针常量,该常量与3.2.2.3中列举的那些不同吗?
特别是,它可以定义为:
#define NULL __builtin_magic_null_pointer
Run Code Online (Sandbox Code Playgroud)
甚至:
#define NULL ((void*)-1)
Run Code Online (Sandbox Code Playgroud)
我对3.2.2.3的解释是,它指定0的整数常量表达式和0强制转换为类型void*的整型常量表达式必须是实现可识别的空指针常量的形式,但它不是意思是一份详尽的清单.我相信实现可以自由地将其他源构造识别为空指针常量,只要没有其他规则被破坏.
例如,可以证明这一点
#define NULL (-1)
Run Code Online (Sandbox Code Playgroud)
不是法律定义,因为在
if (NULL)
do_stuff();
Run Code Online (Sandbox Code Playgroud)
do_stuff()
不能被称为,而与
if (-1)
do_stuff();
Run Code Online (Sandbox Code Playgroud)
do_stuff()
必须被召唤; 因为它们是等价的,所以这不是法律定义NULL
.
但是标准说整数到指针的转换(反之亦然)是实现定义的,因此它可以定义-1到指针的转换,作为产生空指针的转换.在这种情况下
if ((void*)-1)
Run Code Online (Sandbox Code Playgroud)
会评价为假,一切都会好的.
那么别人怎么想呢?
我要求每个人特别记住中描述的"假设"规则2.1.2.3 Program execution
.它是巨大的,有点迂回,所以我不会在这里粘贴它,但它实质上说,一个实现只需要产生与标准描述的抽象机器所需的相同的可观察副作用.它表示,只要程序的可观察副作用不被它们改变,任何优化,转换或编译器想要对程序执行的任何其他操作都是完全合法的.
因此,如果您希望证明特定的定义NULL
不合法,您需要提出一个可以证明它的程序.任何一个像我一样公然打破标准中的其他条款,或者可以合法地检测编译器必须做的任何魔法来使奇怪的NULL定义工作.
Steve Jessop发现了一个程序检测方法的例子,NULL
它没有被定义为3.2.2.3中两种形式的空指针常量之一,它是将常量字符串化:
#define stringize_helper(x) #x
#define stringize(x) stringize_helper(x)
Run Code Online (Sandbox Code Playgroud)
使用这个宏,人们可以
puts(stringize(NULL));
Run Code Online (Sandbox Code Playgroud)
并"检测"NULL不会扩展到3.2.2.3中的一个表单.这足以使其他定义非法吗?我只是不知道.
谢谢!