【Struct(结构体)杂谈之六】无既是有---没有成员变量的Struct(结构体)

2024-03-20 06:08

本文主要是介绍【Struct(结构体)杂谈之六】无既是有---没有成员变量的Struct(结构体),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

没有成员变量的Struct(结构体)


       在开始本篇之前,想问大家一个问题:

       ---0是什么?

       ---呵呵,就是没有呗!

       ---那好,这5块钱拿去,就当抵我上次向你借的500块钱。

       ---什么?这哪和哪啊!这不一样

       ---可是你自己说的, 0就是“没有”。

       ----我说不清,反正不行,你必须还我500.

 

       0是什么?起什么作用呢?为什么500 ≠ 5?

       这节我们来讨论0 的作用。例如,500块钱,它后面0起到了什么作用呢? 500 的0,表示十和个位“没有”。虽说“没有”,但这个0 却不能省略。因为如果省略了0,一件500块的衣服,你只给5块,小心遭到暴打。


那原因是什么呢?

       在按位计数法中,数位具有很重要的意义。即使十位的数“没有”,也不能不写数字。这时就轮到0 出场了,即0 的作用就是占位。换言之,0 占着一个位置以保证数位高于它的数字不会产生错位。

正因为有了表示“没有”的0,数值才能正确地表现出来。可以说在按位计数法中0是不可或缺的。

 

       打住,这和我们讲的struct有什么关系?

       当然有关系了,请问下面这段代码输出的是什么呢?

#include <iostream>using namespace std;struct NoMember
{};int main(void)
{cout<<"The size of the struct NoMem is:"<<endl;cout<<sizeof(NoMember)<<endl;getchar();return 0;
}

是0呢?还是1?2?3?想必大部分人还是说不出来的,那我们先看看输出结果:

一个没有任何变量的Struct居然占了1个字节的空间。

这不科学!


那这是为什么呢?

不急,让子弹先飞一会儿.....


再看下面一段代码:

#include <iostream>using namespace std;struct NoMember
{};struct NoMember1
{int mA[0];
};int main(void)
{//cout<<endl<<"The size of the struct NoMember is: "<<sizeof(NoMember)<<endl;cout<<endl<<"The size of the struct NoMember1 is: "<<sizeof(NoMember1)<<endl;getchar();return 0;
}

一个结构体里面定义了一个空的数组,那这次的输出会是什么呢?
厄, 这还是一个空的结构体,所以输出还是1吧?
好的,你还真蒙对了,请看结果:



但是,如果struct里面还有另外的变量呢?会是什么情况呢?

#include <iostream>using namespace std;struct NoMember
{};struct NoMember1
{int mA[0];
};struct NoMember2
{char mB;int mA[0];
};int main(void)
{//cout<<endl<<"The size of the struct NoMember is: "<<sizeof(NoMember)<<endl;//cout<<endl<<"The size of the struct NoMember1 is: "<<sizeof(NoMember1)<<endl;cout<<endl<<"The size of the struct NoMember2 is: "<<sizeof(NoMember2)<<endl;getchar();return 0;
}

这一次的输出是什么呢?

char mB占据4个字节,然后int mA[0]占据1个字节,所以结果应该是5吧? 


但是结果是4,为什么呢?


其实结合500 ≠ 5的话也很好理解,虽然结构体包含0个成员变量,但是结构体起到“占位”的作用,“空结构体”变量必须被存储,编译器为其分配一个字节的空间用于占位了。这样一来,不光可以取地址,两个不同的“空结构体”变量又可以得以区分

我们可以输出各个结构体的地址:

#include <iostream>using namespace std;struct NoMember
{};struct NoMember1
{int mA[0];
};struct NoMember2
{char mB;int mA[0];
};int main(void)
{NoMember nm0;NoMember1 nm1;NoMember2 nm2;cout<<endl<<"The size of the struct NoMember is: "<<sizeof(NoMember)<<endl;cout<<"The address of the struct of NoMember: 0x"<<&nm0<<endl;cout<<endl<<"The size of the struct NoMember1 is: "<<sizeof(NoMember1)<<endl;cout<<"The address of the struct of NoMember: 0x"<<&nm1.mA<<endl;cout<<endl<<"The size of the struct NoMember2 is: "<<sizeof(NoMember2)<<endl;cout<<"The address of the struct of NoMember: 0x"<<&nm2.mA<<"\t0x"<<&nm2.mA<<endl;getchar();return 0;
}

输出结果如下:



注意下面这份简谱,注意到乐谱上也有很多的休止符,在简谱上也是用0来表示,但是它们不是“没有”, 而是表示不发音!


 

       这些0都不能去掉的,因为去掉,节奏肯定乱了。

0与其说是“空”,还不如说是“填空”更恰当。因为它的作用是占位。


结合这个例子,想必大家就理解了, “空结构体” 为什么需要占用一个字节的空间。


       可见,科学和艺术不分家的啊!


对数组比较熟悉的同学可能会问,怎么可以有0元素数组呢?

