原文链接:http://javascript.crockford.com/private.html
Javascript是世界上最受误解的编程语言。有人认为Javascript缺乏隐藏对象细节信息的特性,因为Javascript的对象没有私有变量和私有方法。这其实是个误解,Javascript对象是可以有这些私有成员的。下面我们就来逐步介绍:
1. 对象
Javascript 本质上就是关于对象的语言。
数组、函数、对象这些在Javascript里都是对象。那么究竟什么是对象呢?通俗地来讲,对象就是键值对的集合。
键的类型是字符串,而值的类型可以是字符串、数值、布尔值以及对象(包括数组和函数)。这些键值对集合(对象)都以哈希表来存储以方便值的快速检索。
如果一个值是函数类型的话,我们可以把它看作是一个方法。当一个对象的方法被调用的时候,this指针就指向这个对象。这样这个方法就可以通过this指针来访问该对象的成员变量了。
对象可以由创建器来创建,创建器在Javascript里成为构造函数。构造函数的作用是用来初始化对象。构造函数充当了其它面向对象语言类的角色,包括创建静态变量和静态函数。
2. 公有成员(Public)
Javascript中对象的成员都是公有成员,任何函数都可以访问、修改或者删除这些成员,或者为对象添加成员。
有两种主要的方法把成员添加到新的对象中:
1. 使用构造函数:
构造函数机制通常用来初始化共有成员变量。我们用构造函数中的this指针来为对象添加成员:
function Container(param){
this.member = param;
}
这样,当我们创建一个新对象之后:
var myContainer = new Container('abc');
访问myContainer.memeber就可以获取到'abc'了。
2. 使用原型:
原型机制通常用来添加公有成员函数。当查找一个成员并且该成员在当前对象中没有被找到时,该成员就会被从当前对象的构造函数的原型中获取。原型机制用来实现继承。原型机制也被用来(conserve memeory)。给所有由构造函数创建的对象添加一个共同的函数,可以为这个构造函数的原型添加一个函数:
Container.prototype.stamp = function (string) {
return this.member + string;
}
这样我们就可以通过如下方式直接执行这个函数
myContainer.stamp('def')
会得到结果:‘abcdef’
3. 私有成员
私有成员由构造函数来创建。构造函数中,普通用var来声明的变量和参数就编程了私有成员:
function Container(param) {
this.member = param;
var secret = 3;
var that = this;
}
上面这个构造函数生成了3个私有实例变量:param,secret和that。他们和对象绑定,对象外部无法访问,对象自己的共有方法也同样无法访问。只有对象的私有方法可以访问。私有方法是构造函数内部的方法:
function Container(param) {
function dec() {
if (secret > 0) {
secret -= 1;
return true;
} else {
return false;
}
}
this.member = param;
var secret = 3;
var that = this;
}
私有方法dec的作用是检查secret这个实例变量。如果secret大于0,则减一,然后返回true。否则返回false。It can be used to make this object limited to three uses.
根据惯例,我们创建了一个that私有变量。that变量用来是私有成员变量可以访问到对象。This is a workaround for an error in the ECMAScript Language Specification which causes this to be set incorrectly for inner functions.
私有方法能被公有方法调用。我们接下来介绍专有方法,来使私有方法可用。
4. 专有方法
专有方法可以访问私有变量和方法,同时它本身也可以被共有方法调用。专有方法可以被删除或替换,但是无法被修改或者删除其专有特性。
我们通过this指针在构造函数里来为专有方法赋值:
function Container(param) {
function dec() {
if (secret > 0) {
secret -= 1;
return true;
} else {
return false;
}
}
this.member = param;
var secret = 3;
var that = this;
以上代码段中,service就是一个专有方法。调用myContainer.service()之后,在前三次被调用时将会返回‘abc’。这个之后就返回null了。service方法调用私有方法dec,dec来访问私有变量secret。其他的对象或方法也可以访问service方法,但是不允许直接访问私有成员。
5. 闭包
Javascript的这些公有、私有、专有成员模式之所以能够实现,是因为Javascript有闭包机制。闭包的含义是:一个外部方法,定义于其内部的方法可以访问该外部方法变量和参数,即使是外部方法的调用已经返回,内部方法依然可以访问。这绝对是Javascript的一个极其给力的特性。就目前来说还没有哪本关于Javascript编程的书讲述了如何来利用这一特性。大部分数压根就没提!
只有当一个对象被创建的时候,私有成员和专有成员才能被创建,而共有成员可以在任何时候创建并添加至对象。
6. 模式
Public
function Constructor(...) {
this.membername = value;
}
Constructor.prototype.membername = value;
Private
function Constructor(...) {
var that = this;
var membername = value;
function membername(...) {...}
}
Note: The function statement
function membername(...) {...}
is shorthand for
var membername = function membername(...) {...};
Privileged
function Constructor(...) {
this.membername = function (...) {...};
}
Copyright 2001 Douglas Crockford. All Rights Reserved Wrrrldwide.