Maison >interface Web >js tutoriel >Maîtriser les types récursifs dans TypeScript : gérer les limitations de profondeur avec élégance
Lorsque vous travaillez avec des structures de données profondément imbriquées dans TypeScript, créer des types d'utilitaires pour transformer ces structures est une tâche courante. Cependant, les types récursifs, bien que puissants, comportent leur propre ensemble de défis.
L'un de ces défis consiste à contrôler efficacement la profondeur de récursion pour empêcher le calcul de type de dépasser les capacités de TypeScript. Cet article explorera une approche courante pour incrémenter et décrémenter les nombres au niveau du type, identifiera ses limites et présentera une solution robuste pour gérer la profondeur de récursion à l'aide des types d'incrément et de décrémentation appropriés.
Pour mieux comprendre les limites, examinons une approche naïve souvent utilisée lors de l'incrémentation ou de la décrémentation de nombres au niveau du type :
type Prev = [never, 0, 1, 2, 3, 4]; type Next = [1, 2, 3, 4, 5, 6]; type MinusOne = Prev[5]; // ? 4 type PlusOne = Next[5]; // ? 6
Supposons que vous ayez un type d'objet profondément imbriqué et que vous souhaitiez tout créer
propriétés facultatives jusqu'à un niveau spécifié :
type DeepObject = { a: number; b: { c: string; d: { e: boolean; f: { g: string; h: { i: number; j: { k: string; }; }; }; }; }; };
Avec une approche naïve et codée en dur, la gestion de la profondeur à laquelle les propriétés deviennent facultatives ressemblerait à ceci :
type Prev = [never, 0, 1, 2, 3, 4]; type DeepOptional< T, Limit extends number = 1 > = Limit extends never ? never : { [K in keyof T]?: T[K] extends object ? DeepOptional<T[K], Prev[Limit]> : T[K]; };
Explication :
Exemple d'utilisation :
type NewDeepObject = DeepOptional<DeepObject, 3>; // Result: // { // a?: number; // b?: { // c?: string; // d?: { // e?: boolean; // f?: { // g: string; // h: { // i: number; // j: { // k: string; // }; // }; // }; // }; // }; // }; type NewDeepObject = DeepOptional<DeepObject, 1>; // Result: // { // a?: number; // b?: { // c: string; // d: { // e: boolean; // f: { // g: string; // h: { // i: number; // j: { // k: string; // }; // }; // }; // }; // }; // };
Pour surmonter les limitations des tableaux prédéfinis, nous pouvons utiliser la manipulation de tuples pour créer des opérations d'incrémentation et de décrémentation de type sécurisé qui évoluent dynamiquement.
type Prev = [never, 0, 1, 2, 3, 4]; type Next = [1, 2, 3, 4, 5, 6]; type MinusOne = Prev[5]; // ? 4 type PlusOne = Next[5]; // ? 6
type DeepObject = { a: number; b: { c: string; d: { e: boolean; f: { g: string; h: { i: number; j: { k: string; }; }; }; }; }; };
type Prev = [never, 0, 1, 2, 3, 4]; type DeepOptional< T, Limit extends number = 1 > = Limit extends never ? never : { [K in keyof T]?: T[K] extends object ? DeepOptional<T[K], Prev[Limit]> : T[K]; };
type NewDeepObject = DeepOptional<DeepObject, 3>; // Result: // { // a?: number; // b?: { // c?: string; // d?: { // e?: boolean; // f?: { // g: string; // h: { // i: number; // j: { // k: string; // }; // }; // }; // }; // }; // }; type NewDeepObject = DeepOptional<DeepObject, 1>; // Result: // { // a?: number; // b?: { // c: string; // d: { // e: boolean; // f: { // g: string; // h: { // i: number; // j: { // k: string; // }; // }; // }; // }; // }; // };
Explorons comment ces types d'utilitaires peuvent être appliqués à un problème réel plus complexe : rendre les propriétés d'un objet facultatives jusqu'à une certaine profondeur.
Supposons que vous ayez un type d'objet profondément imbriqué et que vous souhaitiez tout créer
propriétés facultatives jusqu'à un niveau spécifié :
type Length<T extends any[]> = (T extends { length: number } ? T["length"] : never) & number;
Avec une approche naïve et codée en dur, gérer la profondeur à laquelle les propriétés deviennent facultatives serait complexe. Voici comment un utilitaire DeepOptional de type sécurisé peut résoudre ce problème :
Implémentation de DeepOptional
type TupleOf<N extends number, T extends unknown[] = []> = Length<T> extends N ? T : TupleOf<N, [...T, unknown]>;
Explication :
Exemple d'utilisation :
type Pop<T extends any[]> = T extends [...infer U, unknown] ? U : never;
Chez medusajs, nous nous engageons à trouver les solutions les plus efficaces et innovantes pour surmonter les défis techniques complexes. En tirant parti des types Incrément et Décrément basés sur des tuples, vous pouvez aller au-delà des limites des opérations de base au niveau du type et créer des utilitaires évolutifs et sécurisés. Cette méthode simplifie non seulement la gestion de la profondeur de récursion, mais garantit également que vous conservez la flexibilité nécessaire aux opérations de type complexes sans dépasser les limites de vérification de type de TypeScript.
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!