TypeScript全解:class(下)

成员可见性

  • public 类外可见
  • private 类内可见 #var 真私有属性
  • protected 子类和自己可见
// public 
class Person1 {
  public friend?: Person
  constructor(public name: string, friend?: Person) {
    this.friend = friend
  }
}
const p1 = new Person1('jack')
p1.friend // undefined

// private
class Person2 {
  private friend?: Person
  constructor(public name: string, friend?: Person) {
    this.friend = friend
  }
  xxx() {
    this. friend // TODO: 这里是可以使用的
  }
}
const p2 = new Person2('jack', p1)
p2.friend // TODO: typescript 报错

// protected
class Person3 {
  private friend?: Person
  constructor(public name: string, friend?: Person) {
    this.friend = friend
  }
  xxx() {
    this. friend // TODO: 这里是可以使用的
  }
}

class User extends Person3 {
  constructor(public id: number, name: string, friend?: User) {
    super(name, friend)
  }
  yyy() {
    this.friend  // TODO: 子类这里是可以使用的
  }
}

const u = new User(1, 'jack')
u.friend // TODO: typescript 报错

但是这些代码始终都会变成 JS
那么说,TS 提供的这三个关键词还有用嘛?
显然易见:因为类型擦除,在 JS 中,你又可以随心所欲了
但是我就想要私有属性怎么办,那就使用 JS 的 #

class Person {
  #friend?: Person
  constructor(public name: string, friend?: Person) {
    this.#friend = friend
  }
}

const p = new Person('jack')

p.friend // TODO: typescript 报错

static

static 的意思是这个属性是通过类名访问的

class Person {
  xxx = 1
  name: string
  constructor(name: string) {
    this.name = name
  }
}

Person.xxx // TODO: typescript 报错

所以为了解决上面的问题,面向对象又加了一个关键字,static

class Person {
  static xxx = 1
  name: string
  constructor(name: string) {
    this.name = name
  }
}

Person.xxx // TODO: 可以使用了

这就叫做静态属性
这里在提示一下 name 叫做成员属性
什么区别?静态属性是代码执行到哪一行的时候就初始化了,而成员属性是实例化后才初始化的

但是使用 staic 的时候需要注意一点

他不能在固有属性上使用

class Person {
  static name: string // TODO: 报错啦
  name: stirng
  constructor(name: string) {
    this.name = name
  }
}

为什么会这样呢?
这又要追寻到 JS 的问题了:JS 中的 class 是用函数实现的
怎么证明,用 typeof:


class 是个 function,且自然自带 name,这样看来,JS 中的 class 就是辣鸡,看起来实现了 class,但又好像是个残废,所以有关固有属性的都会报错

class Person {
  static length: string // TODO: 报错啦
  static prototype: string // TODO: 报错啦
  static arguments: string // TODO: 报错啦
  static caller: string // TODO: 报错啦
  name: stirng
  constructor(name: string) {
    this.name = name
  }
}

static block

需求是这样:我们想记录这个功能被创建过多少次?

如果是 JS 程序员,那么很简单

let count = parseInt(loocalStorage.getItem('count') || 0)

count += 1

如果是在 class 里面那就很难做到,除非~再加个语法

class Foo {
  static #count = 0
  static { // TODO: 注意这里,这种语法代表类在创建的时候会执行
    const count = loadFrimLocalStorage() || 0
    Foo.#count += count
  }
  constructor() {
    console.log(Foo.#count)
  }
}

类和泛型

class Hash<K, V> {
  map: Map<K, V> = new Map()
  set(key: K, value: V) {
    this.map.set(key, value)
  }
  get(key: K) {
    return this.map.get(key)
  }
}

const h = new Hash<string | number, string | number>()
h.set('name', 'hi')
g.get('name')

其实上面这个 Hash 就是一个 Map,那么直接用继承就好了

class Hash<K, V> extends Map<K, V> {
  destory() {
    this.clear()
  }
}

抽象类

interface A {}
class B {}

interface 只能写类型,不写实现
class 又写类型,又写实现
人都喜欢折中
能不能有的实现,有的不实现呢?
为了应对这个需求,又来到了面向对象熟悉的操作,加关键字, abstract

abstract class Person {
  a() {
    console.log('a')
  }
  abstract name: string
  abstract b: () => number
}

这就是抽象类,请注意这个抽象类是不能直接使用的!

const p = new Person()
// TODO: 报错:Cannot Create an Instance of an abstract class.ts

把类当作参数(常用)

把类作为参数,而不是把对象作为参数

class Person {}

function fn(X: Person) { // TODO: typescript 报错
  const p = new X()
}

fn(Person)

怎么解决呢?下面两种方式

// 第一种:
function fn1(X: typeof Person) {
  const p = new X()
}

// 第二种:
function fn2(X: new (name: string) => Person) {
  const p = new X()
}

看起来很奇怪把~

个人对 class 的看法

有这么多关键字,有这么多语法,搞的像背书一样,每个关键字是什么意思,怎么组合,该怎么声明类型,全都是固定的,都是套路,都是模版,有什么好处呢?

那就是工业化,千人一面,大家写出来的代码都是一样的,即使你是新手,你也能看懂老手的代码,大家沟通起来就简单多了。

如果说面向对象适合前端就罢了,但是实际上来看,虽然以前转向前端的都是面向随想那帮人,他们自然而然在写前端时候会加入各种组合,各种设计模式,

但是从 ES6 开始,前端就希望代码更加简化,希望更多的函数和对象组合,虽然这个时候出了真正的关键词 class,但是很多前端就是不用,你 class 的功能我全都可以用函数和对象来实现,直到今年 react 和 vue 都出了 hooks API,他们连 this 都不用了,更不用说 new 了,所以事实证明,不用 class 也能写出很好的程序。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容