C++ Union

  • 框架
    • 变体成员
    • 各类构造,析构函数存在的条件等
    • 何时成员变成活跃
    • 类成员的使用方法
    • 匿名联合介绍

变体成员概述

要了解这个概念,首先要知道联合存在的特殊情况:类似联合类(Union-like classes,以下简称类联合类)
类联合类是指,一个非union的classes中有至少一个匿名union成员的类,或者是一个union
那么

变体成员,实际上就是指一个Union-like classes的联合中的非静态数据成员

  • 解释上来说,就是,对于一个类联合类,其变体就是联合的非静态数据成员

构造,析构的存在条件

这一部分,主要是讲,当非静态成员满足一定条件时,联合的构造函数,析构函数不会隐式创建的条件
那么

首先要知道,正常的union是一个classes,是允许拥有函数成员的,那么其隐式创建和删除构造,析构函数的条件自然是在满足union相关规定的情况下满足隐式构造,析构函数创建的条件的
那么对于union,特殊的地方在哪里?

  • 首先,对于一个union,任何一个非静态成员具有非平凡的复制、移动构造那么该特殊成员函数必须显式定义
  • 其次,对于一个union,任何一个非静态数据成员具有非平凡的默认构造函数,那么该特殊成员函数默认删除,除非给予相应成员一个默认成员初始化器(注意,一个union中,只能有一个成员具有此初始化器),注意,此成员必须是aggregate,否则需要调用相应构造函数(以便以构造方式将其初始化),在union显式定义的构造函数中.

何时成员变成活跃

当一个成员被置为活跃是,则开启当前成员的生存期,非活跃是结束

  • 上句话的意义在于,开启成员的生存期,对于一个变体成员来说,需要做什么,只要做到了,即为活跃,具体看下面使用方法.

类成员的使用方法

例1:

union CU1 {
    int foo1;
    float foo2;
};
void fun() {
    CU1 uT1;
    uT1.foo1 = 1;
    uT1.foo2 = 2.0;
}

其中,分析可得,CU1具有隐式的默认构造函数,此时产生uT1对象时,没有任何一个成员为活跃.
之后的两个赋值,分别使得成员foo1为活跃,foo1生命期终止foo2为活跃
例2:

struct CT {
    long sfoo2;
    float sfoo1;
};
union CU1 {
    CT foo1;
    float foo2;
};
void fun() {
    CU1 uT1;
    uT1.foo1.sfoo2 = 1;
    uT1.foo2 = 2.0;
    uT1.foo1.sfoo1 = 2.0;
}

其中,分析可得,CU1具有隐式默认构造函数,此时产生对象uT1时没有任何一个成员为活跃,
之后的三个赋值使得:成员foo1活跃,成员foo1失去活跃foo2活跃,由于违反堆叠,使得foo2隐式调用默认构造,并且该构造先序与赋值完成
例3:

struct CT {
    const long sfoo2;
    float sfoo1;
};
union CU1 {
    CT foo1 = {1,2.0};
    float foo2;
};
void fun() {
    CU1 uT1;
    uT1.foo1.sfoo1 = 3;
    uT1.foo2 = 2.0;
    uT1.foo1.sfoo1 = 2.0;
}

其中,在创建uT1对象时,foo1由于其默认成员初始化存在,使得foo1活跃,之后对foo2修改值,使得foo1变为非活跃,但是在下一句,重新对foo1中成员赋值时,此时由于foo1默认构造被删除,无法正确隐式调用使得foo1为活跃,因此导致未定义行为的发生

匿名联合介绍

匿名联合体实际上就是指联合体没有名字,此时联合体变体成员可位于名称空间直接访问,并且该联合体不能拥有成员函数和静态成员,并且除了在匿名名称空间内,其余的,所有的具有名称空间作用域的匿名联合体必须为static

你可能感兴趣的:(C++ Union)