成员可见性

  • public 类外可见
  • private 类内可见 #var 真私有属性
  • protected 子类和自己可见
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
// 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 的 #

1
2
3
4
5
6
7
8
9
10
class Person {
#friend?: Person
constructor(public name: string, friend?: Person) {
this.#friend = friend
}
}

const p = new Person('jack')

p.friend // TODO: typescript 报错

static

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

1
2
3
4
5
6
7
8
9
class Person {
xxx = 1
name: string
constructor(name: string) {
this.name = name
}
}

Person.xxx // TODO: typescript 报错

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

1
2
3
4
5
6
7
8
9
class Person {
static xxx = 1
name: string
constructor(name: string) {
this.name = name
}
}

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

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

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

他不能在固有属性上使用

1
2
3
4
5
6
7
class Person {
static name: string // TODO: 报错啦
name: stirng
constructor(name: string) {
this.name = name
}
}

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


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

1
2
3
4
5
6
7
8
9
10
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 程序员,那么很简单

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

count += 1

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

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

类和泛型

1
2
3
4
5
6
7
8
9
10
11
12
13
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,那么直接用继承就好了

1
2
3
4
5
class Hash<K, V> extends Map<K, V> {
destory() {
this.clear()
}
}

抽象类

1
2
interface A {}
class B {}

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

1
2
3
4
5
6
7
abstract class Person {
a() {
console.log('a')
}
abstract name: string
abstract b: () => number
}

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

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

把类当作参数(常用)

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

1
2
3
4
5
6
7
class Person {}

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

fn(Person)

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

1
2
3
4
5
6
7
8
9
// 第一种:
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 也能写出很好的程序。