0

0

转换为相同大小的无符号类型

WBOY

WBOY

发布时间:2024-07-11 20:52:01

|

338人浏览过

|

来源于dev.to

转载

转换为相同大小的无符号类型

介绍

在cdecl中,有这样的枚举:

enum cdecl_show {
  cdecl_show_predefined       = 1 << 0,
  cdecl_show_user_defined     = 1 << 1,
  cdecl_show_opt_ignore_lang  = 1 << 2
};
typedef enum cdecl_show cdecl_show_t;

其值是可以按位或在一起的位标志。

标志的作用在这里并不重要,但是,简单地说,它们控制响应 cdecl show 命令而显示哪些类型。

我正在努力增强 show 命令的行为,以便如果没有显示具有特定名称的用户定义类型,则通过以下代码显示具有相同名称的预定义类型(如果有):

if ( !showed_any && (show & cdecl_show_user_defined) != 0 ) {
  show &= ~cdecl_show_user_defined;
  show |= cdecl_show_predefined;
  // ...
}

即关闭cdecl_show_user_defined 位并打开cdecl_show_predefined 位。 问题是,当使用符号转换编译器选项进行编译时,我得到:

show.c:244:8: warning: implicit conversion changes signedness: 'int' to 'unsigned int' [-wsign-conversion]
  show &= ~cdecl_show_user_defined;
       ~~ ^~~~~~~~~~~~~~~~~~~~~~~~
1 warning generated.

发生这种情况是因为 c 中的枚举值隐式转换为其基础类型(此处为 int),但 ~ 将其操作数转换为无符号 int。

消除警告的明显方法是首先转换为 unsigned:

  show &= ~(unsigned)cdecl_show_user_defined;  // no warning

问题是,在 c23 之前的 c 语言中,你无法确定枚举的底层类型是什么。 来自 c11 标准 §6.7.2.2 ¶4:

每个枚举类型应与 char、有符号整数类型或无符号整数类型兼容。类型的选择是实现定义的。

如果您不知道基础类型是什么,特别是它的大小,您就不知道 unsigned int 是无符号类型的正确选择,因为您希望大小匹配。

Closers Copy
Closers Copy

营销专用文案机器人

下载

鉴于 cdecl_show 只有值 1、2 和 4,可以肯定它的基础类型是 int — 但你不能确定。 我们需要的是一种转换为与给定表达式的类型大小相同的无符号类型的方法。

解决方案

使用 _generic 和 static_if (在那篇文章中给出),我们可以实现:

#define to_unsigned(n)                    \
  static_if( sizeof(n) == sizeof(char),   \
    (unsigned char)(n),                   \
  static_if( sizeof(n) == sizeof(short),  \
    (unsigned short)(n),                  \
  static_if( sizeof(n) == sizeof(int),    \
    (unsigned int)(n),                    \
  static_if( sizeof(n) == sizeof(long),   \
    (unsigned long)(n),                   \
    (unsigned long long)(n) ) ) ) )

其中 n 是任何整数或枚举类型的任何数值表达式。 实现很简单:使用 static_if 链来确定表达式 n 的类型的大小,然后将其转换为相同大小的无符号类型。

鉴于此,我现在可以写:

  show &= ~TO_UNSIGNED( CDECL_SHOW_USER_DEFINED );

并且没有任何警告。

结论

再次,_generic 允许您进行一些编译时类型自省。 to_unsigned() 的好处是,即使底层类型发生变化,它也始终有效。

除了对枚举有用之外,它对于整型类型的 typedef 也很有用,因为你永远不必查找 typedef 的基础类型来知道要转换为哪个无符号类型。

相关专题

更多
typedef和define区别
typedef和define区别

typedef和define区别在类型检查、作用范围、可读性、错误处理和内存占用等。本专题为大家提供typedef和define相关的文章、下载、课程内容,供大家免费下载体验。

103

2023.09.26

c语言typedef的用法
c语言typedef的用法

c语言typedef的用法有定义基本类型别名、定义结构体别名、定义指针类型别名、定义枚举类型别名、定义数组类型别名等。本专题为大家提供typedef相关的文章、下载、课程内容,供大家免费下载体验。

95

2023.09.26

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

312

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

522

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

49

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

190

2025.08.29

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

65

2025.12.31

php网站源码教程大全
php网站源码教程大全

本专题整合了php网站源码相关教程,阅读专题下面的文章了解更多详细内容。

43

2025.12.31

视频文件格式
视频文件格式

本专题整合了视频文件格式相关内容,阅读专题下面的文章了解更多详细内容。

35

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号