欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

舉例講解C語言鏈接器的符號(hào)解析機(jī)制

 更新時(shí)間:2016年05月31日 18:45:12   作者:wudaijun  
鏈接器的工作主要分為兩個(gè)階段:符號(hào)解析和重定位,符號(hào)解析的功能是將每個(gè)模塊符號(hào)引用綁定到一個(gè)確切的符號(hào)定義,這里我們就來舉例講解C語言鏈接器的符號(hào)解析機(jī)制

1. 符號(hào)分類
(1)全局符號(hào):非靜態(tài)全局變量,非靜態(tài)函數(shù)
(2)外部符號(hào):定義于其它模塊,而被本模塊引用的全局變量和函數(shù)
(3)本地符號(hào):靜態(tài)變量(包括全局和局部),靜態(tài)函數(shù)
對(duì)于靜態(tài)局部變量,編譯器會(huì)為其生成唯一的名字。如x.fun1,x.fun2。本地符號(hào)對(duì)鏈接器來說是不可見的。
2. 符號(hào)決議
當(dāng)編譯器遇到一個(gè)不是本模塊定義的符號(hào)時(shí),會(huì)假設(shè)該函數(shù)由其它模塊定義,并生成一個(gè)鏈接器符號(hào)表?xiàng)l目,交由鏈接器處理。如果鏈接器在它的任何輸入模塊都沒有找到該符號(hào),會(huì)給出一個(gè)類似undefined reference to 'xxx'的鏈接錯(cuò)誤。而如果鏈接器在輸入模塊中找到了一個(gè)以上的外部符號(hào)定義,這個(gè)時(shí)候就需要鏈接器進(jìn)行符號(hào)決議,鏈接器對(duì)多個(gè)外部符號(hào)定義可能并不報(bào)錯(cuò)甚至警告,而是按照它的規(guī)則去選擇其中一個(gè)符號(hào)定義。
鏈接器將各個(gè)模塊輸出的全局符號(hào),分類為強(qiáng)符號(hào)和弱符號(hào):
(1)強(qiáng)符號(hào):函數(shù)和已初始化的全局變量
(2)弱符號(hào):為初始化全局變量
根據(jù)強(qiáng)弱符號(hào)的定義,鏈接器按照下面的規(guī)則處理多重定義的符號(hào):
規(guī)則1:不允許有多個(gè)強(qiáng)符號(hào)定義
規(guī)則2:如果有一個(gè)強(qiáng)符號(hào)和多個(gè)弱符號(hào),那么選擇強(qiáng)符號(hào)
規(guī)則3:如果有多個(gè)弱符號(hào),那么從這些弱符號(hào)中選擇sizeof大的那個(gè),如果大小相同,則選擇先鏈接的那個(gè)
上面的規(guī)則是很多鏈接錯(cuò)誤的根源,因?yàn)榫幾g器在決議時(shí)可能默默地替你作出了決定,你并不知曉。根據(jù)上面的規(guī)則,可以引出下面幾個(gè)經(jīng)典例子:
例1:

// in lib1.c
int x;
void f()
{
  x = 1235;
}

// in main1.c
#include<stdio.h>
void f(void);

int x = 1234;

int main(void)
{
  f();
  printf("x=%d\n", x);
  return 0;
}

上面的代碼中,main函數(shù)printf輸出: x=1235。因?yàn)殒溄悠魍ㄟ^規(guī)則2決議符號(hào)x的定義為main.c中的強(qiáng)符號(hào)定義,而lib.c的作者并不知情,他對(duì)x的使用和修改影響到了main.c。這種交互修改,相互影響將會(huì)很復(fù)雜,因?yàn)榇蠹叶家詾樽约涸谧鰧?duì)的事情,在用對(duì)的變量。而整個(gè)決議過程,鏈接器悄無聲息地完成了。
例2:

// in lib2.c
double x;
void f()
{
  x = -0.0;
}

// in main2.c
#include<stdio.h>
void f(void);

int x = 1234;
int y = 1235;

int main()
{
  f();
  printf("x=0x%x y=0x%x \n", x, y);
  return 0;
}

這種情況下,程序得到輸出: x=0x0 y=0x80000000,而鏈接器(gcc ld)也終于給出一條警告:

復(fù)制代碼 代碼如下:

ld: warning: tentative definition of '_x' with size 8 from 'obj/Debug/lib2.o' is being replaced by real definition of smaller size 4 from 'obj/Debug/main2.o'


鏈接器決議的是符號(hào)地址,而相鄰的全局變量可能在.data段中的內(nèi)存地址也相鄰,因此也就引發(fā)了更復(fù)雜的問題。這一點(diǎn)和棧溢出很像,但是比棧溢出更復(fù)雜,因?yàn)閱栴}出在多個(gè)模塊之間,而不是在一個(gè)函數(shù)內(nèi)部。
例3:

// in lib3.c
struct
{
  int a;
  int b;
} x;

void f()
{
  x.a = 123;
  x.b = 456;
  printf("in f(): sizeof(x)=%d, (&x)=0x%08x\n", sizeof(x), &x);
}

// in main3.c
#include<stdio.h>
void f(void);

int x;
int y;

