分享
 
 
 

&#106avascript面向对象的支持(6)

王朝html/css/js·作者佚名  2006-11-24
窄屏简体版  字體: |||超大  

======================================================================

Qomolangma OpenProject v0.9

类别 :Rich Web Client

关键词 :JS OOP,JS Framwork, Rich Web Client,RIA,Web Component,

DOM,DTHML,CSS,JavaScript,JScript

项目发起:aimingoo (aim@263.net)

项目团队:aimingoo, leon(pfzhou@gmail.com)

有贡献者:JingYu(zjy@cnpack.org)

======================================================================

八、JavaScript面向对象的支持

~~~~~~~~~~~~~~~~~~

(续)

4. 实例和实例引用

--------

在.NET Framework对CTS(Common Type System)约定“一切都是对象”,并分为“值类型”和“引用类型”两种。其中“值类型”的对象在转换成“引用类型”数据的过程中,需要进行一个“装箱”和“拆箱”的过程。

在JavaScript也有同样的问题。我们看到的typeof关键字,返回以下六种数据类型:

"number"、"string"、"boolean"、"object"、"function" 和 "undefined"。

我们也发现JavaScript的对象系统中,有String、Number、Function、Boolean这四种对象构造器。那么,我们的问题是:如果有一个数字A,typeof(A)的结果,到底会是'number'呢,还是一个构造器指向function Number()的对象呢?

//---------------------------------------------------------

// 关于JavaScript的类型的测试代码

//---------------------------------------------------------

function getTypeInfo(V) {

return (typeof V == 'object' ? 'Object, construct by '+V.constructor

: 'Value, type of '+typeof V);

}

var A1 = 100;

var A2 = new Number(100);

document.writeln('A1 is ', getTypeInfo(A1), '<BR>');

document.writeln('A2 is ', getTypeInfo(A2), '<BR>');

document.writeln([A1.constructor === A2.constructor, A2.constructor === Number]);

测试代码的执行结果如下:

-----------

A1 is Value, type of number

A2 is Object, construct by function Number() { [native code] }

true,true

-----------

我们注意到,A1和A2的构造器都指向Number。这意味着通过constructor属性来识别对象,(有时)比typeof更加有效。因为“值类型数据”A1作为一个对象来看待时,与A2有完全相同的特性。

——除了与实例引用有关的问题。

参考JScript手册,我们对其它基础类型和构造器做相同考察,可以发现:

- 基础类型中的undefined、number、boolean和string,是“值类型”变量

- 基础类型中的array、function和object,是“引用类型”变量

- 使用new()方法构造出对象,是“引用类型”变量

下面的代码说明“值类型”与“引用类型”之间的区别:

//---------------------------------------------------------

// 关于JavaScript类型系统中的值/引用问题

//---------------------------------------------------------

var str1 = 'abcdefgh', str2 = 'abcdefgh';

var obj1 = new String('abcdefgh'), obj2 = new String('abcdefgh');

document.writeln([str1==str2, str1===str2], '<br>');

document.writeln([obj1==obj2, obj1===obj2]);

测试代码的执行结果如下:

-----------

true, true

false, false

-----------

我们看到,无论是等值运算(==),还是全等运算(===),对“对象”和“值”的理解都是不一样的。

更进一步的理解这种现象,我们知道:

- 运算结果为值类型,或变量为值类型时,等值(或全等)比较可以得到预想结果

- (即使包含相同的数据,)不同的对象实例之间是不等值(或全等)的

- 同一个对象的不同引用之间,是等值(==)且全等(===)的

但对于String类型,有一点补充:根据JScript的描述,两个字符串比较时,只要有一个是值类型,则按值比较。这意味着在上面的例子中,代码“str1==obj1”会得到结果true。而全等(===)运算需要检测变量类型的一致性,因此“str1===obj1”的结果返回false。

JavaScript中的函数参数总是传入值参,引用类型(的实例)是作为指针值传入的。因此函数可以随意重写入口变量,而不用担心外部变量被修改。但是,需要留意传入的引用类型的变量,因为对它方法调用和属性读写可能会影响到实例本身。——但,也可以通过引用类型的参数来传出数据。

最后补充说明一下,值类型比较会逐字节检测对象实例中的数据,效率低但准确性高;而引用类型只检测实例指针和数据类型,因此效率高而准确性低。如果你需要检测两个引用类型是否真的包含相同的数据,可能你需要尝试把它转换成“字符串值”再来比较。

