Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

bind perf #9369

Closed
wants to merge 3 commits into from
Closed

bind perf #9369

wants to merge 3 commits into from

Conversation

jimmywarting
Copy link
Contributor

Speeds up chrome with ~10% firefox by ~5%

Speeds up chrome with ~10% firefox by ~5%
@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

I think all of the supported browsers use textContent now, so there may be a few places where we can remove conditionally assigning to textContent / innerText (as an additional improvement) --- such as in ngSanitize

@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

That's unrelated to the issue, just something that may be worth doing.

I think we might want to iterate over each element in thee jquery object though, in case there's more than one (ng-bind-start / ng-bind-end)

@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

I'm wrong about the multi-element thing, i guess we disabled multiElement for ngBind

@jimmywarting
Copy link
Contributor Author

Didn't think there was any senario where there would be more then one element in a directive.
This was intended for 1.3 where we only support IE9+

@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

there used to be scenarios where ng-bind would have multiple elements, and tests which used it, but we changed the tests to use a custom directive since ngBind had multiElement disabled. (see e8066c4#diff-348c2f3781ed66a24894c2046a52c628L5552)

@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

In practice I don't think this is likely to make much of a difference since interpolation is used way more frequently than ng-bind --- but you should measure the difference in the largetable with ng-bind mode and report screenshots

@jimmywarting
Copy link
Contributor Author

I did do it with a large table...
not sure if i did it right but here is my plnkr: http://plnkr.co/edit/pdm7C1i2kF34dHsg3TWp?p=preview

Don't have a single page with both measure or a screenshot

but here is two results:

Firefox:
0.6308539999999994 textContent
0.6628670000009151 text()

Chrome:
1.4599999994970858 textContent
1.6260000120382756 text()

@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

specifically I meant https://github.com/angular/angular.js/tree/master/benchmarks/largetable-bp (there are instructions to run it in the benchmarks folder)

@jbedard
Copy link
Collaborator

jbedard commented Oct 1, 2014

Since we no longer need the jqLite/jQuery object we can let it get GCed...

element = element[0];
$scope.$watch(....
   element.textContent = ...

Personally I use ng-bind quite often instead of interpolation since it is faster (at least when in large ng-repeats, etc.).

@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

most people won't do that since it's much more convenient and more powerful to use interpolation. I don't have much of an opinion on letting the wrapper get GC'd, if you have a lot of these it's probably less helpful than you might think, but I don't care one way or the other there.

Anyways, please post largetable-bp results

@jbedard
Copy link
Collaborator

jbedard commented Oct 1, 2014

I think it saved me ~100k or so with an average dataset in a table, and avoids the [0] per watch. Probably not really noticeable, but why not?

@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

I'm all for it, I just want to supplement the PR with data explaining why we made the change

@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

fwiw, it basically looks good to me, but I think it's important to measure. If you won't, then I'll do it quickly myself

@caitp
Copy link
Contributor

caitp commented Oct 1, 2014

https://www.dropbox.com/s/76wxqbvduade52s/big_table_benchmark_b1ee5396_vs_d580a954.zip?dl=0 so what I'm seeing from this (and I did retry the benches a number of times, this is with a single tab open in the latest Canary and it's pretty consistent), it doesn't make a significant difference, but it actually looks like the master branch is winning very slightly.

This is with the ngBind + function call version of the test. Anyways, it doesn't look like it makes a significant difference one way or the other, WDYT @IgorMinar

@jimmywarting
Copy link
Contributor Author

Wasn't available ATM. but thanks for the benchmark

GC jqlite wrapper.
Same for BindTemplate
@IgorMinar
Copy link
Contributor

lgtm. it doesn't make a huge difference but there are no downsides. so why not.

@IgorMinar IgorMinar modified the milestones: 1.3.0-rc.5, 1.3.0 Oct 2, 2014
@caitp
Copy link
Contributor

caitp commented Oct 2, 2014

will merge after fixing lint issues

$compile.$$addBindingInfo(element, interpolateFn.expressions);
attr.$observe('ngBindTemplate', function(value) {
element.text(value);
element.textContent = value;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why are we not checking for an undefined value as we are above, in this case? gecko and webkit will render undefined, which we probably don't want (do we? Chrome won't, fwiw)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we should use a condition here too, so that the behaviour is consistent across browsers

@@ -57,12 +57,13 @@ var ngBindDirective = ['$compile', function($compile) {
compile: function ngBindCompile(templateElement) {
$compile.$$addBindingClass(templateElement);
return function ngBindLink(scope, element, attr) {
var element = element[0];
$compile.$$addBindingInfo(element, attr.ngBind);
scope.$watch(attr.ngBind, function ngBindWatchAction(value) {
// We are purposefully using == here rather than === because we want to
// catch when value is "null or undefined"
// jshint -W041
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This comment is nonsense, because all supported browsers will output the empty string in the case of null --- only undefined is an issue. I am going to remove the comment and use strict equals

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I just realized you didn't add the comment =) but yeah... removing it

caitp pushed a commit to caitp/angular.js that referenced this pull request Oct 2, 2014
"Speeds up chrome with ~10% firefox by ~5%"

We don't really see this result in benchmarks (https://www.dropbox.com/s/76wxqbvduade52s/big_table_benchmark_b1ee5396_vs_d580a954.zip?dl=0)
However, it's basically harmless.

Side effects:

Use strict equality check for `undefined` to replace with empty string. Most target browsers will output `undefined` rather than the empty
string if we don't do this. Previously, ngBindTemplate did not perform this check. However the change has been made to make behaviour
consistent across all target browsers (chrome does output the empty string).

Closes angular#9369
... and removed the comment
@caitp caitp closed this in 074a146 Oct 2, 2014
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants