C语言中联合体union的使用

el/2024/7/17 0:13:00
本文编辑整理自:
http://bbs.chinaunix.net/forum.php?mod=viewthread&tid=179471
一、前言
联合体”(union)与“结构体”(struct)有一些相似之处。但两者有本质上的不同。在结构体中,各成员有各自的内存空间, 一个结构变量的总长度是各成员长度之和。而在“联合”中,各成员共享一段内存空间, 一个联合变量的长度等于各成员中最长的长度。应该说明的是, 这里所谓的共享不是指把多个成员同时装入一个联合变量内, 而是指该联合变量可被赋予任一成员值,但每次只能赋一种值, 赋入新值则冲去旧值
一个联合体类型必须经过定义之后, 才能使用它,才能把一个变量声明定义为该联合体类型。
关于结构体的更多内容请参考《结构体struct简介
二、联合体的定义
定义一个联合类型的一般形式为: 
union 联合名 
成员表 
};
成员表中含有若干成员,成员的一般形式为: 类型说明符 成员名。成员名的命名应符合标识符的规定。
例如: 
union perdata
{
   int class;
   char office[10];
};
  定义了一个名为perdata的联合类型,它含有两个成员,一个为整型,成员名为class;另一个为字符数组,数组名为office。联合定义之后,即可进行联合变量说明,被说明为perdata类型的变量,可以存放整型量class或存放字符数组office。
三、联合体变量的声明
  联合变量的说明和结构变量的声明方式相同, 也有三种形式。第一种先定义联合体类型,再声明联合体变量;第二种,定义联合体类型的同时声明联合体变量;第三种,直接声明联合体(以匿名的形式定义联合体类型)。
以perdata类型为例,说明如下: 
第一种,先定义联合体类型,再声明联合体变量
union perdata
{
   int class;
   char officae[10];
};
union perdata a,b; /*说明a,b为perdata类型*/
第二种,定义联合体类型的同时声明联合体变量
union perdata
int class;
char office[10]; 
}a,b;
第三种,直接声明联合体(以匿名的形式定义联合体类型)
union
int class;
char office[10]; 
}a,b 
经说明后的a,b变量均为perdata类型。a,b变量的长度应等于 perdata 的成员中最长的长度, 即等于office数组的长度,共10个字节。对a,b变量如赋予整型值时,只使用了2个字节,而赋予字符数组时,可用10个字节。
四、联合体变量的赋值和使用
  对联合体变量的赋值,使用都只能是对变量的成员进行。 
联合变量的成员表示为: 联合变量名.成员名 
例如,a被说明为perdata类型的变量之后,可使用 a.class,a.office 
不允许只用联合变量名作赋值或其它操作。 也不允许对联合变量作初始化赋值,赋值只能在程序中进行。
一个联合体变量, 每次只能赋予一个成员值。一个联合变量的值就是联合变员的某一个成员值。
[例4.1]设有一个教师与学生通用的表格,教师数据有姓名,年龄,职业,教研室四项。学生有姓名,年龄,职业,班级四项。
编程输入人员数据, 再以表格输出。
main()
{
   struct
   {
      char name[10];
      int age;
      char job;
      union
      {
         int class;
         char office[10];
      } depa;
   }body[2];
   int n,i;
   for(i=0;i<2;i++)
   {
      printf("input name,age,job and department\n");
      scanf("%s %d %c",body[i].name,&body[i].age,&body[i].job);
      if(body[i].job=='s')
         scanf("%d",&body[i].depa.class);
      else
         scanf("%s",body[i].depa.office);
   }
   printf("name\tage job class/office\n");
   for(i=0;i<2;i++)
   {
   if(body[i].job=='s')
      printf("%s\t%3d %3c %d\n",body[i].name,body[i].age ,body[i].job,body[i].depa.class);
   else
      printf("%s\t%3d %3c %s\n",body[i].name,body[i].age, body[i].job,body[i].depa.office);
   }
}
  本例程序用一个结构数组body来存放人员数据, 该结构共有四个成员。其中成员项depa是一个联合类型, 这个联合又由两个成员组成,一个为整型量class,一个为字符数组office。在程序的第一个for语句中,输入人员的各项数据,先输入结构的前三个成员name,age和job,然后判别job成员项,如为"s"则对联合depa·class输入(对学生赋班级编号)否则对depa·office输入(对教师赋教研组名)。
  在用scanf语句输入时要注意,凡为数组类型的成员,无论是结构成员还是联合成员,在该项前不能再加"&"运算符。如程序第18行中
