union 和 struct

  1. 聯合說明和聯合變量定義

      聯合也是一種新的數據類型, 它是一種特殊形式的變量。聯合說明和聯合變量定義與結構十分相似。其形式爲:

      union 聯合名{
      數據類型 成員名;
      數據類型 成員名;
      …
      } 聯合變量名;

      聯合表示幾個變量公用一個內存位置, 在不同的時間保存不同的數據類型 和不同長度的變量。下例表示說明一個聯合a_bc:

union a_bc  
  {  
    int i;  
    char mm;  
};  

再用已說明的聯合可定義聯合變量。

  例如用上面說明的聯合定義一個名爲lgc的聯合變量, 可寫成:

  union a_bc lgc;

  在聯合變量lgc中, 整型量i和字符mm公用同一內存位置。

  當一個聯合被說明時, 編譯程序自動地產生一個變量, 其長度爲聯合中最大的變量長度。

  聯合訪問其成員的方法與結構相同。同樣聯合變量也可以定義成數組或指針,但定義爲指針時, 也要用”->”符號, 此時聯合訪問成員可表示成:

  聯合名->成員名

  另外, 聯合既可以出現在結構內, 它的成員也可以是結構。

  例如:

struct  
{  
    int age;  
    char *addr;  
    union  
    {  
        int i;  
        char *ch;  
    }x;  
}y[10];  

若要訪問結構變量y[1]中聯合x的成員i, 可以寫成:

  y[1].x.i;

  若要訪問結構變量y[2]中聯合x的字符串指針ch的第一個字符可寫成:

  *y[2].x.ch;

  若寫成”y[2].x.*ch;”是錯誤的。

  2. 結構和聯合的區別

  結構和聯合有下列區別:

  1. 結構和聯合都是由多個不同的數據類型成員組成, 但在任何同一時刻, 聯合轉只存放了一個被選中的成員, 而結構的所有成員都存在。

  2. 對於聯合的不同成員賦值, 將會對其它成員重寫, 原來成員的值就不存在了, 而對於結構的不同成員賦值是互不影響的。

  下面舉一個例了來加對深聯合的理解。

  例4:

main()  
{  
    union  
    {  
        int i;  
        struct  
        {  
            char first;  
            char second;  
        }half;  
    }number;  
    number.i=0x4241; /*聯合成員賦值*/  
    printf("%c%c/n", number.half.first,number.half.second);  
    number.half.first='a'; /*聯合中結構成員賦值*/  
    number.half.second='b';  
    printf("%x/n", number.i);  
    getch();  
}  

輸出結果爲:
  AB
  6261
  從上例結果可以看出: 當給i賦值後, 其低八位也就是first和second的值;當給first和second賦字符後, 這兩個字符的ASCII碼也將作爲i 的低八位和高八位。

    在實際問題中有很多這樣的例子。 例如在學校的教師和學生中填寫以下表格: 姓 名 年 齡 職 業 單位 “職業”一項可分爲“教師”和 “學生”兩類。 對“單位”一項學生應填入班級編號,教師應填入某系某教研室。 班級可用整型量表示,教研室只能用字符類型。 要求把這兩種類型不同的數據都填入“單位”這個變量中, 就必須把“單位”定義爲包含整型和字符型數組這兩種類型的“聯合”。

  “聯合”與“結構”有一些相似之處。但兩者有本質上的不同。在結構中各成員有各自的內存空間, 一個結構變量的總長度是各成員長度之和。而在 “聯合”中,各成員共享一段內存空間, 一個聯合變量的長度等於各成員中最長的長度。應該說明的是, 這裏所謂的共享不是指把多個成員同時裝入一個聯合變量內, 而是指該聯合變量可被賦予任一成員值,但每次只能賦一種值, 賦入新值則衝去舊值。如前面介紹的“單位”變量, 如定義爲一個可裝入“班級”或 “教研室”的聯合後,就允許賦予整型值(班級)或字符串(教研室)。要麼賦予整型值,要麼賦予字符串,不能把兩者同時賦予它。聯合類型的定義和聯合變量的說明一個聯合類型必須經過定義之後, 才能把變量說明爲該聯合類型。

  一、聯合的定義

  定義一個聯合類型的一般形式爲:

  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 。不允許只用聯合變量名作賦值或其它操作。 也不允許對聯合變量作初始化賦值,賦值只能在程序中進行? 掛僨康魎得韉氖?一個聯合變量, 每次只能賦予一個成員值?瘓浠八?一個聯合變量的值就是聯合變員的某一個成員值。

  設有一個教師與學生通用的表格,教師數據有姓名,年齡,職業,教研室四項。學生有姓名,年齡,職業,班級四項。

  編程輸入人員數據, 再以表格輸出。

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[i].name是一個數組類型,第22行中的body[i].depa.office也是數組類型,因此在這兩項之間不能加”&”運算符。程序中的第二個for語句用於輸出各成員項的值:

  本章小結

  1. 結構和聯合是兩種構造類型數據,是用戶定義新數據類型的重要手段。結構和聯合有很多的相似之處,它們都由成員組成。成員可以具有不同的數據類型。成員的表示方法相同。都可用三種方式作變量說明。

  2. 在結構中,各成員都佔有自己的內存空間,它們是同時存在的。一個結構變量的總長度等於所有成員長度之和。在聯合中,所有成員不能同時佔用它的內存空間,它們不能同時存在。聯合變量的長度等於最長的成員的長度。

  3. “.”是成員運算符,可用它表示成員項,成員還可用“->”運算符來表示。

  4. 結構變量可以作爲函數參數,函數也可返回指向結構的指針變量。而聯合變量不能作爲函數參數,函數也不能返回指向聯合的指針變量。但可以使用指向聯合變量的指針,也可使用聯合數組。

  5. 結構定義允許嵌套,結構中也可用聯合作爲成員,形成結構和聯合的嵌套。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章