学习AO,最重要的是理解“接口”这个概念。接口是什么?有什么具体作用?在多种计算机高级语言中,都可以看到“接口”这个术语,但基本上每一本书对“为什么使用接口”等重要文都都“语焉不详”,使得初学者往往不得要领。

认识接口,必须先要熟悉程序语言发展的历史才行,明白了程序语言发展的前世今生,才能知道这么多计算机语言为什么会是这个样子,为什么会是那种形式,它们之间为什么会有不同的区别产生。

计算机出现以后,数学家和计算机学家们相继开发了多种计算机高级语言,从Small-talk、Pascal、Basic、C、C++、Java到.NET平台上的各种语言,他们的发展步伐,可以看作是计算机语言从面向过程相面对象发展的一段历史。许多面向对象语言的教材都会告诉读者这一段历史,并宣称OO(Oriented-Object,面向对象)编程的优越性。从面向过程向面向对象转变的时候,需要注意的术语从“过程”、“函数”转变到了“类”“对象”上。

什么是接口?为什么要使用接口?对于一个没有编程经验的初学者而言,理解它们是非常困难的。

在了解这些问题的实质前,首先让我们明白一个概念——“粒度”。所谓粒度,其实质是一个程序中使用的代码单元的组合尺度。举一个例子,砂砾—砖块—房屋模块,修建一座房子有很多方法:如果不嫌麻烦,可以使用砂砾一点点来堆砌,或者将砂砾烧制为砖块砌筑,甚至直接从建筑工厂购买房屋的门窗墙组建来安装。这三种不同的方法代表了三种不同的组合尺度。砂砾是最小的单位,使用它搭建小的房屋还可以,但修建大型建筑,就无能为力了,这是因为砂砾太多不便于管理;砖块比砂砾聚合了一层,可以用来修建较大的房子;而房屋模板是最高的尺度,使用它可以快速地搭建大规模的房屋。这三种尺度的区别和联系,与程序员编写程序概念有很大的相似之处。

在Pascal中,这种面向过程语言的基本的单元是过程和函数,它们是程序中最小的组件。过程和函数可以实现最基本的代码重用,当某些固定功能被编写为过程和函数后,就可以在程序中任意调用它们而不必再需要的地方都写上长长的重复代码,这样的好处是显而易见的。在一些小型的程序里面,使用过程和函数是合适的,但是在大中型程序中,它们的弊端就显示出来,过程和函数的粒度还是太低。如果有一个系统中存在10 000个函数和过程,程序员将不得不花费大量的时间去寻找和维护他们,10000个没有任何关系或者关系错综复杂的函数和过程的管理难度是显而易见的,就好像一间拥有10000名员工的企业一样,如果没有任何部门和职务,人人都一样,这岂不乱套了。

面向对象语言的出现,其实就为了一个理由——提高编程的粒度。面向对象语言的基本单位是类(Class),她封装了数据成员(属性)和成员函数(方法),将最小组件的提高了一个等级,程序员需要直接操作的不是过程和函数,而是更高层次上的类。相当于公司把10000个员工分了很多部门,让不同的部门负责不同的事宜,这样公司终于可以走上正轨了。

这是逻辑的抽象。

做成了 CLASS 并没有解决编程中所有问题,新的问题随之而来,假设有一个部门的员工很多,可以做很多类型的工作,如何在部门之间实现更好的管理呢?好比有一个类,它提供了很多种方法和属性,这些方法和属性其实可以分为一群群,为不同的功能服务,但是类并没有做这个管理,她只是一个属性和方法的容器,在AO中,Map对象拥有很多类型的功能,像管理图层、管理元素、管理选择集、显示地图等,每种不同的功能群都有好多方法和属性,现在这些属性和方法是杂乱无章的,没有任何区别地堆积在一个类里面,当程序员需要寻找一个方法的时候,不得不遍历她们,这样做很不方便。

接口(InterFace)的出现,解决了这个问题,就是将类的内部属性和方法进行分类。例如在MAP类中可以做几个接口,在这些接口中定义不同功能群的方法和属性,MAP实现了这些接口,这样就可以使用接口进行定义,如:

Dim pGraphicsContainer As IGraphicsContainer

