-
Notifications
You must be signed in to change notification settings - Fork 113
Public fields refinement proposal with prototype-based WeakMap semantic #174
Comments
This would make the public fields be inherited accessor properties, instead of own data properties. That would break, for example, Object.keys/values/entries/assign. |
@ljharb I have made it Object.defineProperty(Foo.prototype, 'bar', {
set(value) { Foo_fields_bar.set(this, value) },
get() { return Foo_fields_bar.get(this) },
enumerable: true, // <-- this
configurable: true,
}) It should always behave same as current class getter/setter. |
Apologies, I’ve updated my comment. Not being an own property then, remains the problem. |
@ljharb Thanks for point that out, I haven't considered that. But given class is designed to have a static shape, also this is the existing behavior of class getter/setter property, then any library/framework rely on property enumeration already doesn't support getter/setter usage. So I think it could be a minor issue than inheritance, which is a designed use case for class. |
It’s not designed to have a static shape - you have always been able to, and always will be able to, change the shape however you like in the constructor. |
This isn't Java. 😉 |
It's true that one can always change the shape arbitrarily in constructor, but what class fields contributes to are only the static parts of that (even with computed property name it's still static in the lifecycle), so the use case of dynamic property enumeration should not have much overlap with class fields usage. |
@trotyl the names are part of the eventual instance shape, sure, but you can’t get a full picture of the instance from just the prototype - that’s not a guarantee classes have in JS. As for dynamic property enumeration, there are plenty of overlapping use cases - even merely “i don’t want to type out the list of static names both inside and outside the class declaration” is sufficient. |
I believe @zenparsing suggested something along these lines way back when. Two concerns:
|
It's a declaration so engines can proactively optimize it.
We are forced to use own properties to store states because we never have any other ergonomic ways. So saying it is more "javascript-y" is logically meaningless. And it's very "double standards" to criticize getter/setter is not "javascript-y", but all:
...could be "javascript-y". |
These sorts of non-local optimizations can be hard to implement before type profiling kicks in, given how dynamic a language JavaScript is. The assessment about what's difficult to optimize was based on talking with the engine implementers who make these optimizations work in practice.
Well, touche. Not sure if this helps, but I see a sort of spectrum of possible semantics, which you could coarsely call "dynamic vs reliable", with |
Hard to implement != impossible to implement. I don't believe it will be harder than all past optimization engines did.
This is class definitions! Most time it's just static. Engines can proactively optimize it, if some bad dynamic thing happen just deoptimize it.
I heard something different from other engine guys. 🤣 After all, at least JIT just works. Currently it may be a little slow before profiling kicks, so what? Does any programmers do not use getter/setter in the cases where getter/setter is suitable? We in long time did not have good optimization on ES6 features, but we still forward to ES6 and programmers are like to use ES6 features in most cases. So please do not worry about that. I bet you $10000 90%+ js programmers will very happy use such a little temporarily performance loss to exchange no pain of [[Define]] vs [[Set]]. |
Well, there's no need to be ambiguous here; this was from a discussion that I had with @verwaest. I'd like to collect as much implementation feedback as possible to make sure we're not specifying un-optimizable features. There's a lot of effort that V8 is putting into startup performance, and I'd like to not make it worse in this case (for the same reason, I'm working on restrictions to decorators). It's also going to be important to retain the buy-in of engine implementers to continue evolving the language. cc @gsathya
Many ES6 features remain very difficult to optimize, even if some of them have improved. |
I think I already make my point clear. Whenever accessors are suitable, programmers will use it anyway. And we know JIT works for accessors, that's enough for 90%+ programmers, 90%+ cases. If we can get startup much faster, great! Thank you, engine engineer heros! If not, ok, programmers will not blame you. On the other hand, the pain of [[Define]] [[Set]] is very huge, most programmers are just shocked and have no idea why TC39 send them such dilemma. |
Thanks Dan.
Sorry, but I think you misunderstand JS performance at least in the
browser. Do definitely use abstractions that make sense in the places where
they make sense for your code, but the JIT even isn't involved in a lot of
performance critical scenarios. Defaulting to something much slower just
for convenience of small pieces of code where you can just it yourself with
almost no cost is just a bad idea if you want the web to be fast.
The sufficiently smart compiler is a myth we really need to get rid of if
we care about load-time and other types of "warmup" (e.g., latency).
…On Thu, 13 Dec 2018, 18:38 HE Shi-Jun ***@***.*** wrote:
I think I already make my point clear. Whenever accessors are suitable,
programmers will use it anyway. And we know JIT works for accessors, that's
enough for 90%+ programmers, 90%+ cases. If we can get startup much faster,
great! Thank you, engine engineer heros! If not, ok, programmers will not
blame you.
On the other hand, the pain of [[Define]] [[Set]] is very huge, most
programmers are just shocked and have no idea why TC39 send them such
dilemma.
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
<#174 (comment)>,
or mute the thread
<https://github.com/notifications/unsubscribe-auth/ArA8RvXLN_oYeJZhM2froNNS1niAWA0aks5u4pB9gaJpZM4YtLrd>
.
|
Maybe this is a translation thing, but "very huge" implies something much more consequential than Set vs. Define. I agree that it's a very significant and serious problem (well, much more so in the case of Define), but the impact on developers will be on a pretty small scale; there's only one relatively infrequent use cases where it comes up (redeclaring the same property on a subclass). Yes I know it could be a source of insidious bugs that could bite you years later, and that's very concerning, but let's try to have a little perspective. It's not an earth-shattering disaster. |
Hope you can teach me.
Don't understand it. Are you saying JIT can't optimize accessors even it been called frequently?
Don't understand it. Do you mean we should drop JS and always write C++ and compile to wasm so we are not "defaulting to something much slower just for convenience"?
So use wasm? |
On Thu, 13 Dec 2018, 19:37 HE Shi-Jun ***@***.*** wrote:
Sorry, but I think you misunderstand JS performance at least in the
browser.
Hope you can teach me.
Do definitely use abstractions that make sense in the places where they
make sense for your code, but the JIT even isn't involved in a lot of
performance critical scenarios.
Don't understand it. Are you saying JIT can't optimize accessors even it
been called frequently?
It's easy for the JIT. The JIT just doesn't even kick in yet during a lot
of performance-crucial phases.
And for example an interpreter is counter-intuitively better for startup
performance of more code than a simple compiler to native code. You want
the combination of low-latency startup with option to tier up to great
stable peak performance.
The same is true for eagerly making complex assumptions. It's a bad idea
for startup, bit crucial for peak performance.
Defaulting to something much slower just for convenience of small pieces of
code where you can just it yourself with almost no cost is just a bad idea
if you want the web to be fast.
Don't understand it. Do you mean we should drop JS and always write C++
and compile to wasm so we are not "defaulting to something much slower just
for convenience"?
Trolling++. Always very helpful.
|
@verwaest Ok, let's ask a very simple question. If we make class Test {
x = 0
} using class Test {
#x = 0
get x() {return this.#x}
set x(v) {this.#x = v}
} semantic. How bad performance could it be compare to the current own property semantic? 5%? 10%? 50%? |
It's somewhere between 2x and 6x slower depending on the VM.
…On Fri, Dec 14, 2018 at 6:28 AM HE Shi-Jun ***@***.***> wrote:
@verwaest <https://github.com/verwaest> Ok, let's ask a very simple
question.
If we make
class Test {
x = 0
}
using
class Test {
#x = 0
get x() {return this.#x}
set x(v) {this.#x = v}
}
semantic.
How bad performance could it be compare to the current own property
semantic?
5%? 10%? 50%?
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
<#174 (comment)>,
or mute the thread
<https://github.com/notifications/unsubscribe-auth/ArA8RkLJuqIuTZMmRiPJChfrEuN_uPUGks5u4zb_gaJpZM4YtLrd>
.
|
@verwaest Do you mean with JIT or without JIT? I guess the latter? |
@hax We're discussing startup performance, before the JIT kicks in. |
@littledan So is that mean calling getter/setter 2x~6x slower before JIT kicks in? Or mean something else (like class initialize time?) |
Yes |
That; and if it's metamorphic. JITs bail out if code isn't regular enough.
Hence lots of library and framework code at least will always have that
performance difference even with JIT.
…On Fri, 14 Dec 2018, 15:48 Daniel Ehrenberg ***@***.*** wrote:
So is that mean calling getter/setter 2x~6x slower before JIT kicks in?
Yes
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
<#174 (comment)>,
or mute the thread
<https://github.com/notifications/unsubscribe-auth/ArA8RtrJ8BB-zN2nqxiOUUaTS8mwxWRUks5u47o2gaJpZM4YtLrd>
.
|
As discussed in many threads,
[[Define]]
is the semantically right behavior, but current define-on-instance approach having several issues regarding inheritance.Here's a proposal with define-on-prototype with
WeakMap
semantic on this which could avoids those problem.For a class field:
Would be transformed to:
This would preserve the memory-allocation per instance:
In the meantime, it makes inheritance expectable. Consider existing issues:
Issue: tc39/proposal-class-public-fields#16 (comment)
Demo: https://codesandbox.io/s/0qw90xpq3w
Code:
Fixes the current behavior, now class methods on subclass is able to override class fields on base (the base class may not like it, but it's up to the derived class to decide).
Issue: tc39/proposal-class-public-fields#16 (comment)
Demo: https://codesandbox.io/s/jjyrr065mw
Code:
Keeps current behavior that a subclass field will override base accessor.
Issue: tc39/proposal-class-public-fields#38 (comment)
Demo: https://codesandbox.io/s/x72o9zjrlo
Code:
Keeps current behavior that different instance won't share fields.
Issue: tc39/proposal-class-public-fields#57 (comment)
Demo: https://codesandbox.io/s/30mjyl367p
Code:
Keeps current behavior that uninitialized field would override base fields with
undefined
.Issue: #151 (comment)
Demo: https://codesandbox.io/s/6z0jr0wl5n
Code:
Keeps the current behavior that subclass field could override base accessor.
Issue: #151 (comment)
Demo: https://codesandbox.io/s/7w34x852oq
Code:
Fixes current behavior, now subclass accessor can override base class fields.
The text was updated successfully, but these errors were encountered: