Clases en TypeScript
Lectura previa: Clases (MDN) ↗
TypeScript ofrece soporte completo para la palabra clave class
introducida en ES2015.
Al igual que con otras características del lenguaje JavaScript, TypeScript agrega anotaciones de tipo y otras sintaxis para permitirte expresar relaciones entre clases y otros tipos.
Miembros de clases
Aquí está la clase más básica, una vacía:
Esta clase aún no es muy útil, así que comencemos a agregar algunos miembros.
Campos
Una declaración de campo crea una propiedad pública editable en una clase:
Al igual que con otras ubicaciones, la anotación de tipo es opcional, pero será implícitamente any
si no se especifica.
Los campos también pueden tener inicializadores; estos se ejecutarán automáticamente cuando se cree una instancia de la clase:
Al igual que con const
, let
y var
, el inicializador de una propiedad de clase se usará para inferir su tipo:
La opción --strictPropertyInitialization
La configuración strictPropertyInitialization
↗ controla si los campos de clase deben inicializarse en el constructor.
Ten en cuenta que el campo debe inicializarse en el propio constructor. TypeScript no analiza los métodos que invocas desde el constructor para detectar inicializaciones, porque una clase derivada podría sobrescribir esos métodos y no inicializar los miembros.
Si tienes la intención de inicializar definitivamente un campo a través de medios distintos al constructor (por ejemplo, tal vez una biblioteca externa esté completando parte de tu clase por ti), puedes usar el operador de aserción de asignación, !
:
El prefijo readonly
Los campos pueden tener el prefijo readonly
.
Esto evita asignaciones al campo fuera del constructor.
Constructores
Lectura previa: Constructor (MDN) ↗
Los constructores de clases son muy similares a las funciones. Puede agregar parámetros con anotaciones de tipo, valores predeterminados y sobrecargas:
Solo existen algunas diferencias entre las firmas de constructores de clases y las firmas de funciones:
- Los constructores no pueden tener parámetros de tipo; estos pertenecen a la declaración de clase externa, sobre la cual aprenderemos más adelante.
- Los constructores no pueden tener anotaciones de tipo de retorno: el tipo de instancia de clase siempre es lo que se devuelve
Llamadas a super
Al igual que en JavaScript, si tienes una clase base, necesitarás llamar a super();
en el cuerpo de tu constructor antes de usar cualquier miembro de this.
:
Olvidarse de llamar a super
es un error fácil de cometer en JavaScript, pero TypeScript te dirá cuándo es necesario.
Métodos
Lectura previa: Definiciones de métodos ↗
Una propiedad de función en una clase se llama método. Los métodos pueden usar todas las anotaciones del mismo tipo como funciones y constructores:
Aparte de las anotaciones de tipo estándar, TypeScript no agrega nada más nuevo a los métodos.
Ten en cuenta que dentro del cuerpo de un método, todavía es obligatorio acceder a los campos y otros métodos a través de this
.
Un nombre no calificado en el cuerpo de un método siempre se referirá a algo en el ámbito adjunto:
Getters / Setters
Las clases también pueden tener accesorios:
Ten en cuenta que un par get/set respaldado por campos sin lógica adicional rara vez es útil en JavaScript. Está bien exponer campos públicos si no necesitas agregar lógica adicional durante las operaciones get/set.
TypeScript tiene algunas reglas de inferencia especiales para los descriptores de acceso:
- Si
get
existe pero noset
, la propiedad es automáticamentereadonly
- Si no se especifica el tipo de parámetro de establecimiento (setter), se infiere del tipo de retorno del getter.
- Los getters y setters deben tener la misma visibilidad de miembros
Desde TypeScript 4.3 ↗, es posible tener descriptores de acceso con diferentes tipos para obtener y asignar.
Firmas de índices
Las clases pueden declarar firmas de índice; estos funcionan igual que Firmas de índice para otros tipos de objetos:
Debido a que el tipo de firma de índice también debe capturar los tipos de métodos, no es fácil utilizar estos tipos de manera útil. Generalmente es mejor almacenar los datos indexados en otro lugar en lugar de en la instancia de la clase misma.
Herencia de clase
Al igual que otros lenguajes con características orientadas a objetos, las clases en JavaScript pueden heredar de las clases base.
Cláusulas implements
Puedes usar una cláusula de implements
para verificar que una clase satisface una interface
particular.
Se emitirá un error si una clase no logra implementarla correctamente:
Las clases también pueden implementar múltiples interfaces, p. class C implements A, B {
.
Precauciones
Es importante entender que una cláusula implements
es solo una verificación de que la clase puede ser tratada como el tipo de interfaz.
No cambia el tipo de clase ni sus métodos en absoluto.
Una fuente común de error es asumir que una cláusula implements
cambiará el tipo de clase - ¡no es así!
En este ejemplo, quizás esperábamos que el tipo de s
estuviera influenciado por el parámetro name: string
de check
.
No lo es: las cláusulas implements
no cambian la forma en que se verifica el cuerpo de la clase ni se infiere su tipo.
De manera similar, implementar una interfaz con una propiedad opcional no crea esa propiedad:
Cláusulas extends
Lectura previa: la palabra clave extends (MDN) ↗
Las clases pueden “extenderse” desde una clase base. Una clase derivada tiene todas las propiedades y métodos de su clase base y también puede definir miembros adicionales.
Sobrescribiendo métodos
Lectura previa: palabra clave super (MDN) ↗
Una clase derivada también puede sobrescribir un campo o propiedad de una clase base.
Puedes utilizar la sintaxis super.
para acceder a los métodos de la clase base.
Ten en cuenta que debido a que las clases de JavaScript son un objeto de búsqueda simple, no existe la noción de un “super campo”.
TypeScript exige que una clase derivada sea siempre un subtipo de su clase base.
Por ejemplo, aquí tienes una forma legal de sobrescribir un método:
Es importante que una clase derivada siga su contrato de clase base. Recuerda que es muy común (¡y siempre legal!) hacer referencia a una instancia de clase derivada a través de una referencia de clase base:
¿Qué pasaría si Derived
no siguiera el contrato de Base
?
Si compilamos este código a pesar del error, este ejemplo fallaría:
Declaraciones de campos type-only
Cuando target >= ES2022
o useDefineForClassFields
↗ es true
, los campos de clase se inicializan después de que el constructor de la clase madre se completa, sobrescribiendo cualquier valor establecido por la clase madre. Esto puede ser un problema cuando sólo deseas volver a declarar un tipo más preciso para un campo heredado. Para manejar estos casos, puedes escribir declare
para indicar a TypeScript que no debería haber ningún efecto de tiempo de ejecución para esta declaración de campo.
Orden de inicialización
El orden en que se inicializan las clases de JavaScript puede resultar sorprendente en algunos casos. Consideremos este código:
¿Qué pasó aquí?
El orden de inicialización de clases, tal como lo define JavaScript, es:
- Los campos de la clase base están inicializados.
- Se ejecuta el constructor de la clase base.
- Los campos de clase derivada se inicializan.
- Se ejecuta el constructor de la clase derivada.
Esto significa que el constructor de la clase base vio su propio valor para name
durante su propio constructor, porque las inicializaciones del campo de la clase derivada aún no se habían ejecutado.
Heredar tipos integrados
Nota: Si no planeas heredar de tipos integrados como
Array
,Error
,Map
, etc. o si tu objetivo de compilación está configurado explícitamente enES6
/ES2015
o superior, puedes saltarte esta sección
En ES2015, los constructores que devuelven un objeto sustituyen implícitamente el valor de this
por cualquier llamador de super(...)
.
Es necesario que el código constructor generado capture cualquier valor de retorno potencial de super(...)
y lo reemplace con this
.
Como resultado, es posible que las subclasificaciones Error
, Array
y otras ya no funcionen como se esperaba.
Esto se debe al hecho de que las funciones constructoras para Error
, Array
y similares usan new.target
de ECMAScript 6 para ajustar la cadena del prototipo; sin embargo, no hay forma de garantizar un valor para new.target
al invocar un constructor en ECMAScript 5.
Otros compiladores de nivel inferior generalmente tienen la misma limitación de forma predeterminada.
Para una subclase como la siguiente:
es posible que encuentres que:
- Los métodos pueden ser
undefined
en los objetos devueltos al construir estas subclases, por lo que llamar asayHello
resultará en un error. instanceof
se dividirá entre las instancias de la subclase y sus instancias, por lo que(new MsgError()) instanceof MsgError
devolveráfalse
.
Como recomendación, puedes ajustar manualmente el prototipo inmediatamente después de cualquier llamada super(...)
.
Sin embargo, cualquier subclase de MsgError
también tendrá que configurar manualmente el prototipo.
Para runtimes que no admiten Object.setPrototypeOf
↗, es posible que puedas utilizar __proto__
↗.
Desafortunadamente, [estas soluciones no funcionarán en Internet Explorer 10 y versiones anteriores ↗](https://msdn.microsoft.com/en-us/library/s4esdbwz(v=vs.94) .aspx).
Se pueden copiar manualmente métodos del prototipo a la instancia misma (es decir, MsgError.prototype
a this
), pero la cadena del prototipo en sí no se puede arreglar.
Visibilidad de miembros
Puedes usar TypeScript para controlar si ciertos métodos o propiedades son visibles para el código fuera de la clase.
public
La visibilidad predeterminada de los miembros de la clase es public
.
Se puede acceder a un miembro public
desde cualquier lugar:
Debido a que public
ya es el modificador de visibilidad predeterminado, nunca necesitas escribirlo en un miembro de la clase, pero puedes elegir hacerlo por razones de estilo/legibilidad.
protected
Los miembros protected
solo son visibles para las subclases de la clase en la que están declarados.
Exposición de miembros protected
Las clases derivadas deben seguir sus contratos de clase base, pero pueden optar por exponer un subtipo de clase base con más capacidades.
Esto incluye hacer que los miembros protected
sean public
:
Ten en cuenta que Derived
ya podía leer y escribir libremente m
, por lo que esto no altera significativamente la “seguridad” de esta situación.
Lo principal a tener en cuenta aquí es que en la clase derivada, debemos tener cuidado de repetir el modificador protected
si esta exposición no es intencional.
Acceso protected
entre jerarquías
Diferentes lenguajes de programación orientada a objetos no están de acuerdo sobre si es legal acceder a un miembro protected
a través de una referencia de clase base:
Java, por ejemplo, considera que esto es legal. Por otro lado, C# y C++ optaron por que este código fuera ilegal.
TypeScript está del lado de C# y C++ aquí, porque acceder a x
en Derived2
solo debería ser legal desde las subclases de Derived2
, y Derived1
no es una de ellas.
Además, si acceder a x
a través de una referencia Derived1
es ilegal (¡y ciertamente debería serlo!), entonces acceder a través de una referencia de clase base nunca debería mejorar la situación.
Consulta también ¿Por qué no puedo acceder a un miembro protegido desde una clase derivada? ↗ que explica más del razonamiento de C#.
private
private
es como protected
, pero no permite el acceso al miembro ni siquiera desde subclases:
Debido a que los miembros private
no son visibles para las clases derivadas, una clase derivada no puede aumentar su visibilidad:
Acceso private
entre instancias
Diferentes lenguajes de programación orientada a objetos no están de acuerdo sobre si diferentes instancias de la misma clase pueden acceder a los miembros private
de cada uno.
Mientras que lenguajes como Java, C#, C++, Swift y PHP lo permiten, Ruby no.
TypeScript permite el acceso private
entre instancias:
Advertencias
Al igual que otros aspectos del sistema de tipos de TypeScript, private
y protected
solo se aplican durante la verificación de tipos ↗.
Esto significa que las construcciones de tiempo de ejecución de JavaScript como in
o la búsqueda de propiedad simple aún pueden acceder a un miembro private
o protected
:
private
también permite el acceso usando notación entre corchetes durante la verificación de tipos. Esto hace que los campos declarados private
sean potencialmente más fáciles de acceder para cosas como tests unitarios, con el inconveniente de que estos campos son ligeramente privados y no imponen estrictamente la privacidad.
A diferencia del private
de TypeScripts, los campos privados de JavaScript ↗ (#
) permanecen privados después de la compilación y no proporcionan las trampillas de escape mencionadas anteriormente, como el acceso a la notación entre corchetes, lo que los hace duros y privados.
Al compilar en ES2021 o menor, TypeScript usará WeakMaps en lugar de #
.
Si necesitas proteger los valores de tu clase de actores maliciosos, debes usar mecanismos que ofrezcan privacidad estricta en tiempo de ejecución, como closures, WeakMaps o campos privados. Ten en cuenta que estas comprobaciones de privacidad adicionales durante el tiempo de ejecución podrían afectar el rendimiento.
Miembros estáticos
Lectura previa: Miembros estáticos (MDN) ↗
Las clases pueden tener miembros static
.
Estos miembros no están asociados con una instancia particular de la clase.
Se puede acceder a ellos a través del propio objeto constructor de clase:
Los miembros estáticos también pueden usar los mismos modificadores de visibilidad public
, protected
y private
:
Los miembros estáticos también se heredan:
Nombres estáticos especiales
Generalmente no es seguro ni posible sobrescribir propiedades del prototipo Function
.
Debido a que las clases son en sí mismas funciones que se pueden invocar con new
, ciertos nombres static
no se pueden usar.
Las propiedades de funciones como name
, length
y call
no son válidas para definirse como miembros static
:
¿Por qué no hay clases estáticas?
TypeScript (y JavaScript) no tienen una construcción llamada static class
de la misma manera que, por ejemplo, C#.
Esas construcciones sólo existen porque esos lenguajes obligan a que todos los datos y funciones estén dentro de una clase; como esa restricción no existe en TypeScript, no es necesaria. Una clase con una sola instancia normalmente se representa simplemente como un objeto normal en JavaScript/TypeScript.
Por ejemplo, no necesitamos una sintaxis de “clase estática” en TypeScript porque un objeto normal (o incluso una función de nivel superior) hará el trabajo igual de bien:
Bloques static
en clases
Los bloques estáticos te permiten escribir una secuencia de declaraciones con su propio alcance que pueden acceder a campos privados dentro de la clase que los contiene. Esto significa que podemos escribir código de inicialización con todas las capacidades de escribir declaraciones, sin fugas de variables y con acceso completo a las partes internas de nuestra clase.
Clases genéricas
Las clases, al igual que las interfaces, pueden ser genéricas.
Cuando se crea una instancia de una clase genérica con new
, sus parámetros de tipo se infieren de la misma manera que en una llamada de función:
Las clases pueden usar restricciones genéricas y valores predeterminados de la misma manera que las interfaces.
Parámetros de tipo en miembros estáticos
Este código no es legal y puede que no sea obvio por qué:
¡Recuerda que los tipos siempre se borran por completo!
En tiempo de ejecución, solo hay un espacio de propiedad Box.defaultValue
.
Esto significa que configurar Box<string>.defaultValue
(si fuera posible) también cambiaría Box<number>.defaultValue
- no es bueno.
Los miembros static
de una clase genérica nunca pueden hacer referencia a los parámetros de tipo de la clase.
this
en clases en tiempo de ejecución
Lectura previa:palabra clave this (MDN) ↗
Es importante recordar que TypeScript no cambia el comportamiento de ejecución de JavaScript, y que JavaScript es algo famoso por tener algunos comportamientos de ejecución peculiares.
El manejo de this
por parte de JavaScript es realmente inusual:
En pocas palabras, de forma predeterminada, el valor de this
dentro de una función depende de cómo se llamó a la función.
En este ejemplo, debido a que la función fue llamada a través de la referencia obj
, su valor de this
era obj
en lugar de la instancia de clase.
¡Esto rara vez es lo que quieres que suceda! TypeScript proporciona algunas formas de mitigar o prevenir este tipo de error.
Funciones de flecha
Lectura previa: Funciones de flecha (MDN) ↗
Si tienes una función que a menudo se llama de una manera que pierde su contexto this
, puede tener sentido usar una propiedad de función de flecha en lugar de una definición de método:
Esto tiene algunas compensaciones:
- Se garantiza que el valor
this
será correcto en tiempo de ejecución, incluso para el código que no se verifica con TypeScript. - Esto utilizará más memoria, porque cada instancia de clase tendrá su propia copia de cada función definida de esta manera.
- No puedes usar
super.getName
en una clase derivada, porque no hay ninguna entrada en la cadena del prototipo para recuperar el método de la clase base.
Parámetros this
En la definición de un método o función, un parámetro inicial llamado this
tiene un significado especial en TypeScript.
Estos parámetros se borran durante la compilación:
TypeScript verifica que llamar a una función con un parámetro this
se haga con un contexto correcto.
En lugar de usar una función de flecha, podemos agregar un parámetro this
a las definiciones de métodos para hacer cumplir estáticamente que el método se llame correctamente:
Este método hace las compensaciones opuestas al enfoque de la función de flecha:
- Los callers de JavaScript aún pueden usar el método de clase incorrectamente sin darse cuenta.
- Sólo se asigna una función por definición de clase, en lugar de una por instancia de clase.
- Las definiciones de métodos base aún se pueden llamar a través de
super
.
Tipos this
En las clases, un tipo especial llamado this
se refiere dinámicamente al tipo de la clase actual.
Veamos cómo es útil esto:
Aquí, TypeScript infirió que el tipo de retorno de set
era this
, en lugar de Box
.
Ahora hagamos una subclase de Box
:
También puedes usar this
en una anotación de tipo de parámetro:
Esto es diferente a escribir other: Box
: si tienes una clase derivada, su método sameAs
ahora solo aceptará otras instancias de esa misma clase derivada:
Protecciones de tipo basadas en this
Puedes usar this is Type
en la posición de retorno para métodos en clases e interfaces.
Cuando se mezcla con un tipo de estrechamiento (por ejemplo, declaraciones “if”), el tipo del objeto de destino se limitará al Type
especificado.
Un caso de uso común para una protección de tipos basada en esto es permitir la validación diferida (lazy) de un campo en particular. Por ejemplo, este caso elimina un undefined
del valor contenido dentro del cuadro cuando se ha verificado que hasValue
es verdadero:
Propiedades de parámetros
TypeScript ofrece una sintaxis especial para convertir un parámetro de constructor en una propiedad de clase con el mismo nombre y valor.
Estas se denominan propiedades de parámetros y se crean anteponiendo un argumento de constructor con uno de los modificadores de visibilidad public
, private
, pretected
o readonly
.
El campo resultante obtiene esos modificadores:
Expresiones de clase
Lectura previa: Expresiones de clase (MDN) ↗
Las expresiones de clase son muy similares a las declaraciones de clase. La única diferencia real es que las expresiones de clase no necesitan un nombre, aunque podemos referirnos a ellas mediante cualquier identificador al que terminaron vinculadas:
Firmas de constructor
Las clases de JavaScript se instancian con el operador new
. Dado el tipo de una clase en sí, el tipo de utilidad InstanceType ↗ modela esta operación.
Clases y miembros abstract
Las clases, métodos y campos en TypeScript pueden ser abstractos.
Un método abstracto o campo abstracto es aquel al que no se le ha proporcionado una implementación. Estos miembros deben existir dentro de una clase abstracta, de la que no se puede crear una instancia directamente.
La función de las clases abstractas es servir como clase base para las subclases que implementan todos los miembros abstractos. Cuando una clase no tiene miembros abstractos, se dice que es concreta.
Veamos un ejemplo:
No podemos crear una instancia de Base
con new
porque es abstracta.
En lugar de ello, necesitamos crear una clase derivada e implementar los miembros abstractos:
Observa que si nos olvidamos de implementar los miembros abstractos de la clase base, obtendremos un error:
Firmas de construcciones abstractas
A veces quierrás aceptar alguna función constructora de clase que produce una instancia de una clase que deriva de alguna clase abstracta.
Por ejemplo, es posible que desees escribir este código:
TypeScript te dice correctamente que estás intentando crear una instancia de una clase abstracta.
Después de todo, dada la definición de greet
, es perfectamente legal escribir este código, que terminaría construyendo una clase abstracta:
En lugar de eso, quierrás escribir una función que acepte algo con una firma de constructor:
Ahora TypeScript te informa correctamente qué funciones constructoras de clase se pueden invocar: Derived
puede porque es concreta, pero Base
no.
Relaciones Entre Clases
En la mayoría de los casos, las clases en TypeScript se comparan estructuralmente, al igual que otros tipos.
Por ejemplo, estas dos clases se pueden usar en lugar de la otra porque son idénticas:
De manera similar, las relaciones de subtipo entre clases existen incluso si no hay una herencia explícita:
Esto suena sencillo, pero hay algunos casos que parecen más extraños que otros.
Las clases vacías no tienen miembros. En un sistema de tipos estructurales, un tipo sin miembros es generalmente un supertipo de cualquier otra cosa. Entonces, si escribes una clase vacía (¡no lo hagas!), se puede usar cualquier cosa en su lugar: