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

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

介绍

在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 是无符号类型的正确选择,因为您希望大小匹配。

鉴于 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 的基础类型来知道要转换为哪个无符号类型。

以上就是转换为相同大小的无符号类型的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1455549.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月18日 07:51:58
下一篇 2025年12月18日 07:52:07

相关推荐

  • 如何使用C++框架在Web应用程序中实现实时通信?

    使用 c++++ 框架实现实时通信时,选择一个合适的框架至关重要,例如 websocket++、boost.asio 或 pistache。使用 websocket++ 作为示例,服务器端设置监听地址和端口,并使用 on_message 处理程序接收和广播消息。客户端设置连接和消息处理程序,连接到服…

    2025年12月18日
    000
  • C语言中的队列是什么?

    c 中的队列是遵循先进先出(fifo)原则的基本数据结构。这意味着添加到队列中的第一个元素将是第一个被删除的元素。队列在计算机科学中广泛用于各种应用,例如任务调度、图中的广度优先搜索和缓冲数据流。 定义及基本操作 队列通常支持以下主要操作: 1.排队: 将一个元素添加到队列末尾。 立即学习“C语言免…

    2025年12月18日
    000
  • 使用 typedef

    typedef 是我永远无法回头的事情之一。本质上,它允许对 C 中的现有类型进行别名,并使代码更易于阅读和编写。 以下例为例 struct my_struct { 整数a; 字符b;}void init_my_struct(struct my_struct* s, int a, char b); …

    2025年12月18日
    000
  • C 中使用 goto 进行错误处理

    最近,Python Brasil 邮件列表上开始了关于使用异常的原因的讨论。有一次,一位非常有能力的参与者评论了通过函数返回来处理错误是多么困难,就像在 C 中一样。 当你有一个复杂的算法时,每个可能失败的操作都需要一系列的 if 来检查操作是否成功。如果操作失败,您需要恢复之前的所有操作以退出算法…

    2025年12月18日
    000
  • 实现 malloc() 和 free() — 将元数据添加到内存块

    这篇文章是关于实现 malloc() 和 free() 函数的系列文章的一部分。之前,我们实现了一种相当简单的方法,几乎​​不释放任何内存:一个指针指向最后分配的块,使 free() 能够释放它,但只能释放它。 更好的选择是让最后一个块指向倒数第二个块,倒数第二个块指向倒数第三个块,依此类推,形成一…

    2025年12月18日
    000
  • 【Linux的历史】:第一种编程语言和第一个操作系统的起源

    随着第一个用于解释机械指令的复杂系统的出现,需要创建命令行,以便这些相同的机器可以根据给出的命令和/或已经预先保存的命令(存储或已编程)供机器执行。 #include #include #include // 定义表示进程和内存的结构体typedef 结构体过程{ 整数ID; int 执行时间; i…

    2025年12月18日
    000
  • C++ 框架事件处理中的性能优化技巧

    通过应用以下技巧,可以优化 c++++ 框架中事件处理的性能:减少虚拟函数调用,使用函数指针或模版化方法。避免对事件对象进行不必要的拷贝,使用引用或指针传递事件。批量处理同时到达的多个事件,以提高效率。如果事件处理需要大量计算,可以考虑使用多线程来并行化任务。 C++ 框架事件处理中的性能优化技巧 …

    2025年12月18日
    000
  • C++ 框架缓存策略:速度与可靠并驾齐驱

    c++++ 缓存策略可分为两类:读写穿透(所有操作通过缓存完成)和只读(仅读取操作通过缓存完成)。选择合适的策略取决于应用程序需求,读写穿透提供较快性能但可能导致数据不一致,而只读避免不一致但写入性能较慢。 C++ 框架缓存策略:速度与可靠并驾齐驱 前言 缓存是提高现代 C++ 应用程序性能的关键组…

    2025年12月18日
    000
  • 如何使用C++标准库的反射和内省机制?

    反射和内省在 c++++ 标准库中的用法:反射: 使用 typeid 运算符在运行时获取类型的名称和大小等信息。内省: 使用 std::is_same 和 std::is_convertible 比较类型是否相同,或是否可以进行隐式转换。 使用C++标准库进行反射和内省 反射允许程序在运行时检查类型…

    2025年12月18日
    000
  • c语言中pop是什么意思

    C 语言中 pop 操作将栈顶元素移除并返回其值,遵循后进先出的原则。1. 栈结构:遵循后进先出原则。2. pop 操作:移除栈顶元素并返回其值。3. 执行 pop 后:栈大小减 1。 C 语言中 pop 的含义 在 C 语言中,pop 是一种操作栈的操作,它将栈顶元素移除并返回该元素的值。 详细说…

    2025年12月18日
    000
  • 函数指针在 C++ 中如何提升代码的可测试性?

    函数指针增强了代码的可测试性,通过以下方式:隔离代码,使测试易于维护。降低耦合度,提高代码灵活性。函数指针使将函数作为变量传递和存储成为可能,通过解引用函数指针即可调用函数。这将业务逻辑与测试用例分离,提高了测试用例的灵活性,降低了代码耦合度。 函数指针:增强 C++ 代码可测试性的利器 理解函数指…

    2025年12月18日
    000
  • 深入了解函数指针如何在 C++ 中实现代码解耦

    函数指针是一种 c++++ 机制,通过将函数存储在变量中,实现代码解耦,使函数与调用代码分离。具有以下优势: 1. 代码解耦:提高可重用性、可维护性。 2. 运行时多态性:动态调用不同函数。 3. 性能提升:避免虚拟函数调用开销。 深入了解函数指针如何在 C++ 中实现代码解耦 引言 函数指针是一种…

    2025年12月18日
    000
  • 揭开函数指针在 C++ 中增强代码的可维护性的奥秘

    函数指针通过指向函数的变量提升了 c++++ 代码的可维护性。函数指针的语法为:returntype (functionpointername)(parametertypelist)。通过地址运算符 & 赋值函数指针,并使用解引用运算符 调用函数。函数指针在策略模式中很有用,允许在运行时根据…

    2025年12月18日
    000
  • 函数指针在 C++ 代码扩展性中的关键作用

    函数指针在 c++++ 代码中的关键作用是提升扩展性,允许在不指定函数名称的情况下调用函数。其用法包括声明、分配和调用函数指针。通过传递不同的比较器,函数指针在排序算法中发挥着至关重要的作用,实现多种排序方式。这使得 c++ 代码更加灵活、可重用,极大地提高了代码质量。 函数指针在 C++ 代码扩展…

    2025年12月18日
    000
  • 剖析 C++ 函数指针增强代码复用能力的原理

    函数指针是一种指向函数的指针,允许动态调用函数,从而增强代码复用性。例如,可创建一个通用折扣计算函数,接受函数指针作为参数,并为不同折扣类型创建不同的函数,通过传递不同的函数指针实现不同折扣计算。在 c++++ 中,排序策略函数指针可用于根据排序策略对学生列表排序,展示函数指针在代码复用中的应用。 …

    2025年12月18日
    000
  • 函数指针如何提高 C++ 代码的响应能力

    函数指针通过允许在运行时动态调用函数,提升了 c++++ 代码的响应能力。具体用法包括:声明函数指针:returntype (*functionpointername)(parametertypes);赋值函数指针:functionpointername = &function;调用函数指针…

    2025年12月18日
    000
  • c语言队列怎么写

    使用数组或链表可以在 C 语言中实现队列,遵循先进先出 (FIFO) 原则。数组实现包括创建队列、enqueue 和 dequeue 操作,而链表实现也包含上述操作。 如何用 C 语言实现队列 队列是一种遵循先进先出 (FIFO) 原则的数据结构。在 C 语言中,可以使用数组或链表来实现队列。 使用…

    2025年12月18日
    000
  • c语言怎么调用dll

    在 C 语言中,调用 DLL 涉及以下步骤:加载 DLL(LoadLibrary())获取函数指针(GetProcAddress())调用函数(通过函数指针)释放 DLL(FreeLibrary()) 如何在 C 语言中调用 DLL DLL(动态链接库)是一种共享库,它包含了可以在多个应用程序中使用…

    2025年12月18日
    000
  • c语言unsigned怎么用

    c语言中unsigned类型的使用 什么是unsigned类型? unsigned类型是C语言中的一种数据类型,用于存储无符号整数。这意味着它只能存储正整数和零,不能存储负数。 如何使用unsigned类型: 声明一个unsigned类型的变量:使用关键字”unsigned”…

    好文分享 2025年12月18日
    000
  • C++ 泛型编程中的类型推断是如何实现的?

    在 c++++ 中,类型推断通过使用模板和关键字 auto 实现,用于推导容器中元素的类型。模板参数推断(tad)机制则允许编译器从函数参数中推导出模板参数。类型推断简化了代码编写,并提高了 c++ 泛型编程的可重用性。 C++ 泛型编程中类型推断的实现 泛型编程是一种强大的 C++ 特性,它允许编…

    2025年12月18日
    000

发表回复

登录后才能评论
关注微信