The example来自维基百科:
void updatePtrs(size_t *restrict ptrA, size_t *restrict ptrB, size_t *restrict val)
{
*ptrA += *val;
*ptrB += *val;
}
我在main()
:
int main(void)
{
size_t i = 10;
size_t j = 0;
updatePtrs(&i, &j, &i);
printf("i = %lu\n", i);
printf("j = %lu\n", j);
return 0;
}
根据维基百科的描述,val
指针不会被加载两次,因此j
的值应为10,但事实上它是20。
我对这个关键字的理解不正确吗?我应该使用gcc
的一些特定选项吗?
提前致谢。
答案 0 :(得分:5)
您的代码会导致未定义的行为。 restrict
是您对编译器的承诺,即所有指针参数都指向不同的内存区域。
你通过将&i
放在两个参数中来打破这个承诺。
(事实上,对于restrict
,允许传递重叠指针,但前提是不通过函数中的任何重叠指针进行写操作。但通常你不会打扰restrict
如果没有写作的话)。
FWIW,在我的gcc 4.9.2系统上,j = 20
的输出为-O0
,j = 10
或-O1
的输出为restrict
,这表明编译器确实在{{1}}的注释。当然,由于它是未定义的行为,您的结果可能会有所不同。