body.name是一个数组类型,第22行中的body.depa.office也是数组类型,因此在这两项之间不能加"&"运算符。程序中的第二个for语句用于输出各成员项的值:



http://www.ngui.cc/el/5618084.html

相关文章

linux生产者,消费者问题

pthread_cond_wait() &#xff1a;用于阻塞当前线程&#xff0c;等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过…

可重入函数的深入理解以及printf的可重入性

这个概念在嵌入式操作系统中比较重要&#xff0c;由于存在任务的调度&#xff0c;它实时系统&#xff0c;可剥夺型内核中是危险的&#xff0c;如同一个安静的水雷。可能会被触发&#xff0c;也可能安然无恙。由于它运行结果的不可预期性&#xff0c;会使系统带来隐患。 printf(…

设计一组N个数,确定其中第k个最大值

今天看算法分析是&#xff0c;看到一个这样的问题&#xff0c;就是在一堆数据中查找到第k个大的值。 名称是&#xff1a;设计一组N个数&#xff0c;确定其中第k个最大值&#xff0c;这是一个选择问题&#xff0c;当然&#xff0c;解决这个问题的方法很多&#xff0c;本人在网上…

C++调用C生成的动态库

看下面这个例子&#xff0c;其中add函数是用c编写的代码&#xff0c;而主函数是用c编写的代码&#xff0c;将c代码编译成动态库&#xff0c;然后用c调用。 add.h #ifndef ADD_FILE_HEADER_INC #define ADD_FILE_HEADER_INC #include <stdio.h> #include <malloc.h&…

C语言调用C++库接口的方法概述

最近需要在由纯c语言编写的代码中调用C的动态库&#xff0c;在网上找了一些资料&#xff0c;现在总结下解决方法。 主要的思想就是将C的动态库再封装一层&#xff0c;在这一层编写C语言的函数api&#xff0c;这API中使用C动态库提供的类&#xff1b; 具体例子如下&#xff1a; …

linux多进程共享内存

shmget函数原型 shmget(得到一个共享内存标识符或创建一个共享内存对象) 所需头文件 #include <sys/ipc.h> #include <sys/shm.h> 函数说明 得到一个共享内存标识符或创建一个共享内存对象并返回共享内存标识符 函数原型 int shmget(key_t key, size_t size, int s…

fork,vfork,clone的区别

fork和clone的区别&#xff1a; Linux将创建进程和执行所创建的进程分为2个阶段。第一个阶段是创建。父进程首先复制子进程&#xff0c;所复制出来的子进程拥有自己的任务结构体和系统堆栈&#xff0c;除此之外所有资源都与父进程共享。Linux提供两种方式复制子进程&#xff1a…

多进程写文件的三种方式

多进程写log&#xff1a; &#xff08;1&#xff09;通过文件加锁的方式&#xff0c;在一个进程访问的时候将文件加锁&#xff0c;处理完了再关闭&#xff0c;不过这种方式会造成文件大频繁操作&#xff0c;导致性能比较低&#xff08;磁盘慢&#xff09;。 另外对于这种方式&a…

父子进程共享的资源

先来看一个题目&#xff1a; 当父进程调用fork()创建子进程之后&#xff0c;下列哪些变量在子进程中修改之后&#xff0c;父进程里也会相应地作出改动&#xff1f; A.全局变量 B.局部变量 C.静态变量 D.文件指针 答案为D&#xff0c;解释如下&#xff1a; fork()子进程和父进…

Linux下IPC总结

IPC进程间通信(Inter-Process Communication)就是指多个进程之间相互通信&#xff0c;交换信息的方法。Linux IPC基本上都是从Unix平台上继承而来的。主要包括最初的Unix IPC&#xff0c;System V IPC以及基于Socket的IPC。另外&#xff0c;Linux也支持POSIX IPC。 System V,BS…