TypeScript Type-safe Omit Function

前端 未结 4 1838
余生分开走
余生分开走 2021-02-08 20:00

I want to replicate lodash\'s _.omit function in plain typescript. omit should return an object with certain properties removed specified via parameter

相关标签:
4条回答
  • 2021-02-08 20:17

    If we limit the type of keys to string [],It works. But it does not seem to be a good idea.Keys should be string | number | symbol[];

    function omit<T, K extends string>(
      obj: T,
      ...keys: K[]
    ): { [k in Exclude<keyof T, K>]: T[k] } {
      let ret: any = {};
      Object.keys(obj)
        .filter((key: K) => !keys.includes(key))
        .forEach(key => {
          ret[key] = obj[key];
        });
      return ret;
    }
    const result = omit({ a: 1, b: 2, c: 3 }, 'a', 'c');
    // The compiler inferred result as 
    // {
    //   b: number;
    // }
    
    0 讨论(0)
  • 2021-02-08 20:20

    The accepted answer from Nurbol above is probably the more typed version, but here is what I am doing in my utils-min.

    It uses the typescript built-in Omit and is designed to only support string key names. (still need to loosen up the Set to Set, but everything else seems to work nicely)

    export function omit<T extends object, K extends Extract<keyof T, string>>(obj: T, ...keys: K[]): Omit<T, K> {
      let ret: any = {};
      const excludeSet: Set<string> = new Set(keys); 
      // TS-NOTE: Set<K> makes the obj[key] type check fail. So, loosing typing here. 
    
      for (let key in obj) {
        if (!excludeSet.has(key)) {
          ret[key] = obj[key];
        }
      }
      return ret;
    }
    
    0 讨论(0)
  • 2021-02-08 20:28

    Object.keys or for in returns keys as string and excludes symbols. Numeric keys are also converted to strings.

    You need to convert numeric string keys to numbers otherwise it will return the object with string keys.

    function omit<T extends Record<string | number, T['']>,
     K extends [...(keyof T)[]]>(
        obj: T,
        ...keys: K
    ): { [P in Exclude<keyof T, K[number]>]: T[P] } {
        return (Object.keys(obj)
             .map((key) => convertToNumbers(keys, key)) as Array<keyof T>)
            .filter((key) => !keys.includes(key))
            .reduce((agg, key) => ({ ...agg, [key]: obj[key] }), {}) as {
            [P in Exclude<keyof T, K[number]>]: T[P];
        };
    }
    
    function convertToNumbers(
        keys: Array<string | number | symbol>,
        value: string | number
    ): number | string {
        if (!isNaN(Number(value)) && keys.some((v) => v === Number(value))) {
            return Number(value);
        }
    
        return value;
    }
    
    
    // without converToNumbers omit({1:1,2:'2'}, 1) will return {'1':1, '2':'2'}
    // Specifying a numeric string instead of a number will fail in Typescript
    

    To include symbols you can use the code below.

    function omit<T, K extends [...(keyof T)[]]>(
        obj: T,
        ...keys: K
    ): { [P in Exclude<keyof T, K[number]>]: T[P] } {
        return (Object.getOwnPropertySymbols(obj) as Array<keyof T>)
            .concat(Object.keys(obj)
            .map((key) => convertToNumbers(keys, key)) as Array<keyof T>)
            .filter((key) => !keys.includes(key))
            .reduce((agg, key) => ({ ...agg, [key]: obj[key] }), {}) as {
            [P in Exclude<keyof T, K[number]>]: T[P];
        };
    }
    
    0 讨论(0)
  • 2021-02-08 20:37
    interface Omit {
        <T extends object, K extends [...(keyof T)[]]>
        (obj: T, ...keys: K): {
            [K2 in Exclude<keyof T, K[number]>]: T[K2]
        }
    }
    
    const omit: Omit = (obj, ...keys) => {
        const ret = {} as {
            [K in keyof typeof obj]: (typeof obj)[K]
        };
        let key: keyof typeof obj;
        for (key in obj) {
            if (!(keys.includes(key))) {
                ret[key] = obj[key];
            }
        }
        return ret;
    };
    

    For convenience I've pulled most of the typings to an interface.

    The problem was that K had been being inferred as a tuple, not as a union of keys. Hence, I changed it's type constraint accordingly:

    [...(keyof T)[]] // which can be broke down to:
    keyof T // a union of keys of T
    (keyof T)[] // an array containing keys of T
    [...X] // a tuple that contains X (zero or more arrays like the  described one above)
    

    Then, we need to transform the tuple K to a union (in order to Exclude it from keyof T). It is done with K[number], which is I guess is self-explaining, it's the same as T[keyof T] creating a union of values of T.

    Playground

    0 讨论(0)
提交回复
热议问题