我正在处理一个非常大的代码库,最近升级到GCC 4.3,现在触发了这个警告:

警告:不赞成将字符串常量转换为' char* '

显然,解决这个问题的正确方法是找到每一个声明

char *s = "constant string";

或者像这样调用函数:

void foo(char *s);
foo("constant string");

让它们成为const char指针。但是,这意味着至少要接触564个文件,这不是我目前希望执行的任务。现在的问题是我正在使用-Werror运行,所以我需要一些方法来抑制这些警告。我该怎么做呢?


当前回答

从这里到那里,就得到了这个解。这编译干净。

const char * timeServer[] = { "pool.ntp.org" }; // 0 - Worldwide 
#define WHICH_NTP            0 // Which NTP server name to use.
...
sendNTPpacket(const_cast<char*>(timeServer[WHICH_NTP])); // send an NTP packet to a server
...
void sendNTPpacket(char* address) { code }

我知道timeServer数组中只有一项。但可能还有更多。其余部分现在被注释掉以节省内存。

其他回答

如果它是一个活动的代码库,您可能仍然希望升级代码库。当然,手动执行更改是不可行的,但我相信这个问题可以通过一个sed命令一劳永逸地解决。不过,我还没有尝试过,所以对下面的建议半信半疑。

find . -exec sed -E -i .backup -n \
    -e 's/char\s*\*\s*(\w+)\s*= "/char const* \1 = "/g' {} \;

这可能无法找到所有的位置(甚至不考虑函数调用),但它可以缓解问题,并使手动执行剩余的更改成为可能。

查看GCC的Diagnostic Pragma支持,以及-W警告选项列表。

对于GCC,你可以像这里解释的那样使用#pragma警告指令。

我也遇到过类似的问题,我是这样解决的:

#include <string.h>

extern void foo(char* m);
 
int main() {
    // warning: deprecated conversion from string constant to ‘char*’
    //foo("Hello");
   
    // no more warning
    char msg[] = "Hello";
    foo(msg);
}

我没有访问foo,以适应它接受const char*,这将是一个更好的解决方案,因为foo没有改变m。

从常量字符串到字符指针进行类型转换。

char *s = (char *) "constant string";

还可以通过调用strdup()从字符串常量创建可写字符串。

例如,这段代码生成一个警告:

putenv("DEBUG=1");

然而,下面的代码不会这样做(它在将字符串传递给putenv之前在堆上复制了一个字符串):

putenv(strdup("DEBUG=1"));

在这种情况下(也许在大多数情况下)关闭警告是一个坏主意——它的存在是有原因的。另一种替代方法(默认情况下使所有字符串都可写)可能效率很低。

听听编译器在告诉你什么!