Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make Y.bind consistent with native bind. Fixes #2532886 #320

Closed
wants to merge 3 commits into from
Closed

Make Y.bind consistent with native bind. Fixes #2532886 #320

wants to merge 3 commits into from

Conversation

yiminghe
Copy link

@ghost ghost assigned sdesai Oct 22, 2012
@rgrove
Copy link
Contributor

rgrove commented Nov 17, 2012

Can you describe in detail what inconsistencies these changes address? I'm curious, and it's not obvious from the code.

@juandopazo
Copy link
Member

If I understood correctly, it's trying to fix this:

// Test in a modern browser
YUI().use('oop', function (Y) {

  function Foo(a,b,c) {
    this.a = a;
    this.b = b;
    this.c = c;
  }
  // native bind
  var Bound = Foo.bind({}, 1, 2);
  var b = new Bound(3);
  console.log(Object.keys(b));  // ['a', 'b', 'c']
  console.log(b.c); // 3

  // YUI bind
  var YBound = Y.bind(Foo, {}, 1, 2);
  var c = new YBound(3);
  console.log(Object.keys(c));  // [], but should be ['a', 'b', 'c']
  console.log(c.c); // undefined, but should be 3

});

@yiminghe
Copy link
Author

yes,just as @juandopazo shows, maybe my user case is rare, so I totally understand if this patch can not be accepted.

@davglass
Copy link
Member

👎 Failing tests

@juandopazo
Copy link
Member

@davglass is Travis already building YUI before testing? I'm trying this out in my PC and after building all oop tests pass.

return fn.apply(c || fn, args);
};
};
Y.bind = bind_(0, bind_, null, 0);
Copy link
Member

Choose a reason for hiding this comment

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

Is bind(0, bind, null, 0) for binding bind and/or bind.call?

Copy link
Author

Choose a reason for hiding this comment

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

bind(0, bind, null, 0 is for binding bind_, no need for bind.call.

@davglass TC passed on my pc, sorry i do not know why it failed here.

@davglass
Copy link
Member

@jenny
Copy link
Member

jenny commented Jan 10, 2013

@juandopazo @rgrove Looks like tests are passing, how do you feel about this merge?

@rgrove
Copy link
Contributor

rgrove commented Jan 11, 2013

I'm not opposed, although I'd kinda like to see a performance comparison. I don't think this change is worth making Y.bind() slower, so we should make sure it's still roughly as fast as the previous implementation.

@davglass
Copy link
Member

Agreed!

@juandopazo
Copy link
Member

It'd be nice to add my earlier example as a test, since that's what this pull request is fixing.

@davglass
Copy link
Member

Here's a very simple perf test case that shows this new one is quite slower than the original:

http://jsperf.com/y-bind

@juandopazo
Copy link
Member

The biggest culprit for the performance dive is binding bind_. Using Y.Array for this is anther bad idea for performance. Y.Array is mostly for forcing stuff that might be an HTMLCollection into an array. And here we know this are arguments objects that can be sliced safely.

Another thing I just realized is that Y.bind('methodName', obj) is pretty dangerous. It's not a shortcut for Y.bind(obj.methodName, obj), instead it's always looking for the latest value of the obj.methodName property. That sounds bad and slower.

I propose we go with something more like this:

var bind2 = function (method, obj) {
  // bind the method once, so the actual function
  // being bound does not change
  if (typeof method === 'string') {
    method = obj[method];
  }

  if (typeof method !== 'function') Y.log('Y.bind - what is trying to be bound is not callable', 'error');

  var args  = slice.call(arguments, 2),
      Noop  = function () {},
      bound = function () {
        return method.apply(
          // Allow the bound function to be called with "new"
          // In that case |this| will be a new object instace
          // of Noop
          this instanceof Noop ? this : obj,
          args.concat(slice.call(arguments))
        );
      };

  Noop.prototype = method.prototype;
  bound.prototype = new Noop();

  return bound;
};

@davglass
Copy link
Member

If we are going to be modifying Y.bind I would really like to see a
shim that uses native bind when available then fall back to ours.

@rgrove
Copy link
Contributor

rgrove commented Jan 16, 2013

@juandopazo I like this version a lot. +1.

@davglass Native bind may actually be slower, but it's worth testing.

@davglass
Copy link
Member

I want to see the perf numbers from @juandopazo's bind2 and Native bind against our old one.

@ericf
Copy link
Member

ericf commented Jan 17, 2013

Looked at what Lodash does, saw this:
https://github.com/bestiejs/lodash/blob/master/lodash.js#L122

@juandopazo
Copy link
Member

Wow! That sounds like a bad idea! What if a certain browser improves its
performance?

@ericf
Copy link
Member

ericf commented Jan 17, 2013

Yeah… Lodash is going overboard with micro-optimizations here.

@juandopazo
Copy link
Member

@davglass Here you go http://jsperf.com/y-bind/3

@apm
Copy link
Contributor

apm commented Jan 17, 2013

"Another thing I just realized is that Y.bind('methodName', obj) is pretty dangerous. It's not a shortcut for Y.bind(obj.methodName, obj), instead it's always looking for the latest value of the obj.methodName property. That sounds bad and slower."

This was put in as a a way to provide a way for subclasses to override early-bound methods. Please don't completely remove the old version if you choose to replace it with the slow version.

@davglass
Copy link
Member

These new versions are way too slow for this to be merged in.

@davglass
Copy link
Member

And a +1 to @apm's comment.

@davglass
Copy link
Member

Closing this based on the latest Open Hours Hangout.

Since this is a breaking change and the performance is subpar, I'm closing this for now.

If anyone comes up with a better approach that is performant enough, please file a new Pull Request.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

None yet

8 participants