TypeScript 添加合并实用程序类型

rm5edbpk  于 8个月前  发布在  TypeScript
关注(0)|答案(8)|浏览(135)

搜索词

合并类型

建议

如果可以合并两个类型,那就太好了。目前我能得到的最接近的是一个联合体:

  1. type Test1 = { id: number, code: string }
  2. type Test2 = { id: string, code: number }
  3. type Test3 = Test1 | Test2
  4. // This is not allowed, because x doesn't match Test1 OR Test2
  5. export const x: Test3 = {
  6. id: "bob",
  7. code: "bob"
  8. }

基本上,我想要通过合并Test1和Test2来创建以下类型:

  1. type Test3 = { id: number | string, code: number | string }

也许可以有一个工具来合并这些类型:

  1. type Test3 = Merge<Test1, Test2>

用例

React-Native有几个样式类型(TextStyle,ImageStyle,ViewStyle),它们在我的代码中非常有用,可以用于自动补全和类型检查。TextStyle有一个属性 fontSize: number 。我想能够重新定义TextStyle类型,以允许所有属性都有字符串值(这样我就可以使用自己的变量系统)。类似于 fontSize: "$fontSizeMD"

示例

  1. type EImageStyle = ImageStyle | Partial<Record<keyof ImageStyle, string>>
  2. type ETextStyle = TextStyle | Partial<Record<keyof TextStyle, string>>
  3. type EViewStyle = ViewStyle | Partial<Record<keyof ViewStyle, string>>
  4. type EComponentStyle = EImageStyle | ETextStyle | EViewStyle
  5. type EComponentStyles = { [P in keyof any]: EComponentStyle }

有了合并,它看起来会是这样的:

  1. type EImageStyle = Merge<ImageStyle, Partial<Record<keyof ImageStyle, string>>>
  2. type ETextStyle = Merge<TextStyle, Partial<Record<keyof TextStyle, string>>>
  3. type EViewStyle = Merge<ViewStyle, Partial<Record<keyof ViewStyle, string>>>
  4. type EComponentStyle = EImageStyle | ETextStyle | EViewStyle
  5. type EComponentStyles = { [P in keyof any]: EComponentStyle }

检查清单

我的建议满足以下准则:

  • 这不会对现有的TypeScript/JavaScript代码造成破坏性的变化
  • 这不会改变现有JavaScript代码的运行时行为
  • 这可以在不根据表达式的类型发出不同的JS的情况下实现
  • 这不是一个运行时特性(例如库功能、带有JavaScript输出的非ECMAScript语法等)
  • 这个特性将与 TypeScript's Design Goals 的其他部分保持一致。
68de4m5k

68de4m5k1#

我不会使用“合并”这个词,因为它对不同的人意味着不同的事情。
我不确定这是否应该成为TS本身的实用程序类型。我不认为这是一个常见的用例。
你得到了属性的并集。对于每个属性,你得到了类型的并集。
有许多方法可以“合并”依赖于用例的对象。从我的个人经验来看,这里有一些。
对于属性:

  • 属性的并集
  • 属性的交集
  • 仅左侧操作数的属性
  • 仅右侧操作数的属性

对于属性的类型:

  • 类型的并集
  • 类型的交集
  • 左侧操作数的类型的覆盖右侧操作数的类型
  • 右侧操作数的类型的覆盖左侧操作数的类型

仅根据上述内容,你就有16种不同的方式来“合并”两种类型(忽略对称性)

展开查看全部
vddsk6oq

vddsk6oq2#

命名
或许 Merge 不是一个合适的词汇,其他选择可以是:JoinCombineBlend ?
我选择了 Merge ,因为它与 declaration merging 感觉相关。

合并的不同方式

我认为左操作数或右操作数并不重要。如果你想要进行交集操作,你可以使用 Merge 结合 Omit 和/或 Exclude 实用程序来实现。

使用场景

我能想到的一个用例:
假设你有一个对象仓库 Repository<T> ,它有一个查询对象的函数:repository.find({ age: 20 }) 。该函数通过仅接受 Partial<T> 类型的参数来强制类型安全。
接下来,你想让这个函数变得更智能,使其能够执行类似 repository.find({ age: { operator: "lte", value: 20 }}) 的调用。
你可以为此使用联合,但这样你就无法混合两者了:

  1. repository.find({
  2. age: 20,
  3. name: { operator: "regex", value: "^(Lord|Lady) (.+)" }
  4. })
