admin管理员组文章数量:1340299
ES5 has a enumerable flag. Example
Example
var getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor
, pd = getOwnPropertyDescriptor(Object.prototype, "toString");
assert(pd.enumerable === false, "enumerability has the wrong value");
Partial implementation
Partial implementation is do-able by having Object.keys
and Object.getOwnPropertyNames
filter out new non-enumerable properties using the shimmed Object.defineProperty
.
Introduction
This allows for properties to be non enumerable. This clearly means that Example
for (var key in {}) {
assert(key !== "toString", "I should never print");
}
This allows us to add properties to say Object.prototype
(Example)
Object.defineProperty(Object.prototype, "toUpperCaseString", {
value: function toUpperCaseString() {
return this.toString().toUpperCase();
},
enumerable: false
});
for (var key in {}) {
assert(key !== "toUpperCaseString", "I should never print");
}
console.log(({}).toUpperCaseString()); // "[OBJECT OBJECT]"
Question
How can we emulate this in non-ES5 pliant browsers?
Browser pat table
In this case we care about potentially solving this for
- IE < 9 (IE8 only works on DOM objects)
- Firefox 3.6
- Safari 4
- Opera 11.5 (Opera 11.6 solves this).
The ES5-shim does not have a solution for this.
The ES5 shim has a solution for most ES5 features that will break your code if it doesn't work.
Is there any black magic that can be done with propiotory IE only APIs? Maybe with VBScript?
ES5 has a enumerable flag. Example
Example
var getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor
, pd = getOwnPropertyDescriptor(Object.prototype, "toString");
assert(pd.enumerable === false, "enumerability has the wrong value");
Partial implementation
Partial implementation is do-able by having Object.keys
and Object.getOwnPropertyNames
filter out new non-enumerable properties using the shimmed Object.defineProperty
.
Introduction
This allows for properties to be non enumerable. This clearly means that Example
for (var key in {}) {
assert(key !== "toString", "I should never print");
}
This allows us to add properties to say Object.prototype
(Example)
Object.defineProperty(Object.prototype, "toUpperCaseString", {
value: function toUpperCaseString() {
return this.toString().toUpperCase();
},
enumerable: false
});
for (var key in {}) {
assert(key !== "toUpperCaseString", "I should never print");
}
console.log(({}).toUpperCaseString()); // "[OBJECT OBJECT]"
Question
How can we emulate this in non-ES5 pliant browsers?
Browser pat table
In this case we care about potentially solving this for
- IE < 9 (IE8 only works on DOM objects)
- Firefox 3.6
- Safari 4
- Opera 11.5 (Opera 11.6 solves this).
The ES5-shim does not have a solution for this.
The ES5 shim has a solution for most ES5 features that will break your code if it doesn't work.
Is there any black magic that can be done with propiotory IE only APIs? Maybe with VBScript?
Share Improve this question edited Jan 24, 2012 at 20:18 Raynos asked Jan 18, 2012 at 21:54 RaynosRaynos 170k57 gold badges357 silver badges398 bronze badges 11-
6
A shim probably won't ever be able to achieve this, because the behavior of
for... in
is hardwired into the language spec and the interpreter proper. You're asking for something that is just not there in non-ES5-pliant browsers. +1 though, your questions are always interesting. – Frédéric Hamidi Commented Jan 18, 2012 at 22:01 - @FrédéricHamidi after seeing the VB black magic linked above, I'm not so sure of anything anymore. – Domenic Commented Jan 18, 2012 at 22:04
- @FrédéricHamidi My knowledge says "no way is this possible" but browsers have so many weird propietory APIs that it may be possible – Raynos Commented Jan 18, 2012 at 22:09
- Have you tried traceur? code.google./p/traceur-piler – thwd Commented Jan 23, 2012 at 5:46
- @Tom how does traceur help? I can't possibly see how. – Raynos Commented Jan 23, 2012 at 9:51
3 Answers
Reset to default 4You can do it via code-rewriting. Rewrite every use of for (p in o) body
to
for (p in o) {
if (!(/^__notenum_/.test(p) || o['__notenum_' + p])) {
body
}
}
and then you can mark properties not enumerable by defining a __notenum_...
property. To be patible you would have to tweak the above to make sure that __notenum_propname
is defined at the same prototype level as propname
, and if you use them, overwrite eval
and new Function
to rewrite.
That's basically what ES5/3 does.
Partial.js by Jake Verbaten is the answer to it.
The partial.js is as follows
/* partial non-enumerable property implementation
Adds a flag to a weakmap saying on obj foo property bar is not enumerable.
Then checks that flag in Object.keys emulation.
*/
// pd.Name :- https://github./Raynos/pd#pd.Name
var enumerables = pd.Name();
Object.defineProperty = function (obj, name, prop) {
if (prop.enumerable === false) {
enumerables(obj)[name] = true;
}
...
};
Object.keys = function (obj) {
var enumerabilityHash = enumerables(obj), keys = [];
for (var k in obj) {
if (obj.hasOwnProperty(k) && !enumerabilityHash[k]) {
keys.push(k);
}
}
return keys;
};
Object.getOwnPropertyNames = function (obj) {
var keys = [];
for (var k in obj) {
if (obj.hasOwnProperty(k)) {
keys.push(k);
}
}
};
I hope this helps the guys searching for this fix.
If you do care a lot about IE8/IE7 then you can do
for (p in o) {
if (o.hasOwnProperty(p)) { body }
}
There is no real "hack" alternative but this could be a work-around for simple cases
The accepted answer doesn't really work for literals i.e. strings ""
, numbers 3
, or booleans true
本文标签: javascriptIs it possible to emulate nonenumerable propertiesStack Overflow
版权声明:本文标题:javascript - Is it possible to emulate non-enumerable properties? - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743614853a2510577.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论