Tipos cotidianos de TypeScript
En este capítulo, cubriremos algunos de los tipos de valores más comunes que encontrarás en el código JavaScript y explicaremos las formas correspondientes de describir esos tipos en TypeScript. Esta no es una lista exhaustiva y los capítulos futuros describirán más formas de nombrar y utilizar otros tipos.
Los tipos también pueden aparecer en muchos más lugares que solo anotaciones de tipo. A medida que aprendamos sobre los tipos en sí, también aprenderemos sobre los lugares donde podemos referirnos a estos tipos para formar nuevas construcciones.
Comenzaremos revisando los tipos más básicos y comunes que puedes encontrar al escribir código JavaScript o TypeScript. Estos formarán más adelante los componentes básicos de tipos más complejos.
Los tipos primitivos: string
, number
y boolean
JavaScript tiene tres primitivas ↗: string
, number
y boolean
.
Cada uno tiene un tipo correspondiente en TypeScript.
Como es de esperar, estos son los mismos nombres que verías si usaras el operador typeof
de JavaScript en un valor de esos tipos:
string
representa valores de cadena como"Hello, world"
number
es para números como42
. JavaScript no tiene un valor de tiempo de ejecución especial para números enteros, por lo que no existe un equivalente aint
ofloat
: todo es simplementenumber
.boolean
es para los dos valorestrue
yfalse
.
Los nombres de tipo
String
,Number
yBoolean
(que comienzan con letras mayúsculas) son legales, pero se refieren a algunos tipos integrados especiales que muy raramente aparecerán en tu código. Siempre utilizastring
,number
oboolean
para los tipos.
Tipos para el contenido de Arrays
Para especificar el tipo de un array como [1, 2, 3]
, puedes usar la sintaxis number[]
; esta sintaxis funciona para cualquier tipo (por ejemplo, string[]
es un array de cadenas, etc.).
También puedes ver esto escrito como Array<number>
, que significa lo mismo.
Aprenderemos más sobre la sintaxis T<U>
cuando cubramos generics.
Ten en cuenta que
[number]
es una cosa diferente; consulta la sección sobre Tuplas.
El tipo especial any
TypeScript también tiene un tipo especial, any
, que puedes usar siempre que no quieras que un valor en particular cause errores de verificación de tipo.
Cuando un valor es de tipo any
, puedes acceder a cualquier propiedad del mismo (que a su vez será de tipo any
), llamarlo como una función, asignarlo a (o desde) un valor de cualquier tipo, o prácticamente cualquier otra cosa que sea sintácticamente legal:
El tipo any
es útil cuando no quieres escribir un tipo largo solo para convencer a TypeScript de que una línea de código en particular está bien.
El indicador noImplicitAny
Cuando no especificas un tipo y TypeScript no puede inferirlo del contexto, el compilador generalmente usará de manera predeterminada any
.
Normalmente querrás evitar esto porque any
no está tipado.
Utiliza el indicador del compilador noImplicitAny
↗ para marcar cualquier any
implícito como un error.
Anotaciones de tipado en variables
Cuando declaras una variable usando const
, var
o let
, opcionalmente puedes agregar una anotación de tipo para especificar explícitamente el tipo de la variable:
TypeScript no usa declaraciones de estilo “tipos a la izquierda” como
int x = 0;
Las anotaciones de tipo siempre irán después de lo que está siendo tipado.
En la mayoría de los casos, sin embargo, esto no es necesario. Siempre que es posible, TypeScript intenta inferir automáticamente los tipos en tu código. Por ejemplo, el tipo de una variable se infiere en función del tipo de su inicializador:
En su mayor parte, no necesitas aprender explícitamente las reglas de inferencia. Si estás empezando, intenta utilizar menos anotaciones de tipo de las que crees que necesitas; te sorprenderá saber cuántas necesitas para que TypeScript comprenda completamente lo que está sucediendo.
Tipos en Funciones
Las funciones son el medio principal para pasar datos en JavaScript. TypeScript te permite especificar los tipos de valores de entrada y salida de funciones.
Anotaciones de tipo de parámetros
Cuando declaras una función, puedes agregar anotaciones de tipo después de cada parámetro para declarar qué tipos de parámetros acepta la función. Las anotaciones de tipo de parámetro van después del nombre del parámetro:
Cuando un parámetro tiene una anotación de tipo, se verificarán los argumentos de esa función:
Incluso si no tienes anotaciones de tipo en tus parámetros, TypeScript aún verificará que hayas pasado la cantidad correcta de argumentos.
Anotaciones de tipo de retorno
También puedes agregar anotaciones de tipo de retorno. Las anotaciones de tipo de retorno aparecen después de la lista de parámetros:
Al igual que las anotaciones de tipo variable, generalmente no necesitas una anotación de tipo de retorno porque TypeScript inferirá el tipo de retorno de la función en función de sus declaraciones return
.
La anotación de tipo en el ejemplo anterior no cambia nada.
Algunas bases de código especificarán explícitamente un tipo de devolución con fines de documentación, para evitar cambios accidentales o simplemente por preferencia personal.
Funciones que devuelven Promises
Si quieres anotar el tipo de retorno de una función que devuelve una promesa, debes usar el tipo Promise
:
Funciones Anónimas
Las funciones anónimas son un poco diferentes de las declaraciones de funciones. Cuando una función aparece en un lugar donde TypeScript puede determinar cómo se llamará, los parámetros de esa función reciben tipos automáticamente.
Aquí tienes un ejemplo:
Aunque el parámetro s
no tenía una anotación de tipo, TypeScript usó los tipos de la función forEach
, junto con el tipo inferido del array, para determinar el tipo que s
tendrá.
Este proceso se llama tipificación contextual porque el contexto en el que ocurrió la función informa qué tipo debe tener.
Similar a las reglas de inferencia, no necesitas aprender explícitamente cómo sucede esto, pero comprender que sucede puede ayudarte a darte cuenta cuando las anotaciones de tipo no son necesarias. Más adelante veremos más ejemplos de cómo el contexto en el que ocurre un valor puede afectar su tipo.
Tipos de Objetos
Aparte de las primitivas, el suerte de tipo más común que encontrarás es un object type. Esto se refiere a cualquier valor de JavaScript con propiedades, ¡que son casi todas! Para definir un tipo de objeto, simplemente enumeramos sus propiedades y sus tipos.
Por ejemplo, aquí tienes una función que toma un objeto puntual:
Aquí, anotamos el parámetro con un tipo con dos propiedades, x
e y
, que son ambas del tipo number
.
Puedes usar ,
o ;
para separar las propiedades, y el último separador es opcional de cualquier manera.
La parte del tipo de cada propiedad también es opcional.
Si no especificas un tipo, se asumirá que es any
.
Propiedades opcionales
Los tipos de objetos también pueden especificar que algunas o todas sus propiedades sean opcionales.
Para hacer esto, agrega un ?
después del nombre de la propiedad:
En JavaScript, si accedes a una propiedad que no existe, obtendrás el valor undefined
en lugar de un error de tiempo de ejecución.
Debido a esto, cuando leas una propiedad opcional, tendrás que verificar si es undefined
antes de usarla.
Tipos de uniones
El sistema de tipos de TypeScript te permite crear nuevos tipos a partir de los existentes utilizando una gran variedad de operadores. Ahora que sabemos cómo escribir algunos tipos, es hora de comenzar a combinarlos de maneras interesantes.
Definiendo un tipo de unión
La primera forma de combinar tipos que puedes ver es un tipo union. Un tipo de unión es un tipo formado a partir de dos o más tipos diferentes, que representan valores que pueden ser cualquiera de esos tipos. Nos referimos a cada uno de estos tipos como los miembros de la unión.
Escribamos una función que pueda operar con cadenas o números:
Trabajar con tipos de uniones
Es fácil proporcionar un valor que coincida con un tipo de unión; simplemente proporciona un tipo que coincida con cualquiera de los miembros de la unión. Si tienes un valor de tipo unión, ¿cómo trabajas con él?
TypeScript solo permitirá una operación si es válida para todos los miembros de la unión.
Por ejemplo, si tienes la unión string | number
, no puedes usar métodos que solo están disponibles en string
:
La solución es estrechar (narrow) la unión con código, igual que lo harías en JavaScript sin anotaciones de tipo. El estrechamiento ocurre cuando TypeScript puede deducir un tipo más específico para un valor basado en la estructura del código.
Por ejemplo, TypeScript sabe que solo un valor string
tendrá un valor typeof
igual a "string"
:
Otro ejemplo es usar una función como Array.isArray
:
Observa que en la rama else
, no necesitamos hacer nada especial; si x
no era un string[]
, entonces debe haber sido un string
.
A veces tendrás una unión donde todos los miembros tienen algo en común.
Por ejemplo, tanto los arrays como las cadenas tienen un método slice
.
Si cada miembro de una unión tiene una propiedad en común, puedes usar esa propiedad sin estrechar:
Puede resultar confuso que una unión de tipos parezca tener la intersección de las propiedades de esos tipos. Esto no es un accidente: el nombre unión proviene de la teoría de tipos. La unión
number | string
se compone tomando la unión de los valores de cada tipo. Observa que dados dos conjuntos con hechos correspondientes sobre cada conjunto, sólo la intersección de esos hechos se aplica a la unión de los propios conjuntos. Por ejemplo, si tuviéramos una sala de personas altas con sombreros y otra sala de hispanohablantes con sombreros, después de combinar esas salas, lo único que sabemos sobre cada persona es que debe llevar sombrero.
Alias de tipos
Hemos estado usando tipos de objetos y tipos de unión escribiéndolos directamente en anotaciones de tipo. Esto es conveniente, pero es común querer usar el mismo tipo más de una vez y referirse a él con un solo nombre.
Un alias de tipo es exactamente eso: un nombre para cualquier tipo. La sintaxis para un alias de tipo es:
Puedes usar un alias de tipo para darle un nombre a cualquier tipo, no solo a un tipo de objeto. Por ejemplo, un alias de tipo puede nombrar un tipo de unión:
Ten en cuenta que los alias son solo alias: no puedes usar alias de tipo para crear “versiones” diferentes/distintas del mismo tipo. Cuando usas el alias, es exactamente como si hubiera escrito el tipo que representa el alias. En otras palabras, este código puede parecer ilegal, pero está bien según TypeScript porque ambos tipos son alias para el mismo tipo:
Interfaces
Una declaración de interfaz es otra forma de nombrar un tipo de objeto:
Al igual que cuando usamos un alias de tipo arriba, el ejemplo funciona como si hubiéramos usado un tipo de objeto anónimo.
TypeScript solo se preocupa por la estructura del valor que pasamos a printCoord
; solo le importa que tenga las propiedades esperadas.
Preocuparnos únicamente por la estructura y capacidades de los tipos es la razón por la que llamamos a TypeScript un sistema de tipos estructuralmente tipificado.
Diferencias entre alias de tipo e interfaces
Los alias de tipo y las interfaces son muy similares y en muchos casos puedes elegir entre ellos libremente.
Casi todas las características de una interface
están disponibles en type
, la distinción clave es que un tipo no se puede volver a abrir para agregar nuevas propiedades frente a una interfaz que siempre es extensible.
Interface | Type |
---|---|
Ampliar una interfaz
| Extender un tipo mediante intersecciones
|
Agregar nuevos campos a una interfaz existente
| Un tipo no se puede cambiar después de haber sido creado
|
Aprenderás más sobre estos conceptos en capítulos posteriores, así que no te preocupes si no los entiendes todos de inmediato.
- Antes de la versión 4.2 de TypeScript, al escribir nombres de alias pueden aparecer mensajes de error ↗, a veces en lugar del tipo anónimo equivalente (que puede ser deseable o no). Las interfaces siempre se nombrarán en los mensajes de error.
- Los alias de tipo no pueden participar en la fusión de declaraciones, pero las interfaces sí ↗.
- Las interfaces solo pueden usarse para declarar las formas de los objetos, no cambiar el nombre de las primitivas ↗.
- Los nombres de las interfaces siempre aparecerán en su forma original ↗ en mensajes de error, pero sólo cuando se utilizan por nombre.
En su mayor parte, puedes elegir según tus preferencias personales, y TypeScript te dirá si necesita algo para ser el otro tipo de declaración. Si deseas una heurística, usa interface
hasta que necesites usar características de type
.
Aserciones de Tipos
A veces tendrás información sobre el tipo de valor que TypeScript no puede conocer.
Por ejemplo, si estás usando document.getElementById
, TypeScript solo sabe que esto devolverá algún tipo de HTMLElement
, pero es posible que sepas que tu página siempre tendrá un HTMLCanvasElement
con un ID determinado.
En esta situación, puedes usar una aserción de tipo para especificar un tipo más específico:
Al igual que una anotación de tipo, el compilador elimina las aserciones de tipo y no afectarán el comportamiento de ejecución de tu código.
También puedes usar la sintaxis de corchetes angulares (excepto si el código está en un archivo .tsx
), que es equivalente:
Recordatorio: Debido a que las aserciones de tipo se eliminan en tiempo de compilación, no hay verificación en tiempo de ejecución asociada con una aserción de tipo. No se generará una excepción o
null
si la afirmación del tipo es incorrecta.
TypeScript solo permite afirmaciones de tipo que se convierten a una versión más específica o menos específica de un tipo. Esta regla previene coacciones “imposibles” como:
A veces esta regla puede ser demasiado conservadora y no permitirá coacciones más complejas que podrían ser válidas.
Si esto sucede, puedes usar dos aserciones, primero para any
(o unknow
, que presentaremos más adelante), luego para el tipo deseado:
Tipos literales
Además de los tipos generales string
y number
, podemos referirnos a cadenas y números específicos en posiciones de tipo.
Una forma de pensar en esto es considerar cómo JavaScript viene con diferentes formas de declarar una variable. Tanto var
como let
permiten cambiar lo que se contiene dentro de la variable, y const
no. Esto se refleja en cómo TypeScript crea tipos para literales.
Por sí solos, los tipos literales no son muy valiosos:
¡No sirve de mucho tener una variable que solo puede tener un valor!
Pero al combinar literales en uniones, puedes expresar un concepto mucho más útil - por ejemplo, funciones que solo aceptan un cierto conjunto de valores conocidos:
Los tipos literales numéricos funcionan de la misma manera:
Por supuesto, puedes combinarlos con tipos no literales:
Hay un tipo más de literal: los literales booleanos.
Solo hay dos tipos de literales booleanos y, como puedes imaginar, son los tipos true
y false
.
El tipo boolean
en sí mismo es en realidad solo un alias para la unión true | false
.
Inferencia literal
Cuando inicializas una variable con un objeto, TypeScript asume que las propiedades de ese objeto podrían cambiar los valores más adelante. Por ejemplo, si escribiste un código como este:
TypeScript no asume que la asignación de 1
a un campo que anteriormente tenía 0
sea un error.
Otra forma de decir esto es que obj.counter
debe tener el tipo number
, no 0
, porque los tipos se utilizan para determinar el comportamiento de lectura y escritura.
Lo mismo se aplica a las cadenas:
En el ejemplo anterior, se infiere que req.method
es string
, no "GET"
. Debido a que el código se puede evaluar entre la creación de req
y la llamada de handleRequest
, que podría asignar una nueva cadena como "GUESS"
a req.method
, TypeScript considera que este código tiene un error.
Hay dos formas de solucionar este problema.
- Puedes cambiar la inferencia agregando una aserción de tipo en cualquier ubicación: Prueba este código ↗
El cambio 1 significa “Tengo la intención de que req.method
siempre tenga el tipo literal "GET"
”, evitando la posible asignación de "GUESS"
a ese campo posteriormente.
El cambio 2 significa “Sé por otras razones que req.method
tiene el valor "GET"
”.
- Puedes usar
as const
para convertir todo el objeto en literales de tipo: Prueba este código ↗
El sufijo as const
actúa como const
pero para el sistema de tipos, asegurando que a todas las propiedades se les asigne el tipo literal en lugar de una versión más general como string
o number
.
Los valores null
y undefined
JavaScript tiene dos valores primitivos que se usan para señalar un valor ausente o no inicializado: null
y undefined
.
TypeScript tiene dos tipos correspondientes con los mismos nombres. El comportamiento de estos tipos depende de si tienes activada la opción strictNullChecks
↗.
Con la opción strictNullChecks
en off
Con strictNullChecks
↗ en off, los valores que podrían ser null
o undefined
aún pueden se puede acceder normalmente, y los valores null
y undefined
se pueden asignar a una propiedad de cualquier tipo.
Esto es similar a cómo se comportan los lenguajes sin comprobaciones de nulos (por ejemplo, C#, Java).
La falta de verificación de estos valores tiende a ser una fuente importante de errores; Siempre recomendamos que las personas activen strictNullChecks
↗ si es práctico hacerlo en tu código base.
Con la opción strictNullChecks
en on
Con strictNullChecks
↗ en on, cuando un valor es null
o undefined
, tendrás que probar esos valores antes de usar métodos o propiedades en ese valor.
Al igual que verificar por undefined
antes de usar una propiedad opcional, podemos usar estrechamiento para verificar valores que podrían ser null
:
Operador de aserción no nulo (Postfix !
)
TypeScript también tiene una sintaxis especial para eliminar null
y undefined
de un tipo sin realizar ninguna verificación explícita.
Escribir !
después de cualquier expresión es efectivamente una afirmación de tipo de que el valor no es null
o undefined
:
Al igual que otras afirmaciones de tipo, esto no cambia el comportamiento de ejecución de tu código, por lo que es importante usar !
solo cuando sepas que el valor no puede ser null
o undefined
.
Enums
Las enumeraciones son una característica agregada a JavaScript por TypeScript que permite describir un valor que podría ser uno de un conjunto de posibles constantes con nombre. A diferencia de la mayoría de las funciones de TypeScript, esto no es una adición de nivel de tipo a JavaScript, sino algo agregado al lenguaje y al tiempo de ejecución. Debido a esto, es una característica que debes saber que existe, pero tal vez no la uses a menos que estés seguro. Puedes leer más sobre enumeraciones en la Página de referencia de enumeraciones ↗.
Primitivas menos comunes
Vale la pena mencionar el resto de primitivas en JavaScript que están representadas en el sistema de tipos. Aunque aquí no entraremos en profundidad.
La primitiva bigint
Desde ES2020 en adelante, hay una primitiva en JavaScript que se usa para enteros muy grandes, BigInt
:
Puedes obtener más información sobre BigInt en las notas de la versión de TypeScript 3.2 ↗.
La primitiva symbol
Hay una primitiva en JavaScript que se usa para crear una referencia global única a través de la función Symbol()
:
Puedes obtener más información sobre ellos en la Página de referencia de símbolos ↗.