展开查看全部
aemubtdh

aemubtdh3#

你的用例似乎不太常见或通用,我个人认为。但也许其他人会有不同的看法。(我这样说是因为我自己写了很多查询/仓库代码)

eimct9ow

eimct9ow4#

昨天我在开发一个React-Native应用,并遇到了另一个需求。
React-Native提供了一个Image组件,但它没有内置的加载指示器。所以我正在开发一个RemoteImage组件,该组件在图像完全加载之前显示加载指示器。
我希望RemoteImage属性的source比原始的Image组件更智能。你应该能够指定一个直接的图像URL(字符串)或一个更复杂的ImageURISource对象。原始的Image组件仅支持ImageURISource
目前,这是我的RemoteImage组件的Props类型:

  1. type Props = {
  2. source: string | ImageURISource
  3. style?: StyleProp<ImageStyle>
  4. }

Image组件具有更多的属性,所以如果能这样做就太好了:

  1. type RemoteImageProps = {
  2. source: string | ImageURISource
  3. ...
  4. }
  5. type Props = Merge<ImageProps, RemoteImageProps>
  6. ...
  7. class RemoteImage extends React.Component<Props, State> {
  8. ...
  9. render() {
  10. const { source, ...props } = this.props
  11. // For the source property we support direct urls and more complex source objects
  12. const imageSource = (typeof source === "string") ? { uri: source } : source
  13. return (
  14. <Fragment>
  15. <Image source={imageSource} {...props} />
  16. ...
展开查看全部
ergxz8rk

ergxz8rk5#

It would make sense to have a merge as it's already implemented in JS.

  1. const one = {
  2. message: 'Some text',
  3. overridedProp: 5 // number.
  4. }
  5. const two = {
  6. overridedProp: 'test' // string.
  7. }
  8. const three = {
  9. ...one,
  10. ...two
  11. }
  12. /*
  13. {
  14. message: 'Some text',
  15. overridedProp: 'test' // string.
  16. }
  17. */
展开查看全部
ztigrdn8

ztigrdn86#

为什么不使用 Omit ?

  1. interface ButtonProps {
  2. color: 'black' | 'white',
  3. children: React.ReactNode
  4. }
  5. interface ButtonExtendedProps {
  6. color: string
  7. }
  8. type ButtonCombinedProps = Omit<ButtonProps, 'color'> & ButtonExtendedProps
j2qf4p5b

j2qf4p5b7#

@alexandermckay 当有超过两个 prop 时,这并不实用。
我正在使用这个助手:
export declare type Overwrite<T, U> = Omit<T, keyof U> & U; .

brtdzjyr

brtdzjyr8#

快速尝试创建ask...请随时告诉我它的不足之处。..

  1. type Merge<T0, T1, T2 = keyof (T0 | T1)[]> = (Pick<T0, Extract<Extract<T2, keyof T0>, Extract<T2, keyof T1>>> &
  2. Pick<T1, Extract<Extract<T2, keyof T0>, Extract<T2, keyof T1>>>) &
  3. Partial<Pick<T0, Exclude<Extract<T2, keyof T0>, Extract<T2, keyof T1>>>> &
  4. Partial<Pick<T1, Exclude<Extract<T2, keyof T1>, Extract<T2, keyof T0>>>>;

添加了T2以允许指定要合并的键,可以从任一侧进行。希望对某人有所帮助。我相信更高级的开发人员可以做得更好
或者尝试使其更清晰...

  1. type Shared<T0, T1, Keys> = Pick<T0, Extract<Extract<Keys, keyof T0>, Extract<Keys, keyof T1>>> &
  2. Pick<T1, Extract<Extract<Keys, keyof T0>, Extract<Keys, keyof T1>>>;
  3. type OptionalRHS<T0, T1, Keys> = Partial<Pick<T0, Exclude<Extract<Keys, keyof T0>, Extract<Keys, keyof T1>>>>;
  4. type OptionalUnshared<T0, T1, Keys> = OptionalRHS<T0, T1, Keys> &
  5. OptionalRHS<T1, T0, Keys>;
  6. type Merge<T0, T1, Keys = keyof (T0 | T1)[]> = Shared<T0, T1, Keys> &
  7. OptionalUnshared<T0, T1, Keys>;
展开查看全部

相关问题