问题 #
实现一个通用 MyReadonly2<T, K>,它带有两种类型的参数 T 和 K。
K 指定应设置为Readonly的 T 的属性集。如果未提供 K,则应使所有属性都变为只读,就像普通的 Readonly<T> 一样。
例如
interface Todo {
title: string
description: string
completed: boolean
}
const todo: MyReadonly2<Todo, 'title' | 'description'> = {
title: 'Hey',
description: 'foobar',
completed: false,
}
todo.title = 'Hello' // Error: cannot reassign a readonly property
todo.description = 'barFoo' // Error: cannot reassign a readonly property
todo.completed = true // OK解答 #
type MyExclude<T, U> = T extends U ? never : T
type MyOmit<T, K extends keyof T> = { [P in MyExclude<keyof T, K>]: T[P] }
type MyReadonly2<T, K extends keyof T = keyof T> = {
readonly [P in K]: T[P]
} & MyOmit<T, K>拆分 #
K是要限制在T中的属性,所以需要确认T中的属性是否在K中,使用K extends keyof T确认K有可能没有指定,所以需要使用K extends keyof T = keyof T来确认K extends keyof T = keyof T在K没有指定时,等价于keyof T换句话说相当于赋一个默认值&是取交集的意思,后面覆盖前面{ readonly [P in K]: T[P] }把K中的属性名称对应的属性变为只读- 需要从
T中拿到不在K中的属性 - 并把这些属性恢复为原来的类型