Delphi指针详解2007-12-04 06:08:57|  分类: DLL学习 阅读91 评论0   字号:大中小 订阅
大家都认为,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。于是, 字串7
   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中如何实现:
   var
     a : array [1..20] of Char;
     ptr : PChar; //PChar 可以看作 ^Char
   begin
     ptr := @a;
     Inc(ptr); // 这句等价于 C 的 ptr++;
     Inc(ptr, 2); //这句等价于 C 的 ptr+=2;
   end;
  
  六、动态内存分配。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;

Delphi指针详解的更多相关文章

  1. 【C++】智能指针详解(一):智能指针的引入

    智能指针是C++中一种利用RAII机制(后面解释),通过对象来管理指针的一种方式. 在C++中,动态开辟的内存需要我们自己去维护,在出函数作用域或程序异常退出之前,我们必须手动释放掉它,否则的话就会引 ...

  2. c/c++指针详解(一)

    一:相关概念 1.指针数组:int *p[6]               是数组,是一个存放指针的数组,也就是里面存放的是地址. 2.数组指针:int (*p)[6]                 ...

  3. c/c++指针详解(二)

    指针的概念 指针是一个特殊的变量,它里面存储的数值被解释成为内存里的一个地址.要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占 ...

  4. [转]C++ 智能指针详解

    转自:http://blog.csdn.net/xt_xiaotian/article/details/5714477 C++ 智能指针详解 一.简介 由于 C++ 语言没有自动内存回收机制,程序员每 ...

  5. Delphi 关键字详解

    Delphi 关键字详解[整理于 "橙子" 的帖子] absolute //它使得你能够创建一个新变量, 并且该变量的起始地址与另一个变量相同. var   Str: ];   S ...

  6. C++ 智能指针详解(转)

    C++ 智能指针详解   一.简介 由于 C++ 语言没有自动内存回收机制,程序员每次 new 出来的内存都要手动 delete.程序员忘记 delete,流程太复杂,最终导致没有 delete,异常 ...

  7. C++指针详解 (转)

    C++指针详解 指针的概念 指针是一个特殊的变量,它里面存储的数值被解释成为内存里的一个地址.要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区, ...

  8. 深入理解C语言 - 指针详解

    一.什么是指针 C语言里,变量存放在内存中,而内存其实就是一组有序字节组成的数组,每个字节有唯一的内存地址.CPU 通过内存寻址对存储在内存中的某个指定数据对象的地址进行定位.这里,数据对象是指存储在 ...

  9. Delphi多线程详解

    (整理自网络) Delphi多线程处理 1-1多线程的基本概念 WIN 98/NT/2000/XP 是个多任务操作系统,也就是:一个进程可以划分为多个线程,每个线程轮流占用CPU 运行时间和资源,或者 ...

随机推荐

  1. PTA实验报告(循环 数组 函数)

    一.循环PTA实验作业 题目一.7-2 求平方根序列前N项和 1.本题PTA提交列表 2.设计思路 本题调用了sqrt数学函数计算平方根,其次只用了一层循环,计算平方根之后使用循环累加计算总和sum. ...

  2. HTML文档插入JS代码的几种方法

    在HTML文档里嵌入客户端JavaScript代码有4中方法: 1.内联,放置在< script>和标签对之间. 2.放置在由< script>标签的src属性指定的外部文件中 ...

  3. requests快速入门

    Requests 是唯一的一个非转基因的 Python HTTP 库,人类可以安全享用. 警告:非专业使用其他 HTTP 库会导致危险的副作用,包括:安全缺陷症.冗余代码症.重新发明轮子症.啃文档症. ...

  4. C#中构造函数和析构函数的用法

    构造函数与析构函数是一个类中看似较为简单的两类函数,但在实际运用过程中总会出现一些意想不到的运行错误.本文将较系统的介绍构造函数与析构函数的原理及在C#中的运用,以及在使用过程中需要注意的若干事项.一 ...

  5. P1650 田忌赛马

    题目描述 我国历史上有个著名的故事: 那是在2300年以前.齐国的大将军田忌喜欢赛马.他经常和齐王赛马.他和齐王都有三匹马:常规马,上级马,超级马.一共赛三局,每局的胜者可以从负者这里取得200银币. ...

  6. 【题解】WC2008游览计划

    写的第一道斯坦纳树的题目.斯坦纳树在信息学中的应用一般为:在\(n\)个点之间给定\(k\)条边并相应的边权,求在保证给定\(m\)个点联通的条件下的最小边权和.解决此类问题的方法即为SPFA + 状 ...

  7. cf 442 D. Olya and Energy Drinks

    cf 442 D. Olya and Energy Drinks(bfs) 题意: 给一张\(n \times m(n <= 1000,m <= 1000)\)的地图 给出一个起点和终点, ...

  8. 【BZOJ 1724】[Usaco2006 Nov]Fence Repair 切割木板 堆+贪心

    堆对于stl priority_queue ,我们自己定义的类自己重载<,对于非自定义类我们默认大根堆,如若改成小根堆则写成std::priority<int,vector<int& ...

  9. POJ1459:Power Network(多源点多汇点的最大流)

    Power Network Time Limit: 2000MS   Memory Limit: 32768K Total Submissions: 31086   Accepted: 15986 题 ...

  10. BZOJ1051:受欢迎的牛(并查集 / Tarjan)

    1051: [HAOI2006]受欢迎的牛 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 8161  Solved: 4460 Description ...