Generics
Generics permitem que você crie componentes e funções reutilizáveis que podem trabalhar com múltiplos tipos. Com generics, você pode parametrizar tipos, funções e interfaces, permitindo que operem em diferentes tipos sem especificá-los explicitamente de antemão.
Generics permitem que você torne o código mais flexível e reutilizável.
Tipo Genérico
Para definir um tipo genérico, você usa colchetes angulares (<>) para especificar os parâmetros de tipo, por exemplo:
function identity<T>(arg: T): T { return arg;}const a = identity('x');const b = identity(123);
const getLen = <T,>(data: ReadonlyArray<T>) => data.length;const len = getLen([1, 2, 3]);Classes Genéricas
Generics também podem ser aplicados a classes, desta forma elas podem trabalhar com múltiplos tipos usando parâmetros de tipo. Isso é útil para criar definições de classes reutilizáveis que podem operar em diferentes tipos de dados mantendo a segurança de tipos.
class Container<T> { private item: T;
constructor(item: T) { this.item = item; }
getItem(): T { return this.item; }}
const numberContainer = new Container<number>(123);console.log(numberContainer.getItem()); // 123
const stringContainer = new Container<string>('hello');console.log(stringContainer.getItem()); // helloRestrições Genéricas
Parâmetros genéricos podem ser restringidos usando a palavra-chave extends seguida por um tipo ou interface que o parâmetro de tipo deve satisfazer.
No exemplo a seguir, T deve conter uma propriedade length para ser válido:
const printLen = <T extends { length: number }>(value: T): void => { console.log(value.length);};
printLen('Hello'); // 5printLen([1, 2, 3]); // 3printLen({ length: 10 }); // 10printLen(123); // InválidoUm recurso interessante de genéricos introduzido na versão 3.4 RC é a inferência de tipo de função de ordem superior que introduziu argumentos de tipo genérico propagados:
declare function pipe<A extends any[], B, C>( ab: (...args: A) => B, bc: (b: B) => C): (...args: A) => C;
declare function list<T>(a: T): T[];declare function box<V>(x: V): { value: V };
const listBox = pipe(list, box); // <T>(a: T) => { value: T[] }const boxList = pipe(box, list); // <V>(x: V) => { value: V }[]Esta funcionalidade permite programação em estilo pointfree com tipos mais seguros, que é comum em programação funcional.
Narrowing contextual genérico
O narrowing contextual para generics é o mecanismo no TypeScript que permite ao compilador estreitar o tipo de um parâmetro genérico com base no contexto em que é usado, é útil ao trabalhar com tipos genéricos em instruções condicionais:
function process<T>(value: T): void { if (typeof value === 'string') { // Value é reduzido para o tipo 'string' console.log(value.length); } else if (typeof value === 'number') { // Value é reduzido para o tipo 'number' console.log(value.toFixed(2)); }}
process('hello'); // 5process(3.14159); // 3.14