Self
Self语言,是一种基于原型的面向对象的程序设计语言,也是一个集成开发环境和运行环境,由David Ungar和Randy Smith,最初在1986年于施乐帕罗奥多研究中心设计。Self语言在Smalltalk的基础上,采用“槽”取代了“变量”,从而彻底体现了一切都是对象的风格。在实现Self系统的过程中,设计研究人员发展出了一种动态自适应编译技术。 简介Self语言把在概念上精简Smalltalk作为设计原则。它在把消息作为最基本的操作的同时,取消了类的概念,只有对象的概念。它把对象的特性,理解为获取或更改特性的这两种方法,从而把特性的概念简化为方法,并且通过消息来读槽和写槽的方式,取代了变量及其赋值。Self提出了特质的概念,用动态绑定实现了委托。 尽管Self系统一次运行在一个进程中,但实际上可以分成两个部分:Self虚拟机和Self世界。Self世界是一个Self对象库,Self对象包括数据对象和方法对象,方法对象的代码部份,是用一种指令非常简单的字节码表示的,字节码由Self虚拟机来解释。当Self程序从终端、文件或者图形用户界面输入到系统之中时,Self系统把源程序解析转化为Self世界里的对象。 动态自适应编译技术的采用,提高了Self代码的执行效率。对经常执行的方法,虚拟机将进一步把字节码转化为本机代码。Self虚拟机还提供了一些可供调用的原语,用来实现算术运算、对象复制、输入输出等。 Self还拥有一个图形用户界面Morphic,Self的编程环境,也是基于Morphic来实现的。Self在精简语言概念的同时,也把大量的工作转交给环境来处理,语言中的反射机制也同环境密切相关。 历史在1986年,David Ungar和Randy Smith在施乐帕罗奥多研究中心,提出了Self语言的最初设计,并在1987年的OOPSLA'87的论文《Self:简单性的能力》中给出了描述[3],此文在2006年被评为1986年到1996年间三个最有影响的OOPSLA论文之一[4]。 1987年初,Craig Chambers、Elgin Lee和Martin Rinard,在Smalltalk上给出了Self的第一个实验性解释器。1987年夏,Self项目在斯坦福大学正式开始,1988年夏给出了第一个有效率的实现,并发布了1.0和1.1两个版本。1991年初,Self项目移至Sun微系统,并且在1992年发布了2.0版。1993年1月,Self 3.0版发布。 1995年7月,Self 4.0版发布。在这个版本中包括了一个全新的图形用户环境Morphic。在2016年发行了4.3版本并可运行在Mac OS X和Solaris上。在2010年发行了版本4.4[5],由最初团队的某些人和独立编程者形成的小组开发,它和所有后续版本可以运行在Mac OS X和Linux上。2014年1月发行了4.5版本[6]。2017年5月发行了版本2017.1。 Self的发展基本已经停滞,但在发展Self过程中探索出的一些技术,在其他的系统中得到了应用。在Self的实现中采用的各种编译优化技术,直接导致了Java Hotspot虚拟机的产生;在Smalltalk的一个实现Squeak中,采用了Self图形用户界面Morphic的设计方案,放弃了Smalltalk-80中采用的MVC的方案。Self是对JavaScript编程语言设计有最主要影响者之一[7]。 基于原型编程传统的基于类的面向对象语言,基于了根深蒂固的二元性: 例如,假设车辆类 这个例子展示了这种方式的一个问题:Bob的汽车,恰巧是一个跑车,在任何意义上都不能装载和运送建材,但这是建模 这个问题是在原型(prototype)这个概念背后的动机因素之一。除非你能必然性的预测出一组对象和类,在遥远未来时所要有的品质,你不能恰当的设计好一个类的层级。程序最终需要增加行为,实在是太频繁了,而系统的很多节段将需要重新设计或重新构建,来以不同的方式迸发出对象。早期的面向对象语言如Smalltalk的实验,显示出这种问题反反复复的出现。系统趋向于增长到一定程度后,就变得非常僵化,因为在编程者的代码下的深层的基本类,简直就像是逐渐变成了一个“错误”;没有变更原来的类的容易方式,就会出现严重的问题。 动态语言如Smalltalk,允许通过周知的按照类的方法进行这种变更;即通过改变类,基于它的对象就可以改变它们的行为。但是,进行这种变更必须非常小心,因为基于相同类的其他对象,可能把它当作“错误行为”:“错误”经常是依赖于场景的,这是脆弱基类问题的一种形式。进一步的说,在静态语言如C++中,这里的子类可以从超类分别的编译,对超类的变更实际上可以破坏预编译的子类方法;这是脆弱基类问题的另一种形式,也是脆弱二进制接口问题的一种形式。 在Self和其他基于原型的编程语言中,消除了在类和对象之间的这种二元性。不再有基于某种“类”的一个对象“实例”,在Self中,你可以复制一个现存的对象,并改变它。故而 主要用来制作复本的基本对象叫做“原型”。这种技术被称为是一种非常简化的机制。如果一个现存的对象或对象的集合,被证明是个不适当的模型,编程者可以简单的建立有正确行为的一个修改的对象,并转而使用它。使用现存对象的代码不会改变。 语法和语义下面简要描述Self语言的语法和语义。 对象文字(literal)包括:数、用 (| 槽1. 槽2 | 一些代码 )
槽(slot)是名字-值对,槽包含到其他对象的引用。槽列表由点号分隔的(可以为空的)一序列的槽描述符组成。在槽列表结束处的点号是可选的。槽描述符(descriptor)有两种:
在Self中没有单独的赋值运算。其他面向对象语言中的访问子方法对应数据槽,变异子方法对应赋值槽。假如 任何槽都可以通过增加星号后缀,来制成父槽。星号不是槽名字的一部份,在将名字与消息进行匹配时候忽略它。例如一个初始化了的可变的点可以定义为: (| parent* = traits point.
x <- 3 + 4.
y <- 5.
|)
一个对象的代码是以点号分隔的一序列的表达式。尾随的点号是可选的。每个表达式有一系列的消息发送和文字组成。在一个对象的代码中最后的表达式,可以前导着指示返回的 一个真正的空对象,指示为 消息通过消息访问槽的语法,类似于Smalltalk,有三类消息可以获得:
所有消息都返回结果,所以显式指定的接收者和参数自身可以是其他消息的结果。下面是Self版本的hello world程序例子: 'Hello, World!' print.
组合可以通过使用圆括号来进行强制。在缺乏明确组合的情况下,一元消息具有最高优先级,其次是二元消息,而关键字消息最低。一元消息从左至右复合。二元消息对于同一个算符从左至右结合,例如 关键字消息的第一部份必须开始于小写字母,而后续部份都必须开始于大写字母。例如表达式: 5 min: 4 Max: 7
是一个单一的消息 5 min: 4 max: 7
涉及两个消息:第一个消息 5 min: 6 min: 7 Max: 8 Max: 9 min: 10 Max: 11
被解释为: 5 min: (6 min: 7 Max: 8 Max: (9 min: 10 Max: 11))
由于很多消息被发送给当前消息接收者 方法方法(method)是除了参数槽及或局部槽之外,还包含代码的对象。参数(argument)槽名字开始于一个冒号,它不是槽名字的一部分,在将名字与消息进行匹配时候忽略它。参数槽总是只读的,并且不能对它们指定初始化者。下面例子是计算平方的方法对象: (| :arg | arg * arg )
一个普通方法(简称方法),是不嵌入到其他代码之中的方法,它只能存放在只读槽中。普通方法总是有一个叫做 如果一个槽包含一个方法,在求值这个槽来响应发来的消息的时候,这个方法对象被浅层复制(clone),从而新建它的一个活动(activation)对象,它包含这个方法的参数槽和局部槽;复制体的 (| + arg =
( (clone x: x + arg x) y: y + arg y )
|)
可以被无歧义的分析,其含义同于: (| + =
(| :arg | (clone x: ((x + (arg x)))) y: ((y + (arg y))) ).
|)
这里出现了三个隐含接收者一元消息 作为语法约定,参数名字可以直接写在槽名字中对应关键字之后,它不再带有前缀冒号,从而隐含的声明参数槽。例如下面的方法定义: (| ifTrue: False: =
(| :b1. :b2 | b1 value ).
|)
可以等价的定义为: (| ifTrue: b1 False: b2 =
( b1 value ).
|)
返回算符 块块是Self的闭包,Self就像Smalltalk,使用“块”用于控制流程和其他职责。块文字的写法,除了方括号替代了圆括号之外,类似于其他对象文字。例如嵌入在下列表达式中的块: 1 to: 5 * i By: 2 * j Do: [| :k | k print ]
一个块文字定义两个对象:一个块数据对象,和它包围的一个块方法对象。
相应的,块求值分为如下两个阶段:
在块中,返回算符 委托在理论上,所有Self对象都是独立实体,Self既没有类也没有元类。对任何特定对象的变更,都不影响任何其他对象,但是在某些情况下,却需要它们有关联。正常的一个对象,只能理解对应于它的局部槽的消息,但拥有一个或更多的指示父(parent)对象的槽,对象可以将任何自身不理解的消息,委托(delegate)给父对象。 Self采用这种方式,处理在基于类的语言中使用继承来担负的责任。委托还可以用来实现一些特征,比如命名空间和词法作用域。通过下面的例子展示委托与传统的类的不同之处: myObject parent: someOtherObject.
这个句子通过改变与叫做 特质例如,假定在一个简单的账簿应用中,定义了一个对象叫做“银行帐号”(bank account)。通常建立的这个对象,具有内部的方法,比如说“存款”(deposit)和“取款”(withdraw),和任何它所需要的数据槽,比如说“余额”(balance)。这只是一个原型,它只在使用方式上特殊,因为它恰好是一个全功能的银行帐号。 为“Bob的账户”制作银行帐号对象的复制品(clone),将建立一个新对象,它在起初时完全同于原型。在这种情况下,将复制(copy)包括方法和任何数据的槽。但更常用的解决方案,是首先建立叫做它的特质(trait)对象的一个简单对象,它包含通常与一个类有关的项目。 在这个例子中,“银行账户”将没有存款和取款方法,而是委托给一个父对象来做这些。采用这种方式,可以制作银行帐号对象的很多复本,但是我们仍可以通过改变它所委托的特质对象中的槽,来改变它们全体的行为。 Self世界当处在于提示符下键入表达式的场景时,由叫做“大厅”(lobby)的一个对象,引领用户进入Self世界。当建立一个新对象的脚本被读入系统的时候,脚本中的表达式都在大厅的上下文中求值。就是说大厅是这个脚本中所有发送给 要引用在脚本中的某个现存的对象,必须通过发送一个消息到大厅才可以访问到它。大厅的 例如,原型列表的完全路径名字是 不是所有对象都有路径名字,只有那些从大厅可以到达的对象才有,这些对象称为“周知的”。大厅向用户提供三类对象:
在大厅的 新建对象有两个消息与对象复制有关:
考虑一个图形用户界面有关的例子: (desktop activeWindow) draw: (labelWidget copy label: 'Hello, World!').
首先进行的是 增加槽在Self中的对象,可以通过包括新加的槽来修改。这可以通过被推荐使用的图形编程环境来做,或者直接使用原语 _AddSlots: (| newObject = (| entries <- list copy …… |) |)
因为 例子在下面的例子中,将基于类语言中叫做 _AddSlots: (| vehicle <- (|parent* = traits clonable|) |).
在大厅中创建了一个叫做 然后向这个新建对象继续增加 vehicle _AddSlots: (| name <- 'automobile'|).
在大厅中从 _AddSlots: (| sportsCar <- vehicle copy |).
sportsCar _AddSlots: (| driveToWork = ("这个方法的代码") |).
在大厅中从 _AddSlots: (| porsche911 <- sportsCar copy |).
porsche911 name: 'Bobs Porsche'.
对象 环境Self的一个特征,是它基于了早期Smalltalk系统所用的某种虚拟机系统。就是说,程序不是像C语言中那样的独立实体,而是需要它们的整体内存环境来运行。这要求应用程序被装载入保存内存的大块(chunk)之中,这叫做“快照”或映像。这种方式的缺点,是映像有时很大并且笨重;但是调试一个映像,经常被调试一个传统程序要简单,因为运行时状态更容易检查和修改。在基于源代码和基于映像的开发之间的不同,是类似于在面向类的和面向原型的面向对象编程之间的区别。 此外,环境是为了让在系统之中的对象能快速和可持续的变更而定制的。重新构建一个“类”设计,就像从现存的祖先拖动出来方法放入新造的之中一样容易。简单任务像测试方法,可以通过制作复本来处理,拖动方法进入这个复本,接着变更它。不同于传统系统,只有变更了的对象有新代码,不需要重建任何东西来测试它。如果这个方法有效,可以简单的把它拖动回祖先之中。 性能Self的VM实现的性能,在某些测试之中大约是优化的C程序速度的一半[8]。这是通过即时编译技术达到的,它是在Self研究中首创并改进的,能够使高级语言表现得这么好。 垃圾收集Self的垃圾收集器使用分代垃圾回收,它按年龄分离对象。通过使用内存管理系统记录页面写,可以维护一个写屏障。这个技术给出了卓越的性能,尽管在运行一些时间之后,出现完全的垃圾收集,要花相当可观的时间。 优化运行系统选择性的扁平化调用结构。这给出适当的自身提速,但允许了对不同调用者类型的类型信息和多版本的代码的大量缓存。这去除了对做很多方法查找的需要,并允许条件分支语句和硬编码调用被插入,这经常能给出类似C语言的性能,而又不失去语言层面的通用性,但要建立在完全的垃圾收集系统之上[9]。 引用
延伸阅读
站外链接
|