c - 有没有办法使用`const`变量初始化`const``struct`?
问题描述
我想struct
在 C99 中创建一个封装了一个以 null 结尾的字符串缓冲区以及实际字符串长度:
typedef unsigned int uint_t;
typedef unsigned char uchar_t;
typedef struct {
uchar_t * buffer;
uint_t length; // excluding null-terminating character
} string_t;
但是,我遇到了一些关于成员的const
-ness 的困难struct
。具体来说,当我想使用一个接受这种 a 的函数const struct string_t
,并用一个带有 const 成员的初始化程序来提供它时,编译器会对我大喊大叫。
void show_string_internal(const string_t str) {
printf("%s", str.buffer);
}
void show_string(const uchar_t * buffer, uint_t length) {
const string_t str = // <== tricky assignment here
(const string_t){ buffer, length };
show_string_internal(str);
}
int main() {
uchar_t message[] = "Hello, world.";
show_string(message, sizeof(message) - 1);
return 0;
}
这会在 GCC 中的突出显示的行上产生警告...:
警告:初始化从指针目标类型中丢弃“const”限定符
...在 Visual Studio 2015 中:
警告 C4090:“正在初始化”:不同的“常量”限定符
显然,我在这里做错了什么。我发现解决这个问题的唯一方法是声明:
typedef struct {
const uchar_t * buffer;
const uint_t length;
} const_string_t;
但是现在我有两种类型而不是一种,所以我需要创建方便的方法来在两者之间进行转换,并且我正在创建声明性糖而不是使用语言功能,因此代码的可读性较差。
所以我的问题是,正如标题所述:有没有办法为成员初始化const
struct
使用const
变量?是否有其他方法可以达到我希望达到的结果?如果不是,为什么不(请包括对官方文档的引用)?
任何帮助,将不胜感激。
解决方案
关于指针类型的常量应该“从右到左”阅读("const T * buffer"
实际上是指“指向常量内容的指针”,而不是“指向内容的常量指针”或“指向常量内容的常量指针”)。
因此,您将使用指向 const char 缓冲区的可变指针并将其用于结构的 const 版本(询问您的 const 结构是否具有可变字段 - 这是显示警告的原因)。如果你打算摆脱警告,你必须像这样移动“const”的东西:
void show_string(char * const buffer, const int length) { // <== move const to the right of "*"
const string_t str = // <== tricky assignment here
(const string_t) {
buffer, length
};
show_string_internal(str);
}
如果您还计划自己拥有 const 缓冲区,则必须重新定义您的 string_t 结构(或作为单独的类型引入):
typedef struct {
const uchar_t * buffer; // <== add "const" here
uint_t length; // excluding null-terminating character
} string_t;
...
void show_string(const char * const buffer, const int length) { // two "const" clauses there
const string_t str = // <== tricky assignment here
(const string_t) {
buffer, length
};
show_string_internal(str);
}
希望这可以帮助。
推荐阅读
- data-structures - 存储用户名和密码的最佳方式
- ios - 滚动 UITableView 时集合视图单元格的大小发生变化
- bash - 执行从参数接收到的命令到 Bash 脚本中的函数
- git - 将超过 100MB 的文件推送到 GitHub
- azure - 连接到 Azure Signalr 时,appservice 可以充当客户端吗?
- mysql - 从 MySQL 中选择 7 行,除了最后 7、14、...、N*7 行
- asp.net - 成功验证后覆盖signin-oidc回调重定向的位置
- three.js - 如何在three.js中从视口角度获取一个点的Vector3?
- c# - UNITY 在鼠标点击旁边出现文字
- typescript - 我如何改进这个通用 Box 函数的类型?