TS与JS中的Getters和Setter究竟有什么用

做者:Khalil Stemmler

翻译:疯狂的技术宅javascript

原文:https://www.freecodecamp.org/...html

未经容许严禁转载前端

在本文中,咱们讨论了getter 和 setter 在现代 Web 开发中的实用性。它们有用吗?何时使用它们是有意义的?

当 ECMAScript 5(2009)发布时,getters 和 setter(也称为访问器)被引入 JavaScript。vue

问题是,对于引入它们的缘由及实用性存在不少困惑。java

我在 reddit 看到了一个帖子,讨论的内容是它们是不是反模式。react

不幸的是,该主题的广泛共识是 “yes”。我认为这是由于大多数状况下,你所作的前端编程都不会要求提供 getter 和 setter 这样的操做。git

尽管我不一样意 getter 和 setter 彻底是一个反模式。但它们在几种状况下能带来更多的实用性。程序员

它们是什么?

getter 和 setter 是另外一种提供对象属性访问的方法。github

通常的用法以下所示:面试

interface ITrackProps {
  name: string;
  artist: string;
}

class Track {  
  private props: ITrackProps;

  get name (): string {
    return this.props.name;
  }

  set name (name: string) {
      this.props.name = name;
  }

  get artist (): string {
    return this.props.artist;
  }

  set artist (artist: string) {
      this.props.artist = artist;
  }

  constructor (props: ITrackProps) {
    this.props = props;
  } 

  public play (): void {    
      console.log(`Playing ${this.name} by ${this.artist}`);
  }
}

如今问题变成了:“为何不仅使用常规类属性?”

那么,在这种状况下,是能够的

interface ITrackProps {
  name: string;
  artist: string;
}

class Track {  
  public name: string;
  public artist: string;

  constructor (name: string, artist: string;) {
    this.name = name;
    this.artist = artist;
  } 

  public play (): void {    
      console.log(`Playing ${this.name} by ${this.artist}`);
  }
}

这是一个很是简单的例子,让咱们来看一个更好地描述,为何咱们应该关心使用 getter 和 settter 与常规类属性的场景。

防止贫血模式

你还记得贫血模式(译者注:一种反模式)是什么吗?尽早发现贫血模式的方法之一是,假如你的域实体的每一个属性都有getter和setter(即:set 对域特定语言没有意义的操做)暴露的话。

若是你没有明确地使用 getset 关键字,那么会使全部 public 也有相同的负面影响。

思考这个例子:

class User {
  // 很差。你如今能够`set` 用户ID。 
  // 是否须要将用户的 id 变动为其余标识符? 
  // 这样安全吗? 你应该这样作吗?
  public id: UserId;

  constuctor (id: UserId) {
    this.id = id;
  }
}

在领域驱动设计中,为了防止出现贫血模式,并推动特定于领域的语言的建立,对于咱们仅公开对领域有效的操做很是重要。

这意味着你须要了解本身正在工做的领域

我会让本身接受审查。让咱们来看看 White Label 中的 Vinyl 类,这是一个开源的乙烯基交易程序,使用领域驱动进行设计并基于 TypeScript 构建。

import { AggregateRoot } from "../../core/domain/AggregateRoot";
import { UniqueEntityID } from "../../core/domain/UniqueEntityID";
import { Result } from "../../core/Result";
import { Artist } from "./artist";
import { Genre } from "./genre";
import { TraderId } from "../../trading/domain/traderId";
import { Guard } from "../../core/Guard";
import { VinylCreatedEvent } from "./events/vinylCreatedEvent";
import { VinylId } from "./vinylId";

interface VinylProps {
  traderId: TraderId;
  title: string;
  artist: Artist;
  genres: Genre[];
  dateAdded?: Date;
}

export type VinylCollection = Vinyl[];

export class Vinyl extends AggregateRoot<VinylProps> {

  public static MAX_NUMBER_GENRES_PER_VINYL = 3;

    //🔥1. 外观。 VinylId 键实际上并不存在
  //做为属性的 VinylProps,但咱们仍然须要
  //提供对它的访问。
  get vinylId(): VinylId {
    return VinylId.create(this.id)
  }

