Pascal中的指针运算指南(与C比较)

 转载(http://hi.baidu.com/xiaoduo170/blog/item/a4ef7ee9f6ef2e24ba0e2d6d.html

大家都认为,c语言之所以强大,以及其自由性,很大部分体现在其灵活的指针运用上。因此,说指针是c语言的灵魂,一点都不为过。同时,这种说法也让很多人产生误解,似乎只有c语言的指针才能算指针。basic不支持指针,在此不论。其实,pascal语言本身也是支持指针的。从最初的pascal发展至今的object pascal,可以说在指针运用上,丝毫不会逊色于c语言的指针。

以下内容分为八部分,分别是
一、类型指针的定义
二、无类型指针的定义
三、指针的解除引用
四、取地址(指针赋值)
五、指针运算
六、动态内存分配
七、字符数组的运算
八、函数指针

一、类型指针的定义。对于指向特定类型的指针,在c中是这样定义的:
int *ptr;
char *ptr;
与之等价的object pascal是如何定义的呢?
var
ptr : ^integer;
ptr : ^char;
其实也就是符号的差别而已。

二、无类型指针的定义。c中有void *类型,也就是可以指向任何类型数据的指针。object pascal为其
定义了一个专门的类型:pointer。于是,
ptr : pointer;
就与c中的
void *ptr;
等价了。

三、指针的解除引用。要解除指针引用(即取出指针所指区域的值),c 的语法是 (*ptr),object
pascal则是 ptr^。

四、取地址(指针赋值)。取某对象的地址并将其赋值给指针变量,c 的语法是
ptr = &object;
object pascal 则是
ptr := @object;
也只是符号的差别而已。

五、指针运算。在c中,可以对指针进行移动的运算,如:
char a[20];
char *ptr=a;
ptr++;
ptr+=2;
当执行ptr++;时,编译器会产生让ptr前进sizeof(char)步长的代码,之后,ptr将指向a[1]。ptr+=2;这句使得ptr前进两个sizeof(char)大小的步长。同样,我们来看一下object pascal中如何实现:

vara : array [1..20] of char;

ptr : pchar; //pchar 可以看作 ^char

begin

ptr := @a;inc(ptr); // 这句等价于 c 的 ptr++;

inc(ptr, 2); //这句等价于 c 的 ptr+=2;

end;

只是,pascal中,只允许对有类型的指针进行这样的运算,对于无类型指针是不行的。

六、动态内存分配。c中,使用malloc()库函数分配内存,free()函数释放内存。如这样的代码:

int *ptr, *ptr2;

int i;

ptr = (int*) malloc(sizeof(int) * 20);

ptr2 = ptr;

for (i=0; i<20; i++){

*ptr = i;

ptr++;

}

free(ptr2);

object pascal中,动态分配内存的函数是getmem(),与之对应的释放函数为freemem()(传统pascal中获取内存的函数是new()和 dispose(),但new()只能获得对象的单个实体的内存大小,无法取得连续的存放多个对象的内存块)。因此,与上面那段c的代码等价的object pascal的代码为:
var ptr, ptr2 : ^integer;i : integer;

begin

getmem(ptr, sizeof(integer) * 20);

//这句等价于c的 ptr = (int*) malloc(sizeof(int) * 20);

ptr2 := ptr; //保留原始指针位置

for i := 0 to 19 do

begin

ptr^ := i;

inc(ptr);

end;

freemem(ptr2);

end;

对于以上这个例子(无论是c版本的,还是object pascal版本的),都要注意一个问题,就是分配内存的单位是字节(byte),因此在使用getmem时,其第二个参数如果想当然的写成 20,那么就会出问题了(内存访问越界)。因为getmem(ptr, 20);实际只分配了20个字节的内存空间,而一个整形的大小是四个字节,那么访问第五个之后的所有元素都是非法的了(对于malloc()的参数同样)。

七、字符数组的运算。c语言中,是没有字符串类型的,因此,字符串都是用字符数组来实现,于是也有一套str打头的库函数以进行字符数组的运算,如以下代码:

char str[15];

char *pstr;

strcpy(str, "teststr");

strcat(str, "_testok");

pstr = (char*) malloc(sizeof(char) * 15);

strcpy(pstr, str);

printf(pstr);

free(pstr);

而在object pascal中,有了string类型,因此可以很方便的对字符串进行各种运算。但是,有时我们的pascal代码需要与c的代码交互(比如:用object pascal的代码调用c写的dll或者用object pascal写的dll准备允许用c写客户端的代码)的话,就不能使用string类型了,而必须使用两种语言通用的字符数组。其实,object pascal提供了完全相似c的一整套字符数组的运算函数,以上那段代码的object pascal版本是这样的:

var str : array [1..15] of char;

pstr : pchar; //pchar 也就是 ^char

begin

strcopy(@str, 'teststr'); //在c中,数组的名称可以直接作为数组首地址指针来用

//但pascal不是这样的,因此 str前要加上取地址的运算符

strcat(@str, '_testok');

getmem(pstr, sizeof(char) * 15);

strcopy(pstr, @str);

write(pstr);

freemem(pstr);

end;

八、函数指针。在动态调用dll中的函数时,就会用到函数指针。假设用c写的一段代码如下:

typedef int (*pvfn)(int); //定义函数指针类型

int main()

{

hmodule hmodule = loadlibrary("test.dll");

pvfn pvfn = null;

pvfn = (pvfn) getprocaddress(hmodule, "function1");

pvfn(2);

freelibrary(hmodule);}

就我个人感觉来说,c语言中定义函数指针类型的typedef代码的语法有些晦涩,而同样的代码在object pascal中却非常易懂:

type pvfn = function (para : integer) : integer;

var

fn : pvfn;

//也可以直接在此处定义,如:fn : function (para:integer):integer;

hm : hmodule;

begin

hm := loadlibrary('test.dll');

fn := getprocaddress(hm, 'function1');

fn(2);

freelibrary(hm);

end;

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值