2

我正在使用一些内存指针。我不想使用散列定义,请把讨论放在一边。我只想知道为什么这不能编译:

#include <stdio.h>

static const unsigned long *const pMemAddrA = (unsigned long *) 0x00000200ul;
static const unsigned long *const pMemAddrB = pMemAddrA;

int main (void)
{
    printf("%x", (unsigned int) pMemAddrB);
    return 0;
}

编译器输出 gcc:

||=== TestConst, Debug ===|
 ...main.c|4|error: initializer element is not constant|
||=== Build finished: 1 errors, 0 warnings ===|

编辑:

阅读答案后,我很高兴知道如何解决这个问题。

但是我不明白为什么这是一个问题。据我所知,静态内存是在程序启动时分配的。我知道如果变量“活”在不同的文件中,并且编译器不能保证变量的分配顺序,就会出现问题。但是,如果两个变量都“存在”在同一个文件中 - 就像两个变量都存在于同一个函数中一样 - 我认为编译器可以确保按照文件中声明的变量的顺序分配内存,因此我不会不明白为什么声明和初始化指向另一个 const 指针的 const 指针是一个问题。如果有人能启发我,我会很高兴。

4

2 回答 2

2

您的指针具有文件范围,因此初始化程序必须是常量表达式。pMemAddrA不是常量表达式,因此不能用于使用静态存储初始化变量。

它可用于在块范围内初始化变量,因此如果将声明移到内部main(并至少使第二个非静态),它将编译:

#include <stdio.h>

int main (void)
{

    const unsigned long *const pMemAddrA = (unsigned long *) 0x00000200ul;
    const unsigned long *const pMemAddrB = pMemAddrA;

    printf("%x", (unsigned int) pMemAddrB);
    return 0;
}

如果必须在文件范围内声明这两个指针,则无法防止重复初始化表达式,

static const unsigned long *const pMemAddrA = (unsigned long *) 0x00000200ul;
static const unsigned long *const pMemAddrB = (unsigned long *) 0x00000200ul;

#defineing它。

于 2012-08-31T14:19:26.630 回答
2

你没有描述什么“不起作用”,但我猜你的意思是这条线

static const unsigned long *const pMemAddrB = pMemAddrA;

产生错误

error: initializer element is not constant

.

解决方案是确实这个初始化器不被认为是常数。相反,pMemAddrA会留出一个内存区域,并将值0x00000200ul写入其中。从那以后,它是一个存在于内存中某处的值,而不是一个常量表达式。

根据您想要执行的操作,您可以添加另一个指针间接,例如

static const unsigned long *const * const pMemAddrB = &pMemAddrA;

*pMemAddrB并使用而不是访问它pMemAddrB

于 2012-08-31T14:19:36.607 回答