如何编写没有参数的类型化操作

IT技术 javascript reactjs typescript redux react-redux
2021-05-13 12:42:32

一个 Github 问题表明我们可以TypedAction.defineWithoutPayload用于此目的,但我找不到任何有关如何这样做的相关示例。

当访问令牌存储在有效负载中时,我使用它进行登录。如果存在令牌,则用户可以访问私人页面。

export const login = (token: string) => typedAction("LOGIN", token);

现在,在注销按钮上,我正在尝试执行一个操作来删除负载中的存储值。在这种情况下,将没有用于分派动作的参数。那么如何编写 typedAction 呢?

如果我使用:

export const logout = () => typedAction("LOGOUT");

我开始在我的减速器的有效负载上收到一个错误,即类型注销时不存在该属性。

这是我的减速器:

export const tokenReducer = (
  state: IState['token'] = null,
  { type, payload }: AppAction,
): typeof state => {
  switch (type) {
    case 'LOGIN':
      return payload;
      case 'LOGOUT':
        return null;
    default:
      return state;
  }
};

Codesandbox:https ://codesandbox.io/s/keen-brook-kntkm ? file =/ src/store/actions/login.ts:50-118

邮箱:c@c.com 密码:check

编辑:

export interface IState {
    token: string | null;
  }
const initialState: IState = {
  token: null
};

如果我使用state: typeof initialStatestate = initialState按照 IDE 的建议,action.payload 会出错

Type 'string' is not assignable to type 'IState'.ts(2322)

如果我尝试,state: initialState那么显然:

'initialState' refers to a value, but is being used as a type here. Did you mean 'typeof initialState'?ts(2749)
``
1个回答

typedAction定义函数的方式工作正常:

export function typedAction<T extends string>(type: T): { type: T };
export function typedAction<T extends string, P extends any>(
  type: T,
  payload: P
): { type: T; payload: P };
export function typedAction(type: string, payload?: any) {
  return { type, payload };
}

您遇到的问题是由于您的减速器参数中的动作解构了:

export const tokenReducer = (
  state: IState["token"] = null,
  { type, payload }: AppAction
): typeof state => {
  // ...
};

解构和 TypeScript 的难点之一是,一旦你这样做了,变量的类型就会变得相互独立。将动作解构为{ payload, type }atype: 'LOGIN' | 'LOGOUT'payload: string | undefined变量。即使您稍后细化 的值type,就像在您的 switch 语句中一样,payload仍然具有类型string | undefined; 细化后,TypeScript 不会自动细化payloadcase 块中的类型type它们的类型是完全独立的。

因此,您可以使用的一个有点丑陋的 hack 是不解构:

export const tokenReducer = (
  state: IState['token'] = null,
  action: AppAction,
): typeof state => {
  switch (action.type) {
    case 'LOGIN':
      return action.payload;
    case 'LOGOUT':
      return null;
    default:
      return state;
  }
};

这是有效的,因为在您的 switch 语句中,它能够将action: AppAction类型细化为更具体的登录或注销类型,因此action.payload现在与特定于这些操作之一的有效负载类型密切相关。

这是我使用的 redux 操作的替代模式,您可能会在我的 fork发现它更方便它让您享受映射类型的强大功能,以使用更少的样板定义减速器。首先,您必须使用类型/有效负载映射定义一个类型,并定义一些从中派生的类型:

export type ActionPayloads = {
  LOGIN: string;
  LOGOUT: void;
};

export type ActionType = keyof ActionPayloads;

export type Action<T extends ActionType> = {
  type: T;
  payload: ActionPayloads[T];
};

现在可以根据该地图定义您的动作创建者:

export function typedAction<T extends ActionType>(
  type: T,
  payload: ActionPayloads[T]
) {
  return { type, payload };
}

接下来,您可以定义一个辅助函数来创建一个强类型的 reducer:

type ReducerMethods<State> = {
  [K in ActionType]?: (state: State, payload: ActionPayloads[K]) => State
};

type Reducer<State> = (state: State, action: AppAction) => State;

function reducer<State>(
  initialState: State,
  methods: ReducerMethods<State>
): Reducer<State> {
  return (state: State = initialState, action: AppAction) => {
    const handler: any = methods[action.type];
    return handler ? handler(state, action.payload) : state;
  };
}

(对于那个丑陋的: any演员,我还没有找到好的解决方法,但至少我们从逻辑上知道打字是从外面发出的声音)。

现在,您可以使用很好的隐式类型为您的操作处理程序定义您的减速器:

type TokenState = string | null;

export const tokenReducer = reducer<TokenState>(null, {
  LOGIN: (state, token) => token, // `token` is implicitly typed as `string`
  LOGOUT: () => null              // TS knows that the payload is `undefined`
});