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. 数据结构7——BFS

    一.重拾关键 宽度优先搜索,也有称为广度优先搜索,简称BFS.类似于树的按层次遍历的过程. 初始状态:图G所有顶点均未被访问过,任选一点v. 遍历过程:假设从图中某顶点v出发,在访问了v之后依次访问v ...

  2. Python两个内置函数——locals 和globals (学习笔记)

    这两个函数主要提供,基于字典的访问局部和全局变量的方式.在理解这两个函数时,首先来理解一下python中的名字空间概念.Python使用叫做名字空间的东西来记录变量的轨迹.名字空间只是一个字典,它的键 ...

  3. 【iOS开发】UIView之userInteractionEnabled属性介绍

    http://my.oschina.net/hmj/blog/108002 属性作用 该属性值为布尔类型,如属性本身的名称所释,该属性决定UIView是否接受并响应用户的交互. 当值设置为NO后,UI ...

  4. C#中的Stack的Peek操作,曝出异常

    我们在遍历一个栈的时候,有时候需要判断栈顶元素,用到了Peek元素,然后再用Pop元素,但是这个时候会出现一个逻辑错误, 当用Pop删除全部栈的元素时,再用Peek就会报错, InvalidOpera ...

  5. Delphi 之 编辑框控件(TEdit)

    TEdit 组件主要用于数据的输入和显示和编辑等操作. AutoSelect 获取组件焦点.该属性只能在单行文本组件使用.值为True为选中.false则不选中. BorderStyle 设置编辑框控 ...

  6. 【题解】WC2008游览计划

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

  7. BZOJ 2500 幸福的道路(race) 树上直径+平衡树

    structHeal { priority_queue<int> real; priority_queue<int> stack; void push(int x){ real ...

  8. 线程--promise furture 同步

    http://www.cnblogs.com/haippy/p/3279565.html std::promise 类介绍 promise 对象可以保存某一类型 T 的值,该值可被 future 对象 ...

  9. debounce 与 throttle 区别

    原文地址:http://undefinedblog.com/debounce-and-throttle/ 二.什么是debounce    1. 定义 如果用手指一直按住一个弹簧,它将不会弹起直到你松 ...

  10. lwIP RAW_API

    lwIP RAW TCP/IP接口 作者: Adam Dunkels, Leon Woestenberg, Christiaan Simons lwIP为使用TCP/IP协议通信的应用程序编程提供了两 ...