6. 函数的上下文环境

--------

只要写过代码,你应该知道变量是有“全局变量”和“局部变量”之分的。绝大多数的

JavaScript程序员也知道下面这些概念:

//---------------------------------------------------------

// JavaScript中的全局变量与局部变量

//---------------------------------------------------------

var v1 = '全局变量-1';

v2 = '全局变量-2';

function foo() {

v3 = '全局变量-3';

var v4 = '只有在函数内部并使用var定义的,才是局部变量';

}

按照通常对语言的理解来说,不同的代码调用函数,都会拥有一套独立的局部变量。

因此下面这段代码很容易理解:

//---------------------------------------------------------

// JavaScript的局部变量

//---------------------------------------------------------

function MyObject() {

var o = new Object;

this.getValue = function() {

return o;

}

}

var obj1 = new MyObject();

var obj2 = new MyObject();

document.writeln(obj1.getValue() == obj2.getValue());

结果显示false,表明不同(实例的方法)调用返回的局部变量“obj1/obj2”是不相同。

变量的局部、全局特性与OOP的封装性中的“私有(private)”、“公开(public)”具有类同性。因此绝大多数资料总是以下面的方式来说明JavaScript的面向对象系统中的“封装权限级别”问题:

//---------------------------------------------------------

// JavaScript中OOP封装性

//---------------------------------------------------------

function MyObject() {

// 1. 私有成员和方法

var private_prop = 0;

var private_method_1 = function() {

// ...

return 1

}

function private_method_2() {

// ...

return 1

}

// 2. 特权方法

this.privileged_method = function () {

private_prop++;

return private_prop + private_method_1() + private_method_2();

}

// 3. 公开成员和方法

this.public_prop_1 = '';

this.public_method_1 = function () {

// ...

}

}

// 4. 公开成员和方法(2)

MyObject.prototype.public_prop_1 = '';

MyObject.prototype.public_method_1 = function () {

// ...

}

var obj1 = new MyObject();

var obj2 = new MyObject();

document.writeln(obj1.privileged_method(), '<br>');

document.writeln(obj2.privileged_method());

在这里,“私有(private)”表明只有在(构造)函数内部可访问,而“特权(privileged)”是特指一种存取“私有域”的“公开(public)”方法。“公开(public)”表明在(构造)函数外可以调用和存取。

除了上述的封装权限之外,一些文档还介绍了其它两种相关的概念:

- 原型属性:Classname.prototype.propertyName = someValue

- (类)静态属性:Classname.propertyName = someValue

然而,从面向对象的角度上来讲,上面这些概念都很难自圆其说:JavaScript究竟是为何、以及如何划分出这些封装权限和概念来的呢?

——因为我们必须注意到下面这个例子所带来的问题:

//---------------------------------------------------------

// JavaScript中的局部变量

//---------------------------------------------------------

function MyFoo() {

var i;

MyFoo.setValue = function (v) {

i = v;

}

MyFoo.getValue = function () {

return i;

}

}

MyFoo();

var obj1 = new Object();

var obj2 = new Object();

// 测试一

MyFoo.setValue.call(obj1, 'obj1');

document.writeln(MyFoo.getValue.call(obj1), '<BR>');

// 测试二

MyFoo.setValue.call(obj2, 'obj2');

document.writeln(MyFoo.getValue.call(obj2));

document.writeln(MyFoo.getValue.call(obj1));

document.writeln(MyFoo.getValue());

在这个测试代码中,obj1/obj2都是Object()实例。我们使用function.call()的方式来调用setValue/getValue,使得在MyFoo()调用的过程中替换this为obj1/obj2实例。

然而我们发现“测试二”完成之后,obj2、obj1以及function MyFoo()所持有的局部变量都返回了“obj2”。——这表明三个函数使用了同一个局部变量。

由此可见,JavaScript在处理局部变量时,对“普通函数”与“构造器”是分别对待的。这种处理策略在一些JavaScript相关的资料中被解释作“面向对象中的私有域”问题。而事实上,我更愿意从源代码一级来告诉你真相:这是对象的上下文环境的问题。——只不过从表面看去,“上下文环境”的问题被转嫁到对象的封装性问题上了。

(在阅读下面的文字之前,)先做一个概念性的说明:

- 在普通函数中,上下文环境被window对象所持有

- 在“构造

[1] [2] 下一页

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有