前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >TypeScript系列教程十一《装饰器》 -- 属性装饰器

TypeScript系列教程十一《装饰器》 -- 属性装饰器

作者头像
星宇大前端
发布2022-05-06 17:20:54
9890
发布2022-05-06 17:20:54
举报
文章被收录于专栏:大宇笔记

系列教程

属性装饰器和其他装饰器功能类似,其设计也是为了统一的、复用度更高的去监听,改变属性。

属性装饰器声明在一个属性声明之前(紧靠着属性声明)。 属性装饰器不能用在声明文件中(.d.ts),或者任何外部上下文(比如 declare的类)里。 属性装饰器表达式会在运行时当作函数被调用,传入下列2个参数:

  • 对于静态成员来说是类的构造函数,对于实例成员是类的原型对象。
  • 成员的名字。 注意  属性描述符不会做为参数传入属性装饰器,这与TypeScript是如何初始化属性装饰器的有关。 因为目前没有办法在定义一个原型对象的成员时描述一个实例属性,并且没办法监视或修改一个属性的初始化方法。返回值也会被忽略。因此,属性描述符只能用来监视类中是否声明了某个名字的属性。

下面记录三个例子来具体体现 场景。

基础例子


代码目的:

拦截实例属性,处理加工,输出修改后的属性。例如输入艾希 得到 寒冰射手艾希

代码实现:

代码语言:javascript
复制
const nameDecorate:PropertyDecorator = (target: Object, propertyKey: string | symbol)=>{
  let value:string
  Object.defineProperty(target,propertyKey,{
    set:(v)=>{
      value = v
    },
    get:()=>{
      return '寒冰射手'+value
    }
  })
}

class Hero {
  @nameDecorate
  name:string
}

let hb = new Hero()
hb.name = '艾希'
console.log(hb.name);

打印结果:

工厂例子


属性装饰器函数工厂实现

代码目的:

根据工厂装饰器key,返回json,在网络请求的时候经常会占用关键字,根据工厂装饰器key确定json 属性最终名称。

代码实现:

代码语言:javascript
复制
let json:{} = {}

const modelToJsonKey:(jsonKey:string)=>PropertyDecorator = (jsonKey)=>{
  return (target:Object,pkey:string)=>{
    Object.defineProperty(target,pkey,{
      set:(v)=>{
        json[jsonKey] = v
      }
    })
  }
}
class TestMdoel {
  @modelToJsonKey('title')
  name:string
  @modelToJsonKey('ID')
  id:string

  constructor(name:string,id:string){
    this.name = name
    this.id = id
  }
}
 let modell = new TestMdoel('zhangsan','881')
console.log(json);

控制台输出:

代码语言:javascript
复制
 zhangyu@zhangyudeMacBook-Pro  ~/Desktop/study/ts/js2ts-project  ts-node propteryDecorate.ts
{ title: 'zhangsan', ID: '881' }

搭配reflect-metadata 不破坏结构存储


这是来自于官网的示例,中文档运行不通,我按照官网思想写了一遍。

代码目的:

通过属性装饰器格式,来格式化属性文字

代码实现:

代码语言:javascript
复制
import "reflect-metadata";

const format:(formatter:string)=>PropertyDecorator = (formatter)=>{
  return (target: Object, propertyKey: string | symbol) => {
    Reflect.defineMetadata(propertyKey,formatter,target) 
  }
}

class Greeter {
  @format("Hello, %s")
  greeting: string;

  constructor(message: string) {
      this.greeting = message;
  }

  greet(){
    let format = Reflect.getMetadata("greeting",this)
    format = format.replace("%s",this.greeting)
    console.log(format);
  }
}

let greeter =  new Greeter('属性装饰器')
greeter.greet()

打印结果:

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022-04-02,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 系列教程
    • 基础例子
      • 工厂例子
        • 搭配reflect-metadata 不破坏结构存储
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档