pGraphicsContainer=pActiveView.FocusMap

pGraphicsContainer对象现在可以使用的属性和方法就只能是IGraphicsContainer接口定义的那部分,而不能使用其它接口定义的方法和属性,那如何使用其他接口定义的属性和方法呢?这就是所谓的QI(Query InterFace)功能,即从对象的一个接口查询另一个接口定义的属性和方法,如:

Dim pActiveView as IActiveView

pActiveView=pGraphicsContainer

通过上面的操作,pActiveView现在就可以使用MAP类中IActiveView接口定义的属性和方法了,这就实现了在一个类的不同接口之间的转换。

接口是一种用来定义程序的协定。实现接口的类要与接口的定义严格一致。有了这个协定,系统就可以抛开编程语言的限制。接口可以从多个父接口继承,而类可以实现多个接口,接口可以包含方法、属性、事件和索引器,她本身并不提供她所定义的成员的实现,而只是指定实现该接口的类或接口必须提供的成员。在可以使用类的地方,都可以使用接口来替代,除了使用类产生一个对象外。

接口可以看作是一个特殊的类形式,除了不能别实例化为一个对象外,它可以实现类能够完成的任何任务,如声明对象为某种接口类型,接口也可以继承等。接口继承机制是非常有用的,如一个子类对象可以看作一个父类对象,接口也具备这样的特性,在很多时候程序员可以将一个字接口类型的对象定义为父接口类型的对象,从而实现一般化的操作,如:

Public Sub CreateGeometry(Byval pPolygon As IPolygon)

Public Sub CreateGeometry(Byval pPolygon As IGometry)

上面的两个过程的参数一个是使用IPolygon对象,另一个是使用IGometry对象,后者可以使用的更广泛和更安全一点,如果不慎传进去的是一个IPoint对象,在第二个方法里面也是合法的,因为IGometry的对象可以是任何一种几何形体对象,而这种做法在第一个过程中就是错误的。

一个类可以实现多个接口,一个接口也可以被多个类实现。使用形象一点的比拟方法,可以把类当成一个人,接口则当做一个身份。一个人可能有多个身份(一个类可以有多个接口),他可能是军人(军人接口定义军人的属性和行为),作家(作家接口定义作家的属性和行为)等,不同的身份使得他有不同的能力和属性。

一个身份(接口)也能够被多个人使用(一个接口可以被多个类实现)。比如一个军人身份可以给很多人,但是这些人完全可以用不同的能力和属性实现这个身份,他们可能是空军军官,也可能是海军列兵。身份只定义了一个人是什么,却不会告诉别人在这个身份下该这么做。怎么做是这个人本身的事情(接口仅仅定义相关的方法和属性而不实现她们)。

这是方法的具体。

计算机语言的发展历史,其实就是一部不断地寻找更好组件粒度的历史,不断提高代码重用的历史。以前程序员使用过程和函数,后来使用类、接口乃至包、命名空间等,都是为了一个目的,那就是让程序员能够操作的组件在具体和抽象之间寻找一个平衡点,这不是一件容易的事情——太具体了,如过程和函数,就没有了框架;太抽象了,如类,就无法深入细微处。
    这就是编程的哲学。

