@清晰掉 C++ 中的 enum 结构在内存中是怎么存储的?
C++ 中的 enum 结构在内存中是怎么存储的?里面存储的是常量值吗?
关于占用内存的大小,enum类型本身是不占内存的,编译器直接替换。但是enum类型的变量肯定是占内存的(关于占用内存的大小,看后面详述)。
enum需要特别注意的是它的取值范围和类型分配。
关于确定enum的取值范围,如果一个enum中所有的枚举值非负,那么枚举的表示范围是[0,2^k - 1],其中2^k是使得所有枚举值位于此范围内的最小的2的幂。如果是有负值,那么范围就是[-2^k,2^k-1]。因此对一个给定的整数值,如果使用强制转换,而其值又不在枚举的表示范围以内,其行为是未定义的。
举一个例子:
- enum EnumTest{
- a =1,b = 5};
其中没有负数,最大值是5,二进制表示是101,需要3bits能够容纳下。因此取值范围是[0,7]。
- enum EnumTest{
- a =-2,b = 5};
绝对值最大的是5,需要3bits能够容纳下,但是因为有负值,需要增加一个符号位,于是需要4bits,取值范围是[-8,7]。
C++规定枚举的大小只要能够容纳下3bits或者4bits(上面的例子)就行,至于分配1byte还是4bytes,是编译器决定的,但是C++标准有个限制:1<=sizeof(enum)<=sizeof(int)。
实例代码:
- #include <iostream>
- using namespace std;
- class A{
- public:
- //enum类型本身不占内存
- enum EnumTest{
- a =1,b = 5};
- };
- enum EnumTest{
- a =1,b = 5};
- class B{
- public:
- private:
- //enum理想的变量会占用内存
- enum EnumTest enumSample1;
- };
- int main()
- {
- enum EnumTest{
- a =1,b = 5};
- cout << sizeof(enum EnumTest) << endl;
- //注意:此处结果是未定义的
- enum EnumTest enumSample1 = (enum EnumTest)10;
- cout << enumSample1 << endl;
- cout << "sizeof(Class A) = " << sizeof(A) << endl;
- cout << "sizeof(Class B) = " << sizeof(B) << endl;
- return 0;
- }
运行结果:
最后,给你提供好的参考资料:
what is the size of an enum type data in C++?
个人觉得谈论enum的内存分布没有太大的意义,因为enum中的值是在编译阶段就确定了。
虽然我们可以写出如下的代码:
- enum DefType{
- a = 1,
- b = 5
- };
- DefType testDefType; // 创建一个enum DefType的变量
但若像结构体一样调用testDefType.a,则不能通过编译。
若我们在函数外写下如下的语句:
- enum DefType{
- a = 1,
- b = 5
- };
- enum DefType2{
- a = 10,
- b2 = 50
- };
这样编译不会通过:error C2365: “a”: 重定义;以前的定义是“枚举数”。
关于enum的语法细节,楼上几位已经说的很详细了,不在赘述,只是举个enum常用的例子。
- class AA{
- enum DefType{
- a = 10,
- b = 5
- };
- int arr[a];
- };
- class BB{
- enum DefType2{
- a = 100,
- b2 = 50
- };
- int arr[a];
- };
这也是在类中静态声明一个数组的好方法称为enum hack。
@卢俊的答案说对了一半,宏虽然是在预处理阶段做了替换,但enum值不是在运行时确定,而是在编译时确定,从生成的汇编代码可以证明:
- ; 16 : int e = E;
- mov DWORD PTR _e$[ebp], 4
- ; 17 :
- ; 18 : int a = A;
- mov DWORD PTR _a$[ebp], 0
补充:enum结构体定义的的确是常量,但并不是在预编译的时候进行替换,而是在运行时,从enum定义的常量存储区取定义的常量值。因此,同样的常量定义,如果用#define,代码中多处用到该常量的话,编译出来的会比用enum定义使用的常量存储区大,下面这段程序:
//示例代码 a.c
#include <stdio.h>
#define E 4
enum ST
{
A,
B,
C,
D
};
void main()
{
int e = E;
int a = A;
}
使用预编译指令来预编译这段代码:
$ gcc -E a.c
//结果
# 2 "a.c" 2
enum ST
{
A,
B,
C,
D
};
void main()
{
int e = 4; //预编译后被替换
int a = A; //预编译后没有被替换
}
可以看出,#define 定义的常量,在预编译的时候做了替换,而enum定义的常量,并不是在预编译的时候进行替换,只是在运行的时候,根据标识去常量区获取对应的值。
enum结构体定义都是一些常量值,放在静态常量存储区。类似于类里面声明为static const的常量。
class A {
public:
enum Type{
Cat = 0,
Dog,
Hourse,
Cow
};
static const int i = 3;
};
cout << sizeof(A) <<endl;
输出是1。
- enum Type{
- Cat = 0,
- Dog,
- Hourse,
- Cow = 0xffffff
- };
- enum TypeChar : unsigned char {
- A = 0x00,
- B,
- C = 0xff
- };
- struct A {
- Type t;
- TypeChar tc;
- };
- int main() {
- cout << sizeof(Type) << endl;
- cout << sizeof(TypeChar) << endl;
- cout << sizeof(A) << endl;
- Type t1;
- TypeChar tc;
- Type t2;
- cout << &t1 << endl;
- cout << &tc << endl;
- cout << &t2 << endl;
- cout << sizeof(t1) << endl;
- return 0;
- }
输出是:
枚举默认是用int类型来存储的,32位系统下占4个字节。可以存储的最大值是0xffffff。 你可以改变枚举的大小例如enum TypeChar : unsigned char{} 这样可以节省空间, 在嵌入式编程中较为常见。嵌入式编程中,甚至有用位来存储的。
@清晰掉 C++ 中的 enum 结构在内存中是怎么存储的?的更多相关文章
- vs中调试程序查看变量在内存中的内容的方法
vs中调试程序 查看变量在内存中的内容的方法 https://blog.csdn.net/guojg1988/article/details/42922149 原文链接:http://www.sows ...
- Tomcat启动时加载数据到缓存---web.xml中listener加载顺序(例如顺序:1、初始化spring容器,2、初始化线程池,3、加载业务代码,将数据库中数据加载到内存中)
最近公司要做功能迁移,原来的后台使用的Netty,现在要迁移到在uap上,也就是说所有后台的代码不能通过netty写的加载顺序加载了. 问题就来了,怎样让迁移到tomcat的代码按照原来的加载顺序进行 ...
- C/C++中整数与浮点数在内存中的表示方式
在C/C++中数字类型主要有整数与浮点数两种类型,在32位机器中整型占4字节,浮点数分为float,double两种类型,其中float占4字节,而double占8字节.下面来说明它们在内存中的具体表 ...
- java中的各种数据类型在内存中存储的方式
原文地址:http://blog.csdn.net/aaa1117a8w5s6d/article/details/8251456 1.Java是如何管理内存的 java的内存管理就是对象的分配和释放问 ...
- javascript中的链表结构—从链表中删除元素
1.概念 上一个博文我们讲到链表,其中有一个方法remove()是暂时注释的,这个方法有点复杂,需要添加一个Previous()方法找到要删除的元素的前一个节点,这一个博文我们来分析一下这个remov ...
- JVM中java实例对象在内存中的布局
普通的Java对象实例 和 Java数组实例.Java数组实例的对象头多了一个数组的长度.Java虚拟机可以通过普通java对象的元数据来确定java对象的大小,但是从数组的元数据中却无法确定数组的 ...
- MFC中的NMHDR结构体和NMUPDOWN结构体
建立spin控件,创建UDN_DELTAPOS一个消息函数后: void CSpinDlg::OnDeltaposSpin1(NMHDR* pNMHDR, LRESULT* pResult) { NM ...
- Java 数组在内存中的结构
Java中的数组存储两类事物: 原始值(int,char,...),或者引用(对象指针). 当一个对象通过 new 创建,那么将在堆内存中分配一段空间,并且返回其引用(指针). 对于数组,也是同样的方 ...
- Java中内存中的Heap、Stack与程序运行的关系
堆和栈的内存管理 栈的内存管理是顺序分配的,而且定长,不存在内存回收问题:而堆 则是随机分配内存,不定长度,存在内存分配和回收的问题:堆内存和栈内存的区别可以用如下的比喻来看出:使用堆内存就象是自己动 ...
随机推荐
- hdu2444The Accomodation of Students (最大匹配+判断是否为二分图)
题意 首先判断所有的人可不可以分成两部分,每部分内的所有人都相互不认识.如果可以分成 则求两部分最多相互认识的对数. 解题 类似分成两组,同组互不相关,就可能使判断是否为二分图 能否分成两部分 则是判 ...
- Delphi7 IDE
- 架构师必备,带你弄清混乱的JAVA日志体系!
作者:孤独烟 出处:http://rjzheng.cnblogs.com/ 本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任 ...
- AIX中磁带设备的使用
1.AIX中的磁带设备 在UNIX系统中,磁带一般具有如下共同属性. 磁带密度:指磁带使用时采用的低密度方式和高密度方式 回退操作:磁带一次写入操作完成后,是否立刻倒带,回退到磁带头部. 位置 ...
- SecureCRT文件和文件夹显示不同颜色(像linux中那样效果)
如何设置secureCRT使用的他可以像linux文件和文件夹显示不同的颜色呢 原先显示效果如下: 效果图 配置
- Linux系统账户管理指令
sudo passwd -l weblogic 锁定账户 sudo passwd -u weblogic 解锁账户 useradd weblogic -p xxxxx 添加账户指定密码 sudo us ...
- Registry key 'Software\JavaSoft\Java Runtime Environment\CurrentVersion' has value '1.8', but '1.7'
第一种方法:安装1.8之前安装了1.7,将1.7卸载就好了. 第二种方法:删掉Windows\System32下的java.exe, javaw.exe 就行了,但是安装的1.8的jdk会回到1.7的 ...
- Codeforces 965 枚举轮数贪心分糖果 青蛙跳石头最大流=最小割思想 trie启发式合并
A /*#include<cstring>#include<algorithm>#include<queue>#include<vector>#incl ...
- python 反射、动态导入
1. 反射 hasattr(obj,'name') # 判断对象中是否含有字符串形式的方法名或属性名,返回True.False getattr(obj,'name',None) ...
- Python修炼之路-Socket
网络编程 socket套接字 socket通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄,应用程序通常通过“套接字”向网络发出请求或者应答网络请求. socket ...