- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我在 Typescript 将默认值传递给类似于 lodash 的 pick
的函数时遇到问题。
该函数接受一个已知(非通用)接口(interface)的对象和一组从该对象中选取和返回的键。
函数的常规(无默认参数)声明工作正常,但是,我似乎无法将数组设置为选择属性的参数的默认值。
interface Person {
name: string;
age: number;
address: string;
phone: string;
}
const defaultProps = ['name', 'age'] as const;
function pick<T extends keyof Person>(obj: Person, props: ReadonlyArray<T> = defaultProps): Pick<Person, T> {
return props.reduce((res, prop) => {
res[prop] = obj[prop];
return res;
}, {} as Pick<Person,T>);
}
const testPerson: Person = {
name: 'mitsos',
age: 33,
address: 'GRC',
phone: '000'
};
如果您删除默认值 = defaultProps
,它会成功编译,并且从示例调用返回的类型也是正确的,例如:const testPick = pick(testPerson, ['name'] );
但是,设置默认值会产生以下错误:
Type 'readonly ["name", "age"]' is not assignable to type 'readonly T[]'.
Type '"name" | "age"' is not assignable to type 'T'.
'"name" | "age"' is assignable to the constraint of type 'T', but 'T' could be instantiated with a different subtype of constraint 'keyof Person'.
Type '"name"' is not assignable to type 'T'.
'"name"' is assignable to the constraint of type 'T', but 'T' could be instantiated with a different subtype of constraint 'keyof Person'.
如何将默认值成功传递给 props
参数?
Typescript Playground 链接 here
玩了一会儿之后,我尝试使用条件类型并设法使函数签名正常工作,但现在无法正确识别 reduce 的问题:
interface Person {
name: string;
age: number;
address: string;
phone: string;
}
const defaultProps = ['name', 'age'] as const;
type DefaultProps = typeof defaultProps;
type PropsOrDefault<T extends keyof Person> = DefaultProps | ReadonlyArray<T>;
type PickedPropOrDefault<T extends PropsOrDefault<keyof Person>> = T extends DefaultProps ? Pick<Person, DefaultProps[number]> : Pick<Person, T[number]>;
function pick<T extends keyof Person>(obj: Person, props: PropsOrDefault<T> = defaultProps): PickedPropOrDefault<PropsOrDefault<T>> {
return props.reduce<PickedPropOrDefault<PropsOrDefault<T>>>((res, prop) => {
res[prop] = obj[prop];
return res;
}, {} as PickedPropOrDefault<PropsOrDefault<T>>);
}
const testPerson: Person = {
name: 'mitsos',
age: 33,
address: 'GRC',
phone: '000'
};
const result = pick(testPerson) // Pick<Person, "name" | "age">
const result2 = pick(testPerson, ['phone']) // Pick<Person, "phone">
const result3 = pick(testPerson, ['abc']) // expected error
最佳答案
您可以重载 pick
功能:
interface Person {
name: string;
age: number;
address: string;
phone: string;
}
const defaultProps = ['name', 'age'] as const;
type DefaultProps = typeof defaultProps;
function pick(obj: Person): Pick<Person, DefaultProps[number]>
function pick<Prop extends keyof Person, Props extends ReadonlyArray<Prop>>(obj: Person, props: Props): Pick<Person, Props[number]>
function pick<T extends keyof Person>(obj: Person, props = defaultProps) {
return props.reduce((res, prop) => ({
...res,
[prop]: obj[prop]
}), {} as Pick<Person, T>);
}
const testPerson = {
name: 'mitsos',
age: 33,
address: 'GRC',
phone: '000'
};
const result = pick(testPerson) // Pick<Person, "name" | "age">
const result2 = pick(testPerson, ['phone']) // Pick<Person, "phone">
const result3 = pick(testPerson, ['abc']) // expected error
你可以找到更高级的pick
在我的 article 中打字和其他答案: First , second , third
更新
props
有问题此代码中的参数:
function pick<T extends keyof Person>(obj: Person, props: PropsOrDefault<T> = defaultProps): PickedPropOrDefault<PropsOrDefault<T>> {
return props.reduce((res, prop) => {
return {
...res,
[prop]: obj[prop]
}
}, {});
}
PropsOrDefault
可能等于此类型:type UnsafeReduceUnion = DefaultProps | ReadonlyArray<'phone' | 'address'>
您可能已经注意到,联合中的这些数组是完全不同的。他们没有任何共同点。
如果你想调用reduce
:
declare var unsafe:UnsafeReduceUnion;
unsafe.reduce()
你会得到一个错误,因为reduce
是not callable
关于typescript - 如何在 Typescript 中将默认值设置为具有类型 keyof(来自显式类型)的参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69680545/
我有一个使用 keyof 特性的简单函数: interface B { name: string; age: number; } function test(key: keyof B) {}
class Car { engine:number; detials:{ good:'Boy' } } 类 ModelProperty当使用 new Model
遇到以下代码: type RequireOnlyOne 如果有人能解释什么是 keyof T = keyof T,我们将不胜感激? 最佳答案 这意味着该函数采用一个必需的类型参数 T 和一个可选的类型
在 TypeScript 中,一些类型是使用 extends keyof 定义的或 in keyof .我试图理解它们的意思,但到目前为止我没有成功。 我得到的是 keyof alone 返回一个联合
下面的 typescript 定义有区别吗: function prop(obj: T, key: K) { return obj[key]; } 和 function prop2(obj:
假设我有一个为一组数据定义有效值的接口(interface): interface Foo { bar: boolean; } 而且我希望一个类能够使用一种方法公开该数据。我发现如果我使用 key
这是错误还是我误解了 typescript 的东西? 示例代码如下: type Omit = Pick>; const func = () => { const aWithoutB: Omit =
我想从配置对象推断出此对象类型中 on 下所有键的联合: type Config = { initial: string; states: { idle: { on: {
我在理解字符串枚举在使用它们索引类型时的行为方式时遇到了一些问题。似乎有时 TS 认识到字符串枚举的值是某种类型的 keyof 的扩展,而其他时候则不然。举例说明: enum Key { FO
我有以下代码: const KeyboardEventKeys = { Escape: 'Escape', Enter: 'Enter', Tab: 'Tab' }; type Keybo
我正在尝试用 Typescript 编写一个函数,它需要: 一个对象,它至少有一个 X 类型的成员(在这个例子中: bool 值) 指向所述值的对象的键 (eg. {someValue: true}
如果我有以下类型 interface Foo { bar: string; baz: number; qux: string; } 我可以使用 typeof 来键入一个参数,这
例如。给定以下代码。我在最后一个 compat[k] 上收到 typescript 错误,错误说 Type 'keyof T' cannot be used to index type 'Partia
创建一个类型,其中 Type 中的每个属性现在都是 boolean 类型: type OptionsFlags = { [Property in keyof Type]: boolean; };
我想定义一个适配器架构,将一个对象转换为另一个。 export type AdapterSchema = { [key in keyof Partial]: { target:
我有一个按键嵌套的对象,如下所示: const OBJECTS = { OBJECT1: { properties: { prop1: { value:
我正在努力编写一个代码来推断 args.value 的类型里面if范围: class Foo { public id: number; public name: string;
我正在尝试编写一个通用函数,它可以通过键名切换任何对象中的 bool 属性。我读了release notes for TypeScript-2.8并认为条件类型应该可以解决此类问题。但是,我不知道如何
发现很难清楚地解释这一点。 在下面的 Setter 界面中,我将“field”键的类型限制为 keyof Store(在本例中为“name”|“age”)。 然后对于“值”键,我想获取给定“字段”键的
我有一个抽象类 export abstract class ABaseModel { public static isKeyOf(propName: (keyof T)): string {
我是一名优秀的程序员,十分优秀!