- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我开始了解 Javascript 中的原型(prototype)设计和闭包,但还不完全是这样。下面的示例中,我的两个对象,第二个对象似乎失去了范围/上下文并接管了第一个对象的身份。
function broker()
{
var _q = [];
this.add = function(delegate) {_q[_q.length] = delegate; }
this.broadcast = function(message)
{
for(qi = 0; qi < _q.length; qi++)
{
_q[qi](message);
}
}
}
function subscriber(abroker, yourname)
{
me = this;
this.myprop = yourname;
this.subby = function(message){ alert(message + " " + me.myprop + me.dosomething() + secret()); };
this.dosomething = function() {return "...abc";};
function secret(){return "...def";}
abroker.add(this.subby);
}
var thebroker = new broker();
var mysub = new subscriber(thebroker, 'mysub');
var myothersub = new subscriber(thebroker, 'myothersub');
thebroker.broadcast("hello from");
这个想法是有一个公共(public)代理对象可以调用订阅者对象上的委托(delegate)并执行其中的函数。但我失去了代理调用的调用函数的范围。
输出: 2 个警报窗口,均输出:“myothersub”,mysub 似乎失去了范围?
通过在原始对象之外显式声明 subby 委托(delegate)并引用整个对象,我已成功实现了正确的响应,例如:
不要在订阅者对象中声明 this.subby:
mysub.subby = function(message)
{
alert(message + " " + mysub.myprop); // obv the dosomething his hidden
}
thebroker.add(mysub.subby);
如果上面的语法有任何错误,请原谅,直接从内存中输入。它在实践中确实有效,但失去了我习惯的封装。
如何使用原始技术封装对象的范围/上下文?
最佳答案
简短回答:看来问题只是与您的 me
声明有关。在 subscriber
构造函数。至少你需要输入 var
在它前面,使其成为每个对象的本地/私有(private)变量。所以var me = this;
而不是me = this;
.
说明:在 JavaScript 中,当您未使用 var
显式声明变量时,它使变量成为全局变量。所以你的原始脚本中发生的事情是你创建了 mysub
其中声明 me
作为对 this
的全局引用里面mysub
目的。但是一旦你创建了myothersub
全局me
被覆盖为this
新里面myothersub
对象。
因为你的subby
方法创建了一个基于 me
的警报从哪个对象调用它并不重要,因为两个对象中的方法没有使用任何本地或特定于该对象的内容,而只是引用相同的全局变量 - this
在要创建的最后一个此类对象内
只需写 var me = this;
而不是me = this;
您创建 me
的本地版本每次您创建的每个新对象都在一个闭包内,而不是被覆盖的对象。
...
PS。额外提示。您应该对所有变量执行此操作,以确保拥有尽可能少的全局变量,尤其是当您不希望它们是全局变量时!因此我会对变量 qi
做出相同的声明里面broker
构造函数。您可以简单地通过在循环条件内声明来完成此操作,例如 for (var qi = 0; qi < _q.length; qi++)
。这足以阻止qi
是一个全局变量。
但是,为了使代码易于阅读,最好在函数顶部声明所有变量。所以我建议简单地重写 broadcast
方法如下:
this.broadcast = function(message) {
var qi, // STOPS `qi` BECOMING A GLOBAL
ql = q.length; // SO DON'T HAVE TO CHECK LENGTH OF `q` EVERY LOOP
for(qi = 0; qi < ql; qi += 1) {
_q[qi](message);
}
};
如果您以前没有遇到过 Douglas Crockford,那么在涉及闭包、对象创建和良好的代码编写约定方面,Douglas Crockford 是一位非常优秀的 JavaScript 作家。 this page 有很多提示您可以很容易地找到他就该主题进行演讲的视频。当我开始更仔细地研究闭包和 JavaScript 的其他方面时,所有这些确实对我有帮助;希望它也能帮助你。
关于javascript - 如何封装对象属性以便以前的对象不被接管? (原型(prototype)/闭合。),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14547737/
你能比较一下属性吗 我想禁用文本框“txtName”。有两种方式 使用javascript,txtName.disabled = true 使用 ASP.NET, 哪种方法更好,为什么? 最佳答案 我
Count 属性 返回一个集合或 Dictionary 对象包含的项目数。只读。 object.Count object 可以是“应用于”列表中列出的任何集合或对
CompareMode 属性 设置并返回在 Dictionary 对象中比较字符串关键字的比较模式。 object.CompareMode[ = compare] 参数
Column 属性 只读属性,返回 TextStream 文件中当前字符位置的列号。 object.Column object 通常是 TextStream 对象的名称。
AvailableSpace 属性 返回指定的驱动器或网络共享对于用户的可用空间大小。 object.AvailableSpace object 应为 Drive 
Attributes 属性 设置或返回文件或文件夹的属性。可读写或只读(与属性有关)。 object.Attributes [= newattributes] 参数 object
AtEndOfStream 属性 如果文件指针位于 TextStream 文件末,则返回 True;否则如果不为只读则返回 False。 object.A
AtEndOfLine 属性 TextStream 文件中,如果文件指针指向行末标记,就返回 True;否则如果不是只读则返回 False。 object.AtEn
RootFolder 属性 返回一个 Folder 对象,表示指定驱动器的根文件夹。只读。 object.RootFolder object 应为 Dr
Path 属性 返回指定文件、文件夹或驱动器的路径。 object.Path object 应为 File、Folder 或 Drive 对象的名称。 说明 对于驱动器,路径不包含根目录。
ParentFolder 属性 返回指定文件或文件夹的父文件夹。只读。 object.ParentFolder object 应为 File 或 Folder 对象的名称。 说明 以下代码
Name 属性 设置或返回指定的文件或文件夹的名称。可读写。 object.Name [= newname] 参数 object 必选项。应为 File 或&
Line 属性 只读属性,返回 TextStream 文件中的当前行号。 object.Line object 通常是 TextStream 对象的名称。 说明 文件刚
Key 属性 在 Dictionary 对象中设置 key。 object.Key(key) = newkey 参数 object 必选项。通常是 Dictionary 
Item 属性 设置或返回 Dictionary 对象中指定的 key 对应的 item,或返回集合中基于指定的 key 的&
IsRootFolder 属性 如果指定的文件夹是根文件夹,返回 True;否则返回 False。 object.IsRootFolder object 应为&n
IsReady 属性 如果指定的驱动器就绪,返回 True;否则返回 False。 object.IsReady object 应为 Drive&nbs
FreeSpace 属性 返回指定的驱动器或网络共享对于用户的可用空间大小。只读。 object.FreeSpace object 应为 Drive 对象的名称。
FileSystem 属性 返回指定的驱动器使用的文件系统的类型。 object.FileSystem object 应为 Drive 对象的名称。 说明 可
Files 属性 返回由指定文件夹中所有 File 对象(包括隐藏文件和系统文件)组成的 Files 集合。 object.Files object&n
我是一名优秀的程序员,十分优秀!