JavaScript中的私有成员[翻译]
原作者:Douglas Crockford,原文地址:http://www.crockford.com/javascript/private.html
JavaScript 是世界上被误解最深的编程语言. 有人认为它缺少信息隐藏的能力,因为它的对象不能拥有私有的实例变量和方法。事实上这是一种误解。JavaScript 对象可以拥有私有成员。下边就讲讲怎么做。
对象(Objects)
JavaScript 从根本上来说就是关于对象的语言。数组(Arrays)是对象。函数(Functions)是对象。 对象(Objects)是对象。那么对象是什么呢?对象就是键值对的集合(name-value pairs)。键(name)是字符串(strings) ,值(value)可以是字符串(strings)、数值(numbers)、布尔值(booleans)和对象(objects(包括数组(arrays)和函数(functions)))。对象通常是以hashtable的形式实现的,以便于值的快速检索。
如果对象的某个值是一个函数,我们可以认为它是这个对象的方法。当一个对象的方法被调用时,方法中的this变量代表调用它的对象。这样方法就可以通过this变量访问实例中的变量了。
对象可以由构造器(constructors)产生,构造器就是专门用来初始化对象的函数。在其他语言中由“类”(class)提供的功能,在JavaScript中由构造器提供,包括静态变量和方法。
公有的(Public)
在JavaScript中对象的成员都是公有(public)成员。任何函数都能访问、修改、删除这些成员,或者添加新成员。给一个新对象设置成员主要有两种方式:
在构造器中(In the constructor)
这种技术通常用来初始化公有的实例变量。构造器的this变量被用来给对象添加成员。
function Container(param) {
this.member = param;
}
这样,当我们构造一个新对象时
var myContainer = new Container('abc');
那么 myContainer.member 就是 'abc'.
在原型中(In the prototype)
这种技术通常用来添加公有的方法。当一个成员在对象本身中找不到时,那么它是来自对象的构造器的原型(prototype)成员。原型机制用来实现继承。也能节约内存。想要给一个构造器产生出来的所有对象都添加一个方法,只要给这个构造器的原型(prototype)添加一个函数即可:
Container.prototype.stamp = function (string) {
return this.member + string;
}
那么,我们可以调用这个方法
myContainer.stamp('def')
产生了 'abcdef'。
私有的(Private)
私有的(private)成员由构造器产生。构造器的普通vars和parameters变成了私有成员。
function Container(param) {
this.member = param;
var secret = 3;
var that = this;
}
这个构造器创造了三个私有的实例成员: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,将secret递减并返回true。否则返回false。这个方法可以用来将这个对象限制为仅能使用三次。
按照惯例,我们创建了一个私有变量that。这是为了让对象对私有方法可见。这样的j解决方案是因为ECMAScript语言规范中有一个错误,会导致this不能正确地设置给内部函数。
私有方法不能被公有方法调用。想要让私有方法发生作用,我们需要引入一个特权方法。
特权的(Privileged)
特权(privileged)方法能够访问私有变量和方法,并且它自身可以被公有方法和外部访问。可以删除或者替换一个特权方法,但是不能改变它。
特权方法是用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;
this.service = function () {
return dec() ? that.member : null;
};
}
service是一个特权方法。前三次调用myContainer.service()时将返回'abc'。之后,将会返回null。service调用私有方法dec,而dec访问私有的secret变量。service对其他对象和方法可见,但是它不允许对私有成员的直接访问。
闭包(Closures)
这种公有、私有和特权成员的模式之所以可行,是因为JavaScript有闭包(closures)。这意味着内部函数总是能访问到外部函数的vars和parameters,甚至当外部函数已经返回后依然能访问。这是这个语言极其强大的一个特性。目前还没有JavaScript编程方面的书籍讲述如何利用这一特性。大多数甚至都没有提到它。
私有的和特权的成员只能在对象构造时创建。公有的成员在任何时候都能添加。
模式(Patterns)
Public
function Constructor(...) {
this.membername = value;
}
Constructor.prototype.membername = value;
Private
function Constructor(...) {
var that = this;
var membername = value;function membername(...) {...}
}
注意: 函数语句
function membername(...) {...}
是对如下语句的简写
var membername = function membername(...) {...};
Privileged
function Constructor(...) {
this.membername = function (...) {...};
}
JavaScript中的私有成员[翻译]的更多相关文章
- javascript模式(1)--私有成员
javascript是基于对象的一门语言,没有想java等语言那样子拥有封装的特性.但是javascript可以通过闭包来进行模拟. 1.构造函数与私有成员 可以用构造函数形成一个闭包,实现内部成员的 ...
- C#中访问私有成员
首先访问一个类的私有成员不是什么好做法.大家都知道私有成员在外部是不能被访问的.一个类中会存在很多私有成员:如私有字段.私有属性.私有方法.对于私有成员造访,可以套用下面这种非常好的方式去解决. pr ...
- C#中访问私有成员技巧
源代码是别人的,你就不能修改源代码,只提供给你dll.或者你去维护别人的代码,源代码却有丢失.这样的情况如果你想知道私有成员的值,甚至去想直接调用类里面的私有方法.那怎么办呢?其实在.net中访问私有 ...
- C#中访问私有成员--反射
首先我必须承认访问一个类的私有成员不是什么好做法.大家也都知道私有成员在外部是不能被访问的.而一个类中会存在很多私有成员:如私有字段.私有属性.私有方法.对于私有成员访问,可以套用下面这种非常好的方式 ...
- 『无为则无心』Python面向对象 — 55、多层继承和继承中的私有成员
目录 1.Python支持多层继承 (1)多层继承实现 (2)多层继承和多重继承区别 2.继承中的私有成员 (1)继承中父类私有属性和私有方法 (2)获取和修改私有属性值 1.Python支持多层继承 ...
- JavaScript中函数式编程中文翻译
JavaScript 中的函数式编程 原著由 Dan Mantyla 编写 近几年来,随着 Haskell.Scala.Clojure 等学院派原生支持函数式编程的偏门语言越来越受到关注,同时主流的 ...
- C中的私有成员
skynet_context声明在.h里 但定义在.c里面 外部使用的时候无法用ctx->handle获取私有成员,会提示解引用类型错误 必须用.h里函数获取ctx里属性.
- JavaScript中的私有属性
一.使用构造函数获得私有属性: function Gadget(){ var name = 'iPod'; this.getName = function(){ return name; }; }; ...
- javascript中的私有作用域
我们知道js中所有的块级作用域都是无效的,块级作用域内的变量,在外部仍然可以被读取,其实是申明在外部的.如何实现变量的私有化,只在块级作用域起效,避免污染全局的变量呢.而且,挂载在全局的变量很难被回收 ...
随机推荐
- Python面向对象篇(3)-封装、多态、反射及描述符
1. 多态 多态,最浅显的意识就是同一事物具有多种形态,这很好理解,动物是一个大类,猫类属于动物类,狗类属于动物类,人也是属于动物类,那能理解成,猫.狗.人是一样的吗?当然不是,还有,水,分为液体.固 ...
- python3基础入门-知识点简记
1.基础语法 编码.标识符.保留字.注释.行与缩进... 2.变量类型 (1)Python3有6个标准的数据类型: Numbers(数字) 数字数据类型用于存储数值 不可改变的数据类型 可细分为 ...
- WebView性能优化--独立进程
Android允许一个app同时存在多个进程,可以根据需要把不同的模块放到不同进程中处理. 一.WebView独立进程的好处 1.有效增大App的运存,减少由webview引起的内存泄露对主进程内存的 ...
- Spark DataFrame写入HBase的常用方式
Spark是目前最流行的分布式计算框架,而HBase则是在HDFS之上的列式分布式存储引擎,基于Spark做离线或者实时计算,数据结果保存在HBase中是目前很流行的做法.例如用户画像.单品画像.推荐 ...
- Nuget发布自己的DLL
首先说明背景,在asp.net core开发中,使用了Oracle,Oracle官方发布了一个新的sdk用于连接数据库,但是asp.net core有个特性,就是不支持直接引用dll,也就 ...
- iOS视频直播
视频直播技术点 视频直播,可以分为 采集,前处理,编码,传输, 服务器处理,解码,渲染 采集: iOS系统因为软硬件种类不多, 硬件适配性比较好, 所以比较简单. 而Android端市面上机型众多, ...
- HDU - 3038 种类并查集
思路:种类并查集的每个节点应该保存它的父节点以及他和父节点之间的关系.假设root表示根结点,sum[i-1]表示i到根结点的和,那么sum[j-1] - sum[i]可以得到区间[j, i]的和.那 ...
- UVA-714 二分
把可能的进行二分判断,判断的时候尽量向右取,一直取到不能去为止,这样才有可能成功分割. 判断是否可以把up作为最大值的代码: bool judge(LL up){ if(up < Big) re ...
- SMJobBless官方Demo笔记
SMJobBless是苹果官方提供的用于"MacOS app获取root权限"的demo. 具体思路 使用Security.framework和ServiceManagement. ...
- linux Cacti监控服务器搭建
搭建Cacti监控服务器 部署安装环境(lamp) [root@zhuji1 ~]# yum -y install httpd [root@zhuji1 ~]# yum -y install php ...