曙海教育集團(tuán)論壇開發(fā)語言培訓(xùn)專區(qū)C++語言開發(fā) → C(不討論C++)語言


  共有7110人關(guān)注過本帖樹形打印

主題:C(不討論C++)語言

美女呀,離線,留言給我吧!
wangxinxin
  1樓 個性首頁 | 博客 | 信息 | 搜索 | 郵箱 | 主頁 | UC


加好友 發(fā)短信
等級:青蜂俠 帖子:1393 積分:14038 威望:0 精華:0 注冊:2010-11-12 11:08:23
C(不討論C++)語言  發(fā)帖心情 Post By:2010-12-10 14:26:54

C語言中的名字空間, 較少被提及. 下面的寫法乍看之下是會讓人吃驚的:

#include <stdio.h>

struct Foo
{
    int table_id;
    signed int length:4;
} ;

typedef struct Foo Foo;
int main()
{
    Foo Foo;
    printf("size: %lld ", (long long) sizeof(Foo));
    // Foo t1 = {0};
    return 0;
}

Foo 首先是一個struct的tag名字, 其次又被typedef定義了一個同名的別名. 然后, 在main函數(shù)中,
以Foo Foo; 定義了該類型的一個變量, 同樣名為Foo.

這樣的程序竟然是符合標(biāo)準(zhǔn)的. 原因就在于C語言中有4個名字空間, 當(dāng)標(biāo)識符在不同的上下文情境下位于不同的名字空間時, 可同時出現(xiàn)而不會引起沖突.

我查看了C語言標(biāo)準(zhǔn), 6.2.3 Name Spaces and Identifiers
其中定義的4個名字空間如下:
1. label 單獨位于一個名字空間, 由于goto有害論, label受到牽連, 現(xiàn)今其重要性極低.
2. struct, union  , enum的名字, 在C標(biāo)準(zhǔn)中用tag一詞指代, 它們的名字位于一個名字空間, 也就是說, 如果你已經(jīng)
struct Foo { ... };
就不能再
enum Foo {... };

3. struct, 或union  的成員, 位于由相應(yīng)的struct或union  聲明范圍內(nèi)的一個密閉名字空間, 兩個不同的struct, 或struct與union  的成員, 可以有同樣的名字, 這一規(guī)則可以遞歸地施行于struct / union  的子成員. 如果它們本身也是一個struct或union  的話.

4. 所有其它的一切東西, 比如函數(shù)名, 變量名等等.

根據(jù)這4條, 上面的程序該如何解釋? Foo 重復(fù)出現(xiàn)了3次:
struct Tag.
typedef 或
main內(nèi)的變量名.

根據(jù)上面的定義, 作為typedef定義出來的類型名和main內(nèi)的變量名同屬于"其它"類, 應(yīng)該會出現(xiàn)沖突. 但實際上這樣的用法是允許的. 因為在main內(nèi)通過
Foo Foo;
定義變量Foo時, 第一個Foo的語意只能是typedef定義出來的Foo才合理, 此時作為變量的Foo還沒定義完成, 所以沒有沖突.

那一行注釋起來的
// Foo t1...
如果去掉注釋, 就會引起編譯錯誤,

test.c:15: error: expected ';' before 't1'
gcc的這條錯誤并沒提供多少有用的信息.

因為在此時的上下文中, 就有了兩個identifier位于同一個名字空間. 而printf中的sizeof(Foo) 究竟是作為typedef定義出來的別名Foo, 還是變量名Foo.

雖然無法從程序運行結(jié)果上知道, 但可以確定應(yīng)該是變量名Foo, 簡單的實驗加推理可以證實這一點:
將Foo Foo改為char Foo;
此時大小變?yōu)?.

推理:
typdef定義的別名其作為域在最外層, 而在main內(nèi), 變量Foo暫時性地遮蔽了外層Foo的意義.
對上面程序作如下修改, 得到的結(jié)果可以證實:

int main()
{
    {
        char Foo;
        int a = 4, b = a;
        printf("size: %lld ", (long long)sizeof(Foo) );
    }
    printf("size: %lld ", (long long)sizeof(Foo) );
    struct Foo t1 = {0};
    return 0;
}

另一個需要注意的地方是, C語言中定義的結(jié)構(gòu)的可見性, 是平坦的,
struct Foo
{
   struct Bar { ... };
};

熟悉C++類型系統(tǒng)的人可能會懷疑是否能直接使用結(jié)構(gòu)Bar, 要不要Foo::, C里面Bar的可用性跟Foo是平級的.本篇文章


支持(0中立(0反對(0單帖管理 | 引用 | 回復(fù) 回到頂部

返回版面帖子列表

C(不討論C++)語言








簽名
主站蜘蛛池模板: 别揉我奶头~嗯~啊~视频在线观看| 亚洲综合色成在线播放| 日本黄大片在线观看| 成人免费小视频| 亚洲AV无码之日韩精品| 国产精品香蕉在线观看不卡| 永久看一二三四线| 99久久国产综合精品五月天| 亚洲精品在线免费观看| 国产精品真实对白精彩久久| 欧美激情第1页| 亚洲伊人tv综合网色| 久热这里只有精品视频6| 国产伦精品一区二区| 我的好妈妈6中字在线观看韩国| 精品无码久久久久久久动漫| 一区二区三区中文字幕| 亚洲精品成人片在线播放| 国产精品亚洲а∨无码播放不卡 | 久章草在线精品视频免费观看| 国产区精品一区二区不卡中文| 成人免费看www网址入口| 波多野结衣在丈夫面前| 中文字幕日韩丝袜一区| 久久99国产精品视频| 亚洲精品国产品国语在线| 国产精品白丝喷水在线观看| 欧美人与动人物姣配xxxx| 国产卡一卡二卡3卡4卡无卡视频 | 国产高中生粉嫩无套第一次| 拧花蒂尿用力按凸起喷水尿| 污视频免费看软件| 色综合色综合色综合色综合网| 99re国产精品| 中文字幕第15页| 亚洲人av高清无码| 免费看美女脱衣服| 国产在线无码精品无码| 国农村精品国产自线拍| 成年人在线免费观看网站| 欧洲一级毛片免费|