This is a personal blog about things that I like to call my interests; namely Music, Music Production, Photography, and Computer Programming.
For my blog on food and full of lots of delicious recipes go to Foodgeek.
This is a personal blog about things that I like to call my interests; namely Music, Music Production, Photography, and Computer Programming.
For my blog on food and full of lots of delicious recipes go to Foodgeek.
Outlined in this post by John Resig (Mozilla nut and general JavaScript guru) is an idea on how to implement method overloading in JavaScript.
The idea in general is good, but honestly, using only parameter count as the way to distinguish between functions can be a bit limiting in my opinion, so here is my take on it, which builds on prototypes object model.
Also an important part of me is aesthetics, and I like the way that prototype declares its methods. Here is the syntax I came up with:
var Test = Class.create({ initialize: function () { console.log('init'); }, // regular method test1: function (num, str) { console.log('test1(num:'+ num + ',str:"' + str + '")'); }, // overloading using an array of functions with different // parameter counts test2: [ function (a) { console.log('test2(a:"' + a + '")'); }, function (a, b) { console.log('test2(a:"' + a + '",b:"' + b + '")'); } ], // overloading using an array of objects that specify // the function parameter signature and the functions // themselves test3: [ { signature: [String], method: function (str) { console.log('test3(str:"' + str + '")'); } }, { signature: [Number], method: function (num) { console.log('test3(num:' + num + ')'); } }, { signature: [Number, String], method: function(num, str) { console.log('test3(num:'+ num + ',str:"' + str + '")'); } } ] });
Using the following test script:
var test = new Test(); test.test1(1, 'test string 1'); test.test2('test string 2'); test.test2('test string 3', 'test string 4'); test.test3('test string 5'); test.test3(2); test.test3(3, 'test string 6');
We get the following output in Firebug/Firebug light:
init test1(num:1,str:"test string 1") test2(a:"test string 2") test2(a:"test string 3",b:"test string 4") test3(str:"test string 5") test3(num:2) test3(num:3,str:"test string 6")
All in a nice and neat little package. Here is the actual code. Plug it into your site wherever appropriate:
Function.prototype.getName = function () { var name = /\W*function\s+([\w\$]+)\(/.exec(this.toString()); if (!name) return 'No name'; return name[1]; }; Class.Methods.addMethods = Class.Methods.addMethods.wrap( function (callOriginal, source) { var properties = Object.keys(source) for (var i = 0, length = properties.length; i < length; i++) { var property = properties[i], methods = source[property]; if (methods instanceof Array) { if (methods.all(function (m) { return typeof m == 'function'; })) { source[property] = function (methods) { var args = $A(arguments); args.shift(); methods.each(function (m) { if (m.length == args.length) m.apply(this, args); }, this); } .bind(this, methods); } else if (methods.all(function (m) { return typeof m == 'object'; })) { methods.each(function (m) { m.signature = m.signature.inject([], function (s, v) { return s + v.getName(); }) }); source[property] = function (methods) { var args = $A(arguments); args.shift(); var argSig = args.inject([], function (s, v) { return s + v.constructor.getName(); }); methods.each(function (m) { if (m.signature == argSig) m.method.apply(this, args); }, this); } .bind(this, methods); } else throw 'Unknown contents of overload array'; } } callOriginal(source); });
Testing using Prototype 1.7 and Firefox 4.0, Chrome 11.0, Internet Explorer 9.0 (+compatibility mode), Opera 11.01, Safari 5.0.4.