是的,如果我们在struct 和class之后定义0元素数组,会编译报错,如下代码所示:

#include <iostream>using namespace std;struct NoMember
{};struct NoMember1
{int mA[0];
};struct NoMember2
{char mB;int mA[0];
};struct NoMember3
{int arr[0];
};int arr[0];int main(void)
{NoMember nm0;NoMember1 nm1;NoMember2 nm2;cout<<endl<<"The size of the struct NoMember is: "<<sizeof(NoMember)<<endl;cout<<"The address of the struct of NoMember: 0x"<<&nm0<<endl;cout<<endl<<"The size of the struct NoMember1 is: "<<sizeof(NoMember1)<<endl;cout<<"The address of the struct of NoMember: 0x"<<&nm1.mA<<endl;cout<<endl<<"The size of the struct NoMember2 is: "<<sizeof(NoMember2)<<endl;cout<<"The address of the struct of NoMember: 0x"<<&nm2.mA<<"\t0x"<<&nm2.mA<<endl;getchar();return 0;
}

出错信息如下:

1>e:\code\vs2010_prjs\struct\structdeclare\nomems.cpp(28): error C2466: 不能分配常量大小为 0 的数组


而Stuct里面的0元素数组报的是warning,如下所示:

1>e:\code\vs2010_prjs\struct\structdeclare\nomems.cpp(24): warning C4200: 使用了非标准扩展 : 结构/联合中的零大小数组
1>          当 UDT 包含大小为零的数组时,无法生成复制构造函数或副本赋值运算符

那么,问题又来了,这些 struct里面的0元素数组有什么意义呢?


欲知后事如何,请听下回分解......



这篇关于【Struct(结构体)杂谈之六】无既是有---没有成员变量的Struct(结构体)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/828512

相关文章

SQL Server安装时候没有中文选项的解决方法

《SQLServer安装时候没有中文选项的解决方法》用户安装SQLServer时界面全英文,无中文选项,通过修改安装设置中的国家或地区为中文中国,重启安装程序后界面恢复中文,解决了问题,对SQLSe... 你是不是在安装SQL Server时候发现安装界面和别人不同,并且无论如何都没有中文选项?这个问题也

Spring Boot中的路径变量示例详解

《SpringBoot中的路径变量示例详解》SpringBoot中PathVariable通过@PathVariable注解实现URL参数与方法参数绑定,支持多参数接收、类型转换、可选参数、默认值及... 目录一. 基本用法与参数映射1.路径定义2.参数绑定&nhttp://www.chinasem.cnbs

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

MySQL中的索引结构和分类实战案例详解

《MySQL中的索引结构和分类实战案例详解》本文详解MySQL索引结构与分类,涵盖B树、B+树、哈希及全文索引,分析其原理与优劣势,并结合实战案例探讨创建、管理及优化技巧,助力提升查询性能,感兴趣的朋... 目录一、索引概述1.1 索引的定义与作用1.2 索引的基本原理二、索引结构详解2.1 B树索引2.2

如何使用Maven创建web目录结构

《如何使用Maven创建web目录结构》:本文主要介绍如何使用Maven创建web目录结构的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录创建web工程第一步第二步第三步第四步第五步第六步第七步总结创建web工程第一步js通过Maven骨架创pytho

Python循环结构全面解析

《Python循环结构全面解析》循环中的代码会执行特定的次数,或者是执行到特定条件成立时结束循环,或者是针对某一集合中的所有项目都执行一次,这篇文章给大家介绍Python循环结构解析,感兴趣的朋友跟随... 目录for-in循环while循环循环控制语句break语句continue语句else子句嵌套的循

Python变量与数据类型全解析(最新整理)

《Python变量与数据类型全解析(最新整理)》文章介绍Python变量作为数据载体,命名需遵循字母数字下划线规则,不可数字开头,大小写敏感,避免关键字,本文给大家介绍Python变量与数据类型全解析... 目录1、变量变量命名规范python数据类型1、基本数据类型数值类型(Number):布尔类型(bo

一文全面详解Python变量作用域

《一文全面详解Python变量作用域》变量作用域是Python中非常重要的概念,它决定了在哪里可以访问变量,下面我将用通俗易懂的方式,结合代码示例和图表,带你全面了解Python变量作用域,需要的朋友... 目录一、什么是变量作用域?二、python的四种作用域作用域查找顺序图示三、各作用域详解1. 局部作

Python struct.unpack() 用法及常见错误详解

《Pythonstruct.unpack()用法及常见错误详解》struct.unpack()是Python中用于将二进制数据(字节序列)解析为Python数据类型的函数,通常与struct.pa... 目录一、函数语法二、格式字符串详解三、使用示例示例 1:解析整数和浮点数示例 2:解析字符串示例 3:解

C++/类与对象/默认成员函数@构造函数的用法

《C++/类与对象/默认成员函数@构造函数的用法》:本文主要介绍C++/类与对象/默认成员函数@构造函数的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录名词概念默认成员函数构造函数概念函数特征显示构造函数隐式构造函数总结名词概念默认构造函数:不用传参就可以