C++学习之cstdbool和cstddef头文件封装源码分析

 更新时间:2022年09月05日 14:23:23   作者:桑榆晚  
这篇文章主要为大家介绍了C++学习之cstdbool和cstddef头文件封装源码分析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

引言

cstdbool是C++对stdbool.h头文件的封装,里面定义了与bool变量相关的宏;

cstddef是C++对stddef.h头文件的封装,里面定义一些特殊类型(如size_t),有用的宏函数(offsetof)。

平时我们都是使用这些宏或者宏函数,对于它们的原理还不是很清楚,是怎么实现这些功能的呢?接下来我们就一一来看一看源码实现。

stdbool.h

cstdbool实现

代码参考:www.aospxref.com/android-13.…

在cstdbool文件中的逻辑很简单,解除__bool_true_false_are_defined的定义,然后将其定义为1,标识bool,true,false都已经被C++定义了。

 #include <__config>
 #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
 #pragma GCC system_header
 #endif
 #undef __bool_true_false_are_defined
 #define __bool_true_false_are_defined 1

C语言的原生实现stdbool.h

参考代码:www.aospxref.com/android-13.… 在该文件中,如果当前不是C++环境,那么我们将bool/true/false都进行定义,同时将__bool_true_false_are_defined定义为1,以便后续使用。

 #ifndef __cplusplus
 #define true 1
 #define false 0
 #define bool _Bool
  #endif
  #define __bool_true_false_are_defined 1

小结

stdbool.h实际上是为了解决C/C++的兼容问题出现的。

stddef.h

代码参考:www.aospxref.com/android-13.…

常量NULL的定义

从这里可以看出,实际上NULL有三种实现方式,0L,((void*)0)都是我们在C语言中常用的方式,其中的nullptr是C++定义的内部类型,能够做到对很多情况的适配,不是一个单独的基本类型。

 #if __cplusplus >= 201103L
 #define NULL nullptr
 #elif defined(__cplusplus)
 #define NULL 0L
 #else
 #define NULL ((void*)0)
 #endif

类型的定义

其中定义了如下的类型:

  • ptrdiff_t---保存指针减法的结果
  • size_t---无符号整数,表示类型占据的size大小
  • max_align_t---最大默认对齐的类型
  • nullptr_t---C++中的空指针类型 参考代码:www.aospxref.com/android-13.…

x86_64的相关实现如下: ptrdiff_t使用long定义,指针减法,实际上是64位数的减法,long足够表示; size_t使用long表示; max_align_t定义为拥有long long数据和long double数据的结构体; nullptr_t定义为nullptr的类型。

 #define _Addr long
  #if defined(__NEED_ptrdiff_t) && !defined(__DEFINED_ptrdiff_t)
  typedef _Addr ptrdiff_t;
  #define __DEFINED_ptrdiff_t
  #endif
  #if defined(__NEED_size_t) && !defined(__DEFINED_size_t)
  typedef unsigned _Addr size_t;
  #define __DEFINED_size_t
  #endif
  #if defined(__NEED_max_align_t) && !defined(__DEFINED_max_align_t)
  typedef struct { long long __ll; long double __ld; } max_align_t;
  #define __DEFINED_max_align_t
  #endif
typedef decltype(nullptr) nullptr_t;

offsetof宏

offsetof (type,member)接受两个参数,类型和成员名,返回该成员的偏移地址。

听上去这个实现起来很简单,我们来看看这个精巧的宏是如何设计的。

  #if __GNUC__ > 3
  #define offsetof(type, member) __builtin_offsetof(type, member)
  #else
  #define offsetof(type, member) ((size_t)( (char *)&(((type *)0)->member) - (char *)0 ))
  #endif

不考虑使用内建函数实现的方式,我们来看看第二种方式,这是一个比较复杂的宏,让我们来一步步拆开它,并体会其中的精妙:

((size_t)( (char *)&(((type *)0)->member) - (char *)0 ))

第一步:返回值类型强转为size_t,规范为byte数,也即是偏移量;

(char *)&(((type *)0)->member) - (char *)0

第二步:char*指针转换然后做减法,得到byte数的差值;

第三步:((type )0)->member这里使用了0指针强转为type指针,然后指向成员member,此时这个变量的地址相当于0+member偏移,然后取地址,指针转换为char*后做减法

这个方法的好处在于,只使用了0指针,没有通过其它变量的构造,很简单地找到了偏移地址,而且这个过程中做的每一步类型转换都是必要的。具体细节读者可以再品味一下,这也是我们阅读源码的收获。

以上就是C++学习之cstdbool和cstddef头文件封装源码分析的详细内容,更多关于C++头文件封装cstdbool cstddef的资料请关注脚本之家其它相关文章!

相关文章

  • C++基础学生管理系统

    C++基础学生管理系统

    这篇文章主要介绍了C++基础学生管理系统的相关资料,包括了文件处理代码,链表处理代码,以及自定义代码,主函数自定义,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-04-04
  • OpenMP task construct 实现原理及源码示例解析

    OpenMP task construct 实现原理及源码示例解析

    这篇文章主要为大家介绍了OpenMP task construct 实现原理及源码示例解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • 如何用C写一个web服务器之I/O多路复用

    如何用C写一个web服务器之I/O多路复用

    本文主要介绍了如何用C写一个web服务器之I/O多路复用,本次选择了 I/O 模型的优化,因为它是服务器的基础,这个先完成的话,后面的优化就可以选择各个模块来进行,不必进行全局化的改动了。
    2021-05-05
  • 利用C++编写一个Json解析器

    利用C++编写一个Json解析器

    这篇文章主要为大家详细介绍了如何利用C++编写一个简单又好用的Json解析器,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下
    2023-03-03
  • C++实现简易文本编辑器

    C++实现简易文本编辑器

    这篇文章主要为大家详细介绍了C++实现简易文本编辑器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • C++中fork函数的使用及原理

    C++中fork函数的使用及原理

    这篇文章主要介绍了C++中fork函数的使用及原理,在C++中,fork函数用于创建一个新的进程称为子进程,该进程与原始进程几乎完全相同,需要的朋友可以参考下
    2023-05-05
  • C语言链表实现贪吃蛇小游戏

    C语言链表实现贪吃蛇小游戏

    这篇文章主要为大家详细介绍了C语言链表贪吃蛇小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-05-05
  • 深入理解大数与高精度数的处理问题

    深入理解大数与高精度数的处理问题

    本篇文章是对大数与高精度数的处理进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C++拷贝构造函数和赋值运算符重载详解

    C++拷贝构造函数和赋值运算符重载详解

    拷贝构造函数是特殊的构造函数,是用一个已经存在的对象,赋值拷贝给另一个新创建的已经存在的对象,这篇文章主要介绍了C++拷贝构造函数和赋值运算符重载,需要的朋友可以参考下
    2024-05-05
  • C++超集C++/CLI模块的基本用法

    C++超集C++/CLI模块的基本用法

    这篇文章介绍了C++超集C++/CLI模块的基本用法,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07

最新评论