John Resig (di fama jQuery) fornisce un'implementazione concisa di Simple JavaScript Inheritance. Il suo approccio ha ispirato il mio tentativo di migliorare ulteriormente le cose. Ho riscritto funzione originaria Class.extend
di Resig per includere i seguenti vantaggi:Miglioramento dell'ereditarietà JavaScript semplice
prestazioni - meno overhead durante la definizione della classe, la costruzione oggetto e metodo della classe base chiama
Flessibilità - ottimizzato per nuovi browser compatibili con ECMAScript 5 (ad es. Chrome), ma fornisce uno "shim" equivalente per i browser più vecchi (es. IE6)
Compatibilità - convalida in modalità rigorosa e offre una migliore compatibilità degli strumenti (ad es. commenti VSDoc/JSDoc, Visual Studio IntelliSense, ecc)
Semplicità - non c'è bisogno di essere un "ninja" per capire il codice sorgente (ed è ancora più semplice se si perde le caratteristiche ECMAScript 5)
Robustezza - passa più test di unità "case d'angolo" (es imperativi toString in IE)
Perché sembra quasi troppo bello per essere vero, voglio assicurare la mia logica non lo fa ha qualche difetto fondamentale o bu gs, e vedi se qualcuno può suggerire miglioramenti o smentire il codice. Con questo, presento la funzione classify
:
function classify(base, properties)
{
/// <summary>Creates a type (i.e. class) that supports prototype-chaining (i.e. inheritance).</summary>
/// <param name="base" type="Function" optional="true">The base class to extend.</param>
/// <param name="properties" type="Object" optional="true">The properties of the class, including its constructor and members.</param>
/// <returns type="Function">The class.</returns>
// quick-and-dirty method overloading
properties = (typeof(base) === "object") ? base : properties || {};
base = (typeof(base) === "function") ? base : Object;
var basePrototype = base.prototype;
var derivedPrototype;
if (Object.create)
{
// allow newer browsers to leverage ECMAScript 5 features
var propertyNames = Object.getOwnPropertyNames(properties);
var propertyDescriptors = {};
for (var i = 0, p; p = propertyNames[i]; i++)
propertyDescriptors[p] = Object.getOwnPropertyDescriptor(properties, p);
derivedPrototype = Object.create(basePrototype, propertyDescriptors);
}
else
{
// provide "shim" for older browsers
var baseType = function() {};
baseType.prototype = basePrototype;
derivedPrototype = new baseType;
// add enumerable properties
for (var p in properties)
if (properties.hasOwnProperty(p))
derivedPrototype[p] = properties[p];
// add non-enumerable properties (see https://developer.mozilla.org/en/ECMAScript_DontEnum_attribute)
if (!{ constructor: true }.propertyIsEnumerable("constructor"))
for (var i = 0, a = [ "constructor", "hasOwnProperty", "isPrototypeOf", "propertyIsEnumerable", "toLocaleString", "toString", "valueOf" ], p; p = a[i]; i++)
if (properties.hasOwnProperty(p))
derivedPrototype[p] = properties[p];
}
// build the class
var derived = properties.hasOwnProperty("constructor") ? properties.constructor : function() { base.apply(this, arguments); };
derived.prototype = derivedPrototype;
derived.prototype.constructor = derived;
derived.prototype.base = derived.base = basePrototype;
return derived;
}
E l'utilizzo è quasi identico a Resig di eccezione per il nome del costruttore (constructor
vs. init
) e la sintassi per le chiamate metodo della classe base.
/* Example 1: Define a minimal class */
var Minimal = classify();
/* Example 2a: Define a "plain old" class (without using the classify function) */
var Class = function()
{
this.name = "John";
};
Class.prototype.count = function()
{
return this.name + ": One. Two. Three.";
};
/* Example 2b: Define a derived class that extends a "plain old" base class */
var SpanishClass = classify(Class,
{
constructor: function()
{
this.name = "Juan";
},
count: function()
{
return this.name + ": Uno. Dos. Tres.";
}
});
/* Example 3: Define a Person class that extends Object by default */
var Person = classify(
{
constructor: function(name, isQuiet)
{
this.name = name;
this.isQuiet = isQuiet;
},
canSing: function()
{
return !this.isQuiet;
},
sing: function()
{
return this.canSing() ? "Figaro!" : "Shh!";
},
toString: function()
{
return "Hello, " + this.name + "!";
}
});
/* Example 4: Define a Ninja class that extends Person */
var Ninja = classify(Person,
{
constructor: function(name, skillLevel)
{
Ninja.base.constructor.call(this, name, true);
this.skillLevel = skillLevel;
},
canSing: function()
{
return Ninja.base.canSing.call(this) || this.skillLevel > 200;
},
attack: function()
{
return "Chop!";
}
});
/* Example 4: Define an ExtremeNinja class that extends Ninja that extends Person */
var ExtremeNinja = classify(Ninja,
{
attack: function()
{
return "Chop! Chop!";
},
backflip: function()
{
this.skillLevel++;
return "Woosh!";
}
});
var m = new Minimal();
var c = new Class();
var s = new SpanishClass();
var p = new Person("Mary", false);
var n = new Ninja("John", 100);
var e = new ExtremeNinja("World", 200);
E qui sono le mie prove QUnit cui tutto passa:
equals(m instanceof Object && m instanceof Minimal && m.constructor === Minimal, true);
equals(c instanceof Object && c instanceof Class && c.constructor === Class, true);
equals(s instanceof Object && s instanceof Class && s instanceof SpanishClass && s.constructor === SpanishClass, true);
equals(p instanceof Object && p instanceof Person && p.constructor === Person, true);
equals(n instanceof Object && n instanceof Person && n instanceof Ninja && n.constructor === Ninja, true);
equals(e instanceof Object && e instanceof Person && e instanceof Ninja && e instanceof ExtremeNinja && e.constructor === ExtremeNinja, true);
equals(c.count(), "John: One. Two. Three.");
equals(s.count(), "Juan: Uno. Dos. Tres.");
equals(p.isQuiet, false);
equals(p.canSing(), true);
equals(p.sing(), "Figaro!");
equals(n.isQuiet, true);
equals(n.skillLevel, 100);
equals(n.canSing(), false);
equals(n.sing(), "Shh!");
equals(n.attack(), "Chop!");
equals(e.isQuiet, true);
equals(e.skillLevel, 200);
equals(e.canSing(), false);
equals(e.sing(), "Shh!");
equals(e.attack(), "Chop! Chop!");
equals(e.backflip(), "Woosh!");
equals(e.skillLevel, 201);
equals(e.canSing(), true);
equals(e.sing(), "Figaro!");
equals(e.toString(), "Hello, World!");
Chiunque vede qualcosa di sbagliato con il mio approccio vs John Resig di original approach? Suggerimenti e feedback sono ben accetti!
NOTA: il codice precedente è stato modificato in modo significativo da quando ho inizialmente postato questa domanda. Quanto sopra rappresenta l'ultima versione. Per vedere come si è evoluto, si prega di controllare la cronologia delle revisioni.
Consiglierei 'Object.create' e [traitsjs] (http://traitsjs.org/). L'ereditarietà non va bene in javascript, usa la composizione dell'oggetto – Raynos
Forse non mi sono ancora abituato, ma la sintassi dei tratti mi fa girare la testa. Penso che passerò fino a quando non otterrà un seguito ... – Will