  get title (): string {
    return this.props.title;
  }

  // 🔥2. 全部这些属性都做为 props 嵌套
  // 在一层,这样咱们就能够控制对 ACTUAL 值
  // 的访问和变化。
  get artist (): Artist {
    return this.props.artist
  }

  get genres (): Genre[] {
    return this.props.genres;
  }

  get dateAdded (): Date {
    return this.props.dateAdded;
  }

  // 🔥3. 你会发现到目前为止尚未 setter,
  // 由于在建立以后去改变这些东西是没有意义的
  
  get traderId (): TraderId {
    return this.props.traderId;
  }

  // 🔥4. 这种方法称为“封装集合”。
  // 是的,咱们须要添加类型。 但咱们仍
  // 然没有公开 setter,由于这里有一
  // 些咱们想要确保强制执行的不变逻辑。

  public addGenre (genre: Genre): void {
    const maxLengthExceeded = this.props.genres
      .length >= Vinyl.MAX_NUMBER_GENRES_PER_VINYL;

    const alreadyAdded = this.props.genres
      .find((g) => g.id.equals(genre.id));

    if (!alreadyAdded && !maxLengthExceeded) {
      this.props.genres.push(genre);
    }
  }

  // 🔥 5. 提供一种删除方式。

  public removeGenre (genre: Genre): void {
    this.props.genres = this.props.genres
      .filter((g) => !g.id.equals(genre.id));
  }

  private constructor (props: VinylProps, id?: UniqueEntityID) {
    super(props, id);
  }

  // 🔥 6. 这就是咱们建立 Vinyl 的方法。
  // 建立以后,除了 Genre 以外,全部属性
  // 都会变为“只读”,由于启用修改是有意义的。
  public static create (props: VinylProps, id?: UniqueEntityID): Result<Vinyl> {
    const propsResult = Guard.againstNullOrUndefinedBulk([
      { argument: props.title, argumentName: 'title' },
      { argument: props.artist, argumentName: 'artist' },
      { argument: props.genres, argumentName: 'genres' },
      { argument: props.traderId, argumentName: 'traderId' }
    ]);

    if (!propsResult.succeeded) {
      return Result.fail<Vinyl>(propsResult.message)
    } 

    const vinyl = new Vinyl({
      ...props,
      dateAdded: props.dateAdded ? props.dateAdded : new Date(),
      genres: Array.isArray(props.genres) ? props.genres : [],
    }, id);
    const isNewlyCreated = !!id === false;

    if (isNewlyCreated) {
      // 🔥 7. 这就是咱们须要 VinylId 的缘由:
      // 为这个域事件的全部订阅者提供标识符。
      vinyl.addDomainEvent(new VinylCreatedEvent(vinyl.vinylId))
    }

    return Result.ok<Vinyl>(vinyl);
  }
}

充当外观、维护只读值、强制执行模型表达、封装集合以及建立域事件领域驱动设计中 getter 和 setter 的一些很是可靠的用例。

在 Vue.js 中更改检测

Vue.js 是一个较新的前端框架,以其快速和响应式而闻名。

Vue.js 可以如此有效地检测改变的缘由是它们用 Object.defineProperty() API监视对 View Models 的更改!

来自 Vue.js 关于响应式的文档:

当你将纯 JavaScript 对象做为其数据选项传递给 Vue 实例时,Vue 将遍历其全部属性并用 Object.defineProperty 将它们转换为 getter/setter。 getter/setter 对用户是不可见的,可是在幕后,它们使 Vue 可以在访问或修改属性时执行依赖关系跟踪和更改通知。 —— Vue.js 文档:响应式

总之,getter 和 setter 针对不少问题有很大的实用性。不过在现代前端 Web 开发中,这些问题并无太多出现。


本文首发微信公众号:前端先锋

欢迎扫描二维码关注公众号,天天都给你推送新鲜的前端技术文章

欢迎扫描二维码关注公众号,天天都给你推送新鲜的前端技术文章

欢迎继续阅读本专栏其它高赞文章:


相关文章
相关标签/搜索