1. 程式人生 > >遠指標和近指標,還有巨指標到底是啥玩意

遠指標和近指標,還有巨指標到底是啥玩意

char near p; /定義一個字元型“近”指標*/
char far p; /定義一個字元型“遠”指標*/
char huge p; /定義一個字元型“巨”指標*/

首先要從8086處理器體系結構和彙編淵源講起。大家知道,8086是一個16位處理器,它設定了四個段暫存器,專門用來儲存段地址:CS(Code Segment):程式碼段暫存器;DS(Data Segment):資料段暫存器;SS(Stack Segment):堆疊段暫存器;ES(Extra Segment):附加段暫存器。8086採用段式訪問,訪問本段(64K範圍內)的資料或指令時,不需要變更段地址(意味著段地址暫存器不需修改),而訪問本段範圍以外的資料或指令時,則需要變更段地址(意味著段地址暫存器需要修改)。
因此,在16位處理器環境下,如果訪問本段內地址的值,用一個16位的指標(表示段內偏移)就可以訪問到;而要訪問本段以外地址的值,則需要用16位的段內偏移+16位的段地址,總共32位的指標。
這樣,我們就知道了遠、近指標的區別:
**近指標是隻能訪問本段、只包含本段偏移的、位寬為16位的指標;
遠指標是能訪問非本段、包含段偏移和段地址的、位寬為32位的指標。**
近指標只能對64k位元組資料段內的地址進行存取,如:
char near *p;
p=(char near *)0xffff;
遠指標是32位指標,它表示段地址:偏移地址,遠指標可以進行跨段定址,可以訪問整個記憶體的地址。如定義遠端指標p指向0x1000段的0x2號地址,即1000:0002,則可寫作:
char far *p;
p=(char far *)0x10000002;
除了遠指標和近指標外,還有一個巨指標的概念。
和遠指標一樣,巨指標也是32位的指標,指標也表示為16位段:16位偏移,也可以定址任何地址。它和遠指標的區別在於進行了規格化處理。

遠指標沒有規格化,可能存在兩個遠指標實際指向同一個實體地址,但是它們的段地址和偏移地址不一樣,如23B0:0004和23A1:00F4都指向同一個實體地址23604!巨指標通過特定的例程保證:每次操作完成後其偏移量均小於10h,即只有最低4位有數值,其餘數值都被進位到段地址上去了,這樣就可以避免Far指標在64K邊界時出乎意料的迴繞的行為。當然,一次操作必須小於64K。下面的函式可以將遠指標轉換為巨指標:
void normalize(void far ** p)
{
  p=(void far )(((long)*p&0xffff000f)+(((long)*p&0x0000fff00<<12));
}
從上面的函式中我們再一次看到了指標之指標的使用,這個函式要修改指標的值,因此必須傳給它的指標的指標作為引數。
講到這裡,筆者要強調的是:近指標、遠指標、巨指標是段定址的16bit處理器的產物(如果處理器是16位的,但是不採用段定址的話,也不存在近指標、遠指標、巨指標的概念),當前普通PC所使用的32bit處理器(80386以上)一般執行在保護模式下的,指標都是32位的,可平滑地址,已經不分遠、近指標了。但是在嵌入式系統領域下,8086的處理器仍然有比較廣泛的市場,如AMD公司的AM186ED、AM186ER等處理器,開發這些系統的程式時,我們還是有必要弄清楚指標的定址範圍。