- 论坛徽章:
- 0
|
请问差别在何处
pmerofc 发表于 2010-10-19 07:38 ![]()
TC是16位程序,生成代码也是16位代码。有6中模式tiny,small,medium,compact, larger,huge,3种地址关键字far、near、huge。
以下是抄来的。
在tiny、small模式下,所有的函数定义、全局变量定义和指针变量的定义,如果没有显示的加上far、near、huge等关键字,都默认为使用了near关键字;
在medium模式下,函数定义默认使用了far关键字,变量定义默认使用了near关键字;
在compact模式下函数定义模式使用了near关键字,变量定义默认使用了far关键字;
在large模式下函数定义和变量定义模认使用了far关键字;
在huge模式下函数定义模认使用了far关键字,变量定义默认使用了huge关键字。
near、far、huge关键字的真正含义是什么?这三个关键字只能用于修饰函数、全局变量和指针变量,对于非指针类型的局部变量,这些关键字没有实际意义。
当这三个关键字用于修饰函数时,huge的含义与far相同,用于指明该函数的调用方式为far调用方式,即调用时需要一个段值和一个段偏移组成的32bits调用地址,使用far
call进行跳转,跳转前先压栈保存当前CS:IP。near修饰函数时,用于指明该函数的调用方式为near调用方式,调用时先压栈保存当前IP,需要一个16bits的近地址,即当前CS的段
内偏移。
当这三个关键字用于修饰指针时,near型指针实质上为16bits的无符号整型数,该整数给出了所指向变量在当前数据段内的偏移地址,也就是说,在使用near型指针寻址时
实际上是进行如下的寻址操作:[DS:指针变量值]。对于far型的指针变量,可以寻址1MB地址空间的任意一个地方,far型指针的实质是一个32bits的整型数,高16bits为段值,低
16bits为段内偏移,Turbo C中在使用far型指针时,会先将高16bits放入ES寄存器中,然后再进行如下的寻址操作:[ES:指针变量低16bits值]。对于hug型的指针变量,与far型指
针变量的不同之处在于,在对far型指针变量进行+/++/-/--等操作时,far型指针变量保持段值不变(也就是高16bits),而只对段内偏移进行加减操作,所以会出现段内回绕的现
象,而huge型的指针,在进行加减操作时将会自动的改变段值,不会出现段内回绕。所以给人的感觉就是huge指针能比far指针寻址更大的内存空间。
对于局部变量,由于是创建在堆栈上,所以near、far等关键字将不具备任何意义,因为创建在堆栈上的变量的寻址方式就只有一种,即使用sp和bp维护函数堆栈,利用
bp+/-一个偏移来寻址函数参数变量和局部变量。这样的寻址方式是固定而唯一的,near和far等关键字都派不上用场,这里的near和far将没有任何的实际含义。
对于使用near、far和huge修饰的全局变量的含义也很容易理解了。near型的全局变量,被分配到了当前的数据段上,寻址这个变量只需要一个16bits的偏移量,而far型全
局变量在寻址时,需要给出段值和偏移量。huge型数组可以使用大于64K的内存空间。
far、near、huge型指针变量之间的相互转换,从小尺寸的指针到大尺寸的指针将进行自动的类型转换,转换方式为加上当前的DS形成32bits的指针。从大尺寸的指针到小尺
寸的指针需要进行强制类型转换,转换的结果为只保留低16bits,但是这样俄转换没有实际的意义或者说用处不大,并且极其容易引入内存访问的错误,所以要严格避免使用。 |
|