Si preguntas a mucha gente cual es el objetivo de new en JavaScript te dirán que el de crear objetos. Y estarán en lo cierto, aunque esa definición no es del todo precisa. Y es que mucha gente no entiende exactamente que hace new así que vamos a dedicarle este post.
Creación de objetos sin new
La verdad es que new no es necesario en JavaScript. Se pueden crear objetos perfectamente sin necesidad de usar new:
- var foo = {
- name: 'foo',
- getUpperName: function () {
- return this.name.toUpperCase();
- }
- }
Esta manera de crear objetos la conocemos como object notation y nos permite definir objetos con campos y métodos. Por supuesto, en este caso solo creamos un objeto pero basta con declarar una función que tenga ese código para crear tantos objetos como queramos:
- var Foo = function (name) {
- var foo = {
- name: name || '',
- getUpperName: function () {
- return this.name.toUpperCase();
- }
- }
- return foo;
- }
- var f2 = Foo("f2");
- f2.getUpperName(); // devuelve F2
Nos podemos preguntar cual es el prototipo de cada objeto creado mediante una llamada a Foo. Pues, como es de esperar, el prototipo de f2 será Object.prototype.
Podríamos indicar que los objetos creados por Foo heredan de Foo.prototype, con un código similar al:
- var Foo = function (name) {
- var foo = {
- name: name || '',
- getUpperName: function () {
- return this.name.toUpperCase();
- }
- }
- Object.setPrototypeOf(foo, Foo.prototype);
- return foo;
- }
- var f2 = Foo("f2");
- f2.getUpperName(); // devuelve F2
- f2 instanceof Foo; // devuelve true
Lo interesante es que establecer como prototipo del objeto creado el objeto Foo.prototype, automáticamente hace que instanceof Foo devuelva true sobre todos los objetos creados por la función Foo. Vale, estoy usando Object.setPrototypeOf que es una función nueva de ES2015, pero en ES5 clásico tenemos Object.create que nos permite hacer lo mismo:
- var Foo = function (name) {
- var foo = Object.create(Foo.prototype);
- foo.name = name || '';
- foo.getUpperName = function () {
- return this.name.toUpperCase();
- }
- return foo;
- }
- var f2 = Foo("f2");
- f2.getUpperName(); // devuelve F2
- f2 instanceof Foo; // devuelve true
¿Entonces… qué hace new?
Bueno, new se introdujo para dar una “sintaxis familiar” a los desarrolladores que venían de Java, pero básicamente hace lo mismo que hemos hecho nosotros. En JavaScript hablamos de funciones constructoras, pero no hay sintaxis para declarar una función constructora. Una función constructora es aquella pensada para ser llamada con new, pero que quede claro: es usar o no new al invocar una función lo que convierte a esa en constructora.
Cuando usamos new en una función, llamémosla Bar, ocurre lo siguiente:
- Se crea automáticamente un objeto vacío que hereda de Bar.prototype
- Se llama a la función Bar, con el valor de this enlazado al objeto creado en el punto anterior
- Si (y solo sí) la función Bar devuelve undefined, new devuelve el valor de this dentro de Bar (es decir, el objeto creado en el punto 1).
Como, gracias al punto 2, dentro de una función llamada con new el valor de this es el objeto creado en el punto 1, decimos que en una función constructora el valor de this es “el objeto que se está creando”. Y es por ello que las codificamos de esa manera:
- var Foo = function (name) {
- this.name = name || '';
- this.getUpperName = function () {
- return this.name.toUpperCase();
- }
- }
Pero para que veas exactamente hasta que punto new no era “imprescindible” es porque incluso podríamos hacer una especie de polyfill para new:
- var functionToCtor = function (ctor) {
- var o = Object.create(ctor.prototype);
- var args = Array.prototype.slice.call(arguments)
- args.splice(0, 1);
- var v = ctor.apply(o, args);
- return v === undefined ? o : v;
- }
La función functionToCtor hace lo mismo que new. Es decir, en este caso dada la función constructora Foo ambas expresiones son equivalentes:
- var foo = functionToCtor(Foo, "foo");
- var foo2 = new Foo("foo2");
Tanto foo como foo2:
- Tienen como prototipo Foo.prototype
- Tienen como constructor la función Foo
- Devuelven true a la expresión instanceof Foo
¿Y si una función constructora devuelve un valor?
Esa es buena… Pues en este caso cuando se use new el valor obtenido será el devuelto por la función constructora (siempre que este valor sea distinto de undefined):
- var Foo = function (name) {
- this.name = name || '';
- this.getUpperName = function () {
- return this.name.toUpperCase();
- }
- return { a: 10 };
- }
- var f = new Foo();
- f.a; // 10
- f instanceof Foo // false
Si… a pesar de que f ha sido creado usando new Foo, el prototipo de f ya no es Foo.prototype, ya que f vale el objeto devuelto por la función Foo. En este caso el objeto “original” cuyo prototipo es Foo.prototype y al que nos referíamos con this dentro de Foo se ha perdido. Lo mismo ocurre si usas el “polyfill” functionToCtor que hemos visto antes.
Funciones duales
Podemos hacer una función que sepa si se ha invocado con new o no? Pues sí:
- var Foo = function (name) {
- if (this instanceof Foo) {
- // Invocada con new
- }
- else {
- // Invocada sin new
- }
- }
Esto nos permite trabajar con this en el caso de que se haya usado new o bien no trabajar con this y devolver un valor en el caso de que no se haya invocado a la función con this. De todos modos si quieres que el usuario obtenga el mismo valor use new o no, lo más fácil es devolver el valor desde la función constructora. En este caso, eso sí, pierdes el hecho de que el prototipo del objeto sea la función constructora. Si quieres asegurarte que el usuario obtiene siempre un objeto con el prototipo asociado a la función constructora, es decir como si siempre usase new, puedes hacer:
- var Foo = function (name) {
- var _create = function (name) {
- this.name = name || '';
- this.getUpperName = function () {
- return this.name.toLocaleUpperCase();
- }
- }
- if (this instanceof Foo) {
- _create.bind(this)(name);
- }
- else {
- var o = Object.create(Foo.prototype);
- _create.bind(o)(name);
- return o;
- }
- }
La función _create es la que realmente rellena el objeto (le llega ya creado en la variable this). La función Foo lo único que hace es mirar si ha sido llamada con new o no. En el primer caso llama a _create pero vinculando el valor de this dentro de _create al propio valor de this dentro de Foo (que en este caso es el objeto que se está creando) y no devolver nada. En el caso que el usuario no haya usado new la función Foo crea un objeto vacío, le asigna Foo.prototype como prototipo y luego llama a _create vinculando el valor de this dentro de _create al del objeto que acaba de crear. Y finalmente, en este caso, devuelve el objeto creado.
Así, tanto si el usuario hace var x = Foo() como var x2 = new Foo() obtiene lo mismo: un objeto cuyo prototipo es Foo.prototype y que por lo tanto devuelve true a instanceof Foo.
Espero que este post os haya ayudado a entender como funciona new en JavaScript.
Saludos!