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. BFS实现模板

    以如下图的无向图G4为例,进行图的深度优先搜索: 假设从顶点v1出发进行搜索,首先访问v1和v1的邻接点v2和v3,然后依次访问v2的邻接点v4和v5及v3的邻接点v6和v7,最后访问v4的邻接点v8 ...

  2. iterator 的设计原则和traits

    iterator我前面写过是作为algorithm和container之间的一个桥梁,algorithm进程操作的时候向iterator进行提问,iterator并对提问进行了回答,其中主要就是回答5 ...

  3. lintcode-111-爬楼梯

    111-爬楼梯 假设你正在爬楼梯,需要n步你才能到达顶部.但每次你只能爬一步或者两步,你能有多少种不同的方法爬到楼顶部? 样例 比如n=3,1+1+1=1+2=2+1=3,共有3中不同的方法 返回 3 ...

  4. java线程(5)——线程池(上)

    引入: 在之前的例子中,我们需要使用线程时就直接去创建一个线程,这样既不浪费资源又十分方便.但如果我们需要创建多个并发的线程,而且短时间执行就结束了,如果还用之前的方式,就会大大降低效率和性能了. 因 ...

  5. 从零开始配置Jenkins(二)——常见问题及排错思路

    [前言] 一年多以前就听说Jenkins了,那时知道是它可以完成自动构建,感觉蛮强大的.后来,很多人都说它很恶心.最近,公司需要搭建新的服务器,小编就负责从头开始配置并且发布部署成功每一条线每一个项目 ...

  6. Ubuntu desktop基本操作

    2018-03-03 11:48:52 ubuntu16 lts 更换源,系统安装的时候可以跳过语言包的安装 打开software & updates应用,Other software选项页, ...

  7. elementUI中的el-form怎么使用正则进行验证

    http://element.eleme.io/#/zh-CN/component/form里给出了一个form验证的例子,但是都是使用el-form带有的验证规则,怎么使用自己定义的规则进行验证呢? ...

  8. 在submit_bio处使用stapn

    想着在submit_bio的地方,发现在guru模式下,stap是经常性地把内核整挂呀,不得已,也没有发现stap什么比较好的调试方法,所以索性直接使用stap的语法了,但是发现有问题呢,有的时候bv ...

  9. struts2 下载文件

    作者:禅楼望月 当下载的文件名字中不含有汉字,或者下载的文件不需要考虑用户的权限问题时.直接让超链接的href属性为所要下载的文件名即可.否则最好使用struts2的文件下载机制. 以下载图片为例 完 ...

  10. 全局 Ajax 事件处理器

    jQuery中将Ajax请求和响应分成了若干(5)个阶段 并且允许开发者在Ajax请求和响应的不同阶处理不同的逻辑, 这些方法用于注册事件处理器,用来处理页面上的任何 Ajax 请求,当某些事件触发后 ...