【问题标题】:How to properly cast char ** x to const char **如何正确地将 char ** x 转换为 const char **
【发布时间】:2020-06-30 02:22:35
【问题描述】:

我有以下变量。

char **arr

然后我想对数组进行一些修改,这意味着它不能声明为常量。

现在我有一个接受const char ** arr 类型参数的函数。但是我无法控制这个函数的签名。

现在,当我将 arr 转换为 const char ** arr 时,g++ 会生成一个警告,即 [-Werror=cast-qual]

如需更多说明,请考虑关注 MCVE:

#include<cstdio>

void print(const char** x){
    printf("%s", x[0]);
}

int main(int argc, char **argv){
    if(argc>1){
        print((const char **)argv);     
    }
     return 0;
}

//Then compile it as follow:

$ g++ -Wcast-qual test.cpp

//gives the following output:

MCVE.cpp: In function ‘int main(int, char**)’:
MCVE.cpp:5:36: warning: cast from type ‘char**’ to type ‘const char**’ casts away qualifiers [-Wcast-qual]
   const char ** q = (const char**) argv;


所以我的问题是为什么这会产生警告?这样做有什么风险吗?

以及如何实现我想要实现的行为?

【问题讨论】:

  • 您的问题中没有与数组相关的内容。你有指针。请提供A Minimal, Complete, and Verifiable Example (MCVE)。是的,有风险。永远不要接受代码,直到它在没有警告的情况下编译。猜猜,你需要char * const *
  • 如果你不能从char** 改变函数的参数,那么你几乎被char** 卡住了。 argv 中的字符串不是常量,因此您可以有效地将其传递为 char** - 如果这是您的目标,您只需确保您的函数不会修改任何内容,或者复制字符串.
  • @VimalPatel 试试const_cast&lt;const char **&gt;(argv)。不确定我是否理解 g++ 警告,但我猜是在像 C++ 一样使用 C++ 转换时。
  • @dxiv,不会产生任何警告。让我检查一下其他行为。
  • 咯咯笑,所以const_cast conversion 将允许进行转换,并将防止-Wcast-qual 警告通过正常的C 样式转换防止的滥用。我试过了——确实如此——**p = '?'; 现在是一个错误。 @dxiv - 看来你应该在这个问题上撰写答案。

标签: c++ gcc-warning


【解决方案1】:

允许从char** 转换为const char** 提供了修改const charconst char* 的漏洞。

示例代码:

const char c = 'A';

void foo(const char** ptr)
{
   *ptr = &c; // Perfectly legal.
}

int main()
{
   char* ptr = nullptr;
   foo(&ptr);  // When the function returns, ptr points to c, which is a const object.
   *ptr = 'B'; // We have now modified c, which was meant to be a const object.
}

因此,将char** 转换为const char** 不是安全的转换。


你可以使用

if(argc>1)
{
   const char* ptr = argv[0];
   print(&ptr);     
}

让您的代码在没有cast-qual 警告的情况下编译。

如果您需要传递的不仅仅是第一个参数,则需要构造一个 const char* 数组并使用它。

if(argc>1)
{
   int N = <COMPUTE N FIRST>;
   const char** ptr = new const char*[N];
   for (int i = 0; i < N; ++i )
   {
      ptr[i] = argv[i];
   }

   print(ptr);

   delete [] ptr; // Make sure to deallocate dynamically allocated memory.
}

【讨论】:

  • 您可能想提供一种解决方法来实现我想要实现的行为,以便我可以接受这个作为答案。
  • @VimalPatel,我希望我的更新答案能满足您的需求。
  • @RSahu 您是否看到const_cast&lt;const char **&gt;(argv) 不会 触发相同警告的任何原因,或者仅仅是g++ 不一致?
  • @dxiv,我认为它应该触发相同的警告。我将把它归结为编译器中不同的执行路径,然后从裂缝中溜走。
猜你喜欢
  • 2019-09-10
  • 2011-02-17
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-04-02
  • 1970-01-01
相关资源
最近更新 更多