ArcGIS 编程中对接口的理解的更多相关文章

  1. Java网络编程中异步编程的理解

    目录 前言 一.异步,同步,阻塞和非阻塞的理解 二.异步编程从用户层面和框架层面不同角度的理解 用户角度的理解 框架角度的理解 三.为什么使用异步 四.理解这些能在实际中的应用 六.困惑 参考文章 前 ...

  2. 理解函数式编程中的函数组合--Monoids(二)

    使用函数式语言来建立领域模型--类型组合 理解函数式编程语言中的组合--前言(一) 理解函数式编程中的函数组合--Monoids(二) 继上篇文章引出<范畴论>之后,我准备通过几篇文章,来 ...

  3. 如何理解 TS 类型编程中的 extends 和 infer

    extends extends 在TS类型编程中用法(T extends U),表示 T 中的某些在 U 里面,比较难描述,用法如下: T extends U ? X : Y 分为两种情况理解更直观一 ...

  4. 你不知道的this—JS异步编程中的this

    Javascript小学生都知道了javascript中的函数调用时会 隐性的接收两个附加的参数:this和arguments.参数this在javascript编程中占据中非常重要的地位,它的值取决 ...

  5. Java EE 编程中路径

    版权声明:未经博主允许,不得转载 首先我们要限定一个范围,是一个项目,或是以个访问地址..就先以一个项目为限定的范围 前述: 学过物理学的都知道相对运动和绝对运动, 虽然是相似的概念,但这里的要简单得 ...

  6. Java编程中“为了性能”尽量要做到的一些地方

    最近的机器内存又爆满了,除了新增机器内存外,还应该好好review一下我们的代码,有很多代码编写过于随意化,这些不好的习惯或对程序语言的不了解是应该好好打压打压了. 下面是参考网络资源总结的一些在Ja ...

  7. linux中socket的理解

    对linux中socket的理解 一.socket 一般来说socket有一个别名也叫做套接字. socket起源于Unix,都可以用“打开open –> 读写write/read –> ...

  8. TCP/IP网络编程中socket的行为

    一. read/write的语义:为什么会阻塞? 先从write说起: #include <unistd.h>ssize_t write(int fd, const void *buf, ...

  9. 浅谈TCP/IP网络编程中socket的行为

    我认为,想要熟练掌握Linux下的TCP/IP网络编程,至少有三个层面的知识需要熟悉: 1. TCP/IP协议(如连接的建立和终止.重传和确认.滑动窗口和拥塞控制等等) 2. Socket I/O系统 ...

随机推荐

  1. 大数据平台-修改主机名及ssh免密码登录

    一.查看服务器初始配置: 1.总核数 = 物理CPU个数 X 每颗物理CPU的核数 2.总逻辑CPU数 = 物理CPU个数 X 每颗物理CPU的核数 X 超线程数 # 查看物理CPU个数cat /pr ...

  2. Sum(欧拉降幂+快速幂)

    Input 2 Output 2 Hint 1. For N = 2, S(1) = S(2) = 1. 2. The input file consists of multiple test cas ...

  3. eclipse svn 报错 文件夹已经不存在

    最近做项目用eclipse 遇到个很奇怪的问题,前几天svn还是可以用的,突然一下子不能用了,于是网上各种找解决方法啊,终于问题解决了,总结一下. 查看svn报错信息: svn number is l ...

  4. CentOS&.NET Core初试-4-安装守护服务(Supervisor)

    系列目录 CentOS的安装和网卡的配置 安装.NET Core SDK和发布网站 Nginx的安装和配置 安装守护服务(Supervisor) Supervisor是什么? Supervisor 是 ...

  5. Android SurfaceFlinger

    Android 系统启动过程Activity 创建过程Activity 与 Window 与 View 之间的关系 Android 系统从按下开机键到桌面,从桌面点击 App 图标到 Activity ...

  6. CSAPP阅读笔记-汇编语言初探(控制类指令)-来自第三章3.6的笔记-P135-P163

    1.正溢出与负溢出: 首先,一个正数与一个负数相加,不可能溢出,因为结果的绝对值一定小于两个加数的绝对值,既然两个加数能合理表示出来,结果一定也能合理表示出来. 其次,正溢出是由于两个很大的正数相加, ...

  7. 宜人贷项目里-----正则匹配input输入月份规则

    在标签上可以直接进行校验如下,如果只调数字键盘type=number不好用可以用type=tel <input name="creditDate" oninput=" ...

  8. springboot实现服务器端消息推送(H5原生支持)

    随着互联网的发展,传统的HTTP协议已经很难满足Web应用日益复杂的需求了.近年来,随着HTML5的诞生,WebSocket协议被提出,它实现了浏览器与服务器的全双工通信,扩展了浏览器与服务端的通信功 ...

  9. Ubuntu下的RabbitMQ安装与web管理配置

    首先在Ubutnu的/etc/apt/sources.list文件中加入一行 deb http://cz.archive.ubuntu.com/ubuntu trusty main 然后执行 sudo ...

  10. 对Map的一些总结

    1:Map接口. Collection体系中存储的是单个元素,单身汉,而Map中存储的是2个元素,存储的是成对的元素. Map和Collection是没有联系的!!不要以为Map是Collection ...