int main()
{
  f();
  printf("in main(): sizeof(x)=%d, (&x)=0x%08x, (&x)=0x%08x, x=%d,y=%d \n", sizeof(x), &x, &y, x, y);
  return 0;
}

程序輸出:

in f(): sizeof(x)=8, (&x)=0x02489018
in main(): sizeof(x)=4, (&x)=0x02489018, (&y)=0x02489020, x=123,y=0

始終記住,外部符號(hào)決議的是地址,因此無論lib3.c和main3.c中,符號(hào)x地址都是唯一的,無論其被定義了幾次。其次sizeof是編譯器決議,與鏈接無關(guān),編譯器只看得到本模塊的定義或聲明。最后,由于符號(hào)x決議到lib3.c中的x,其size是8,因此main3.c中的y的地址比x大8,這是由鏈接器將lib3.o和main3.o合并后填入可執(zhí)行文件的.data段的。因此y是無關(guān)變量,被初始化為0,注意和例2的區(qū)別。
3. 總結(jié)
由于符號(hào)決議容易引發(fā)的種種問題,我們?cè)趯慍的時(shí)候應(yīng)注意:
盡量用static屬性隱藏變量和函數(shù)在模塊內(nèi)的聲明,就像在C++中盡量用private保護(hù)類私有成員一樣。
少定義弱符號(hào),盡量初始化全局變量,這樣鏈接器會(huì)根據(jù)規(guī)則1給出多個(gè)符號(hào)定義的錯(cuò)誤。
為鏈接器設(shè)置必要選項(xiàng),如gcc的 -fno-common,這樣在遇到多重符號(hào)定義時(shí),鏈接器會(huì)給出警告。
4. C++的符號(hào)決議
C++并不支持強(qiáng)弱符號(hào)同時(shí)存在,所有符號(hào)都只能有一個(gè)定義(函數(shù)重載通過改寫函數(shù)符號(hào)來確保其唯一),因此在很大程度上避免了C中的鏈接器困擾。

相關(guān)文章

  • Qt+QListWidget實(shí)現(xiàn)氣泡聊天界面(附源碼)

    Qt+QListWidget實(shí)現(xiàn)氣泡聊天界面(附源碼)

    由于最近的項(xiàng)目需要,做了些相關(guān)IM的工作。所以聊天框也是必不可少的一部分。本文以QListWidget+QPainter繪制的Item做了一個(gè)Demo。該Demo只是做一個(gè)示例,感興趣的可以了解一下
    2022-12-12
  • C++利用PCL點(diǎn)云庫操作txt文件詳解

    C++利用PCL點(diǎn)云庫操作txt文件詳解

    這篇文章主要為大家詳細(xì)介紹了C++如何利用PCL點(diǎn)云庫操作txt文件,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,感興趣的小伙伴可以了解一下
    2024-01-01
  • 用C語言實(shí)現(xiàn)簡單掃雷小游戲

    用C語言實(shí)現(xiàn)簡單掃雷小游戲

    這篇文章主要為大家詳細(xì)介紹了用C語言實(shí)現(xiàn)簡單掃雷小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-07-07
  • boost.asio框架系列之socket編程

    boost.asio框架系列之socket編程

    這篇文章介紹了boost.asio框架系列之socket編程,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-06-06
  • 基于VC編寫COM連接點(diǎn)事件的分析介紹

    基于VC編寫COM連接點(diǎn)事件的分析介紹

    本篇文章是對(duì)VC編寫COM連接點(diǎn)事件進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • c++ lambda捕獲this 導(dǎo)致多線程下類釋放后還在使用的錯(cuò)誤問題

    c++ lambda捕獲this 導(dǎo)致多線程下類釋放后還在使用的錯(cuò)誤問題

    Lambda表達(dá)式是現(xiàn)代C++的一個(gè)語法糖,挺好用的。但是如果使用不當(dāng),會(huì)導(dǎo)致內(nèi)存泄露或潛在的崩潰問題,這里總結(jié)下c++ lambda捕獲this 導(dǎo)致多線程下類釋放后還在使用的錯(cuò)誤問題,感興趣的朋友一起看看吧
    2023-02-02
  • swift Character類型詳解及實(shí)例

    swift Character類型詳解及實(shí)例

    這篇文章主要介紹了 swift Character類型詳解及實(shí)例的相關(guān)資料,需要的朋友可以參考下
    2017-06-06
  • 結(jié)構(gòu)體對(duì)齊的規(guī)則詳解及C++代碼驗(yàn)證

    結(jié)構(gòu)體對(duì)齊的規(guī)則詳解及C++代碼驗(yàn)證

    在c語言的結(jié)構(gòu)體里面一般會(huì)按照某種規(guī)則去進(jìn)行字節(jié)對(duì)齊。本文就來介紹一下如何實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解下
    2021-08-08
  • C++表達(dá)式求值詳解

    C++表達(dá)式求值詳解

    下面小編就為大家?guī)硪黄獪\談C++ 語言中的表達(dá)式求值。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2021-10-10
  • C語言實(shí)現(xiàn)學(xué)生信息管理系統(tǒng)開發(fā)

    C語言實(shí)現(xiàn)學(xué)生信息管理系統(tǒng)開發(fā)

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)學(xué)生信息管理系統(tǒng)開發(fā),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-08-08

最新評(píng)論