La Propiedad «prototype»
La propiedad «prototype» es extensamente utilizada por el núcleo de JavaScript. Todas las funciones constructoras incorporadas la emplean.
Primero examinaremos los detalles y luego veremos cómo agregar nuevas capacidades a los objetos incorporados.
Object.prototype
Supongamos que tenemos un objeto vacío y lo mostramos en pantalla:
let obj = {};
alert(obj); // "[object Object]" ?
¿De dónde proviene la cadena «[object Object]»? Es el método nativo toString
, pero ¿dónde se encuentra? ¡El objeto está vacío!
La notación abreviada obj = {}
es equivalente a obj = new Object()
, donde Object
es una función constructora incorporada con su propio prototype
, que apunta a un objeto con el método toString
y otros métodos.
¿Qué Ocurre con Object.prototype?
Cuando se llama a new Object()
(o se crea un objeto literal {...}
), el [[Prototype]] se asigna a Object.prototype
conforme a la regla discutida anteriormente. Por eso, cuando se invoca obj.toString()
, el método se obtiene de Object.prototype
.
Podemos verificarlo así:
let obj = {};
console.log(obj.__proto__ === Object.prototype); // true
console.log(obj.toString === obj.__proto__.toString); // true
console.log(obj.toString === Object.prototype.toString); // true
Tenga en cuenta que no hay más [[Prototype]] en la cadena después de Object.prototype
:
console.log(Object.prototype.__proto__); // null
Otros Prototipos Incorporados
Otros objetos integrados, como Array
, Date
y Function
, también almacenan sus métodos en sus respectivos prototipos.
Por ejemplo, cuando creamos un array [1, 2, 3]
, se utiliza internamente el constructor predeterminado new Array()
. Así, Array.prototype
se convierte en su prototipo y proporciona sus métodos. Esto es muy eficiente en términos de memoria.
Según la especificación, todos los prototipos incorporados tienen Object.prototype
en su parte superior. Por eso se dice que «todo hereda de los objetos».
Verificando Prototipos
Podemos verificar manualmente los prototipos:
let arr = [1, 2, 3];
// ¿hereda de Array.prototype?
console.log(arr.__proto__ === Array.prototype); // true
// ¿y después de Object.prototype?
console.log(arr.__proto__.__proto__ === Object.prototype); // true
// Y null en el tope.
console.log(arr.__proto__.__proto__.__proto__); // null
Algunos métodos en los prototipos pueden superponerse. Por ejemplo, Array.prototype
tiene su propia versión de toString
que enumera los elementos separados por comas:
let arr = [1, 2, 3];
console.log(arr); // 1,2,3 <-- el resultado de Array.prototype.toString
Herramientas del Navegador
Las herramientas en el navegador, como la consola de desarrollador de Chrome, también muestran la herencia (puede que necesite usar console.dir
para los objetos incorporados).
Funciones y sus Prototipos
Las funciones también son objetos creados por el constructor Function
incorporado, y sus métodos (call
, apply
, entre otros) provienen de Function.prototype
. Las funciones también tienen su propia versión de toString
.
function f() {}
console.log(f.__proto__ === Function.prototype); // true
console.log(f.__proto__.__proto__ === Object.prototype); // true, hereda de Object
Primitivos
Lo más complicado sucede con las cadenas, números y booleanos.
Como recordamos, no son objetos. Pero si tratamos de acceder a sus propiedades, se crean temporalmente objetos contenedores usando los constructores String
, Number
y Boolean
. Estos objetos proporcionan los métodos y luego desaparecen.
Estos objetos se crean de manera invisible para nosotros y la mayoría de los motores los optimizan, pero la especificación los describe exactamente de esta manera. Los métodos de estos objetos también residen en prototipos disponibles como String.prototype
, Number.prototype
y Boolean.prototype
.
Los valores null
y undefined
no tienen objetos contenedores ni prototipos asociados, por lo que no tienen métodos ni propiedades disponibles.
Modificando Prototipos Nativos
Los prototipos nativos pueden ser modificados. Por ejemplo, si añadimos un método a String.prototype
, estará disponible para todas las cadenas:
String.prototype.show = function() {
alert(this);
};
"BOOM!".show(); // BOOM!
Sin embargo, modificar prototipos nativos puede causar conflictos si diferentes bibliotecas intentan añadir el mismo método al mismo prototipo. Por lo tanto, modificar un prototipo nativo se considera generalmente una mala práctica.
Polyfills
En la programación moderna, el único caso en el que se permite modificar prototipos nativos es cuando implementamos un polyfill. Esto ocurre cuando un método existe en la especificación de JavaScript, pero aún no está soportado por algunos motores de JavaScript.
if (!String.prototype.repeat) { // si no hay tal método
String.prototype.repeat = function(n) {
return new Array(n + 1).join(this);
};
}
console.log("La".repeat(3)); // LaLaLa
Préstamo de Métodos
En el capítulo de Decoradores y Redirecciones, discutimos el préstamo de métodos. Esto ocurre cuando tomamos un método de un objeto y lo utilizamos en otro.
Por ejemplo, si estamos creando un objeto similar a un array, podemos querer copiar algunos métodos de Array
.
let obj = {
0: "Hola",
1: "mundo!",
length: 2,
};
obj.join = Array.prototype.join;
console.log(obj.join(',')); // Hola,mundo!
Esto funciona porque el método join
integrado solo se preocupa por los índices y la propiedad length
. No verifica si el objeto es realmente un array. Muchos métodos integrados funcionan de esta manera.
Otra opción es heredar estableciendo obj.__proto__
en Array.prototype
, de modo que todos los métodos de Array
estén disponibles automáticamente en obj
. Sin embargo, esto no es posible si obj
ya hereda de otro objeto, ya que solo se puede heredar de un objeto a la vez.
Resumen
Todos los objetos incorporados siguen el mismo patrón:
- Los métodos se almacenan en el prototipo (
Array.prototype
,Object.prototype
,Date.prototype
, etc.). - El objeto en sí solo almacena los datos (elementos del array, propiedades del objeto, la fecha).
- Los primitivos también almacenan métodos en prototipos de objetos contenedores:
Number.prototype
,String.prototype
yBoolean.prototype
. Soloundefined
ynull
no tienen objetos contenedores. - Los prototipos incorporados se pueden modificar o extender con nuevos métodos. Sin embargo, generalmente no se recomienda hacerlo, salvo para implementar un polyfill para un método estándar que aún no esté soportado por todos los motores de JavaScript.