解读TypeScript 5.2(二)——— 聊聊装饰器元数据

元数据

首先我们先来看一下元数据的用途有哪些,官网给的用途元数据可能会用于许多用途,例如调试、序列化或使用装饰器执行依赖项注入

看到这里元数据在开发中还是有些应用场景的(ps:在后端依赖注入和序列化还是很有场景的)。

案例分析

这里我们来分析一下官网的第一个案例。

ts 复制代码
interface Context {
    name: string;
    metadata: Record<PropertyKey, unknown>;
}

function setMetadata(_target: any, context: Context) {
    context.metadata[context.name] = true;
}

class SomeClass {
    @setMetadata
    foo = 123;

    @setMetadata
    accessor bar = "hello!";

    @setMetadata
    baz() { }
}
const ourMetadata = SomeClass[Symbol.metadata];
console.log(JSON.stringify(ourMetadata));
// { "bar": true, "baz": true, "foo": true }

首先我们先来分析一下metadata方法,其实我们看TypeScript最新的源代码中Symbol添加了一个新的属性在lib.esnext.decorators文件中。

ts 复制代码
/// <reference no-default-lib="true"/>

/// <reference lib="es2015.symbol" />
/// <reference lib="decorators" />

interface SymbolConstructor {
    readonly metadata: unique symbol;
}

interface Function {
    [Symbol.metadata]: DecoratorMetadata | null;
}

这个属性就是用来存储元数据的。在一个类中我们去调用[Symbol.metadata]方法就可以获取到这个类中的元数据。我们可以在装饰器中去给这个元数据对象(通过源码查看DecoratorMetadata的类型是Record<PropertyKey, unknown> & object)注入数据,例如例子中的这段代码 context.metadata[context.name] = true;。最后我们可以通过类调用[Symbol.metadata]方法获取注入的内容。

应用场景

通过上面的案例我们已经了解元数据应该如何写了,让我们看一下开发中具体的案例来看一下元数据的场景。

ts 复制代码
const serializables = new WeakMap<object, string[]>();
type Context =
    | ClassAccessorDecoratorContext
    | ClassGetterDecoratorContext
    | ClassFieldDecoratorContext
    ;
export function serialize(_target: any, context: Context): void {
    if (context.static || context.private) {
        throw new Error("Can only serialize public instance members.")
    }
    if (typeof context.name !== "string") {
        throw new Error("Can only serialize string properties.");
    }
    let propNames = serializables.get(context.metadata);
    if (propNames === undefined) {
        serializables.set(context.metadata, propNames = []);
    }
    propNames.push(context.name);
}
export function jsonify(instance: object): string {
    const metadata = instance.constructor[Symbol.metadata];
    const propNames = metadata && serializables.get(metadata);
    if (!propNames) {
        throw new Error("No members marked with @serialize.");
    }
    const pairStrings = propNames.map(key => {
        const strKey = JSON.stringify(key);
        const strValue = JSON.stringify((instance as any)[key]);
        return `${strKey}: ${strValue}`;
    });
    return `{ ${pairStrings.join(", ")} }`;
}

class Person {

    firstName: string;

    lastName: string;
    
    @serialize
    age: number

    @serialize
    get fullName() {
        return `${this.firstName} ${this.lastName}`;
    }
    toJSON() {
        return jsonify(this)
    }
    constructor(firstName: string, lastName: string, age: number) {
    }
}

这里是官方给的一个案例,具体的逻辑我就不解释了我想看一遍应该就能看懂,他实现的功能是通过jsonify去打印被serialize装饰器注入的字段。

总结

对于ts新支持的元数据特性我认为主要的应用场景可能是angular或者nodejs(服务端)的场景(ps:现在大趋势是函数式编程)。

相关推荐
徐小夕2 小时前
我们开源了一款“框架无关”的思维导图编辑器,3分钟集成到任意系统
前端·javascript·github
PBitW2 小时前
GPT训练我的第三天,明白了应该咋说满分回答!😕😕😕
前端·javascript·面试
像我这样帅的人丶你还2 小时前
Java 后端详解(四):分页与搜索
java·javascript·后端
labixiong2 小时前
还原一个完整符合规范的 Promise(二)
前端·javascript
To_OC3 小时前
万字解析《JS 语言精粹》之第五章:继承 5 大核心精髓(JS 原型核心)
前端·javascript·代码规范
裕波4 小时前
AI 正在重写应用开发。Vue 与 Vite,给出新的答案。
javascript·vue.js
妙码生花4 小时前
现代前端的极致性能 icon 加载方案(死磕成功版)
前端·vue.js·typescript
kyriewen6 小时前
折腾了半年 AI 编程工作流,最后发现效率瓶颈是桌上那块屏幕
前端·javascript·ai编程
张元清8 小时前
React useDebounce Hook:给状态和回调做防抖(2026)
javascript·react.js
MonkeyKing8 小时前
鸿蒙ArkTS深度剖析:ArkTS与TS/JS核心差异、静态强类型实战优势
typescript·harmonyos