为什么在TypeScript中,具有未定义值的记录类型应具有严格定义的字段?

huangapple go评论79阅读模式
英文:

Why Record type in TS with undefined as a value should have strictly defined fields?

问题

以下是您要翻译的内容:

  1. 实际上,我的问题与下面的情况相关:
  2. 我有一个用于表单值的类型
  3. ```typescript
  4. interface FormValues {
  5. name: string
  6. description: string
  7. expirationDate: Date | null
  8. }

以及一个用于在表单中存在错误时可以创建错误的类型

  1. const errors: Record<keyof FormValues, string | undefined> = {
  2. name: undefined,
  3. description: undefined,
  4. expirationDate: undefined
  5. }

然而,只有在我将每个字段定义为undefined时,错误类型才起作用,如果我尝试将其更改为空对象,那么TypeScript会返回错误

  1. const errors: Record<keyof FormValues, string | undefined> = {}
  2. /*
  3. Type &#39;{}&#39; is missing the following properties
  4. from type &#39;Record<keyof FormValues, string | undefined>&#39;: name, description, expirationDate
  5. */

如何更改错误类型以反映我的需求,即将字段定义为字符串或不存在的对象:

  1. const validate = values => {
  2. const errors: ??? = {}
  3. if (!values.name) errors.name = 'Required'
  4. if (!values.description) errors.description = 'Required'
  5. if (!values.expirationDate) errors.expirationDate = 'Required'
  6. return errors
  7. }
英文:

Practically my question is related to a situation below:

I have a type for form values

  1. interface FormValues {
  2. name: string
  3. description: string
  4. expirationDate: Date | null
  5. }

and a type for errors that can be created if there are errors in a form

  1. const errors: Record&lt;keyof FormValues, string | undefined&gt; = {
  2. name: undefined,
  3. description: undefined,
  4. expirationDate: undefined
  5. }

however, the errors type works only if I define each fields as undefined, if I try to change it to an empty object, then TS responses with an error

  1. const errors: Record&lt;keyof FormValues, string | undefined&gt; = {}
  2. /*
  3. Type &#39;{}&#39; is missing the following properties
  4. from type &#39;Record&lt;keyof FormValues, string | undefined&gt;&#39;: name, description, expirationDate
  5. */

how to change the error type to reflect my need in having an object with the fields defined as a string or non-exist:

  1. const validate = values =&gt; {
  2. const errors: ??? = {}
  3. if (!values.name) errors.name = &#39;Required&#39;
  4. if (!values.description) errors.description = &#39;Required&#39;
  5. if (!values.expirationDate) errors.expirationDate = &#39;Required&#39;
  6. return errors
  7. }

答案1

得分: 1

请看这个(你的类型):

  1. type errorType1 = Record<keyof FormValues, string | undefined>;

等同于

  1. type errorType1 = {
  2. name: string | undefined;
  3. description: string | undefined;
  4. expirationDate: string | undefined;
  5. }

然后看看

  1. type errorType2 = Partial<Record<keyof FormValues, string>>;

等同于

  1. type errorType2 = {
  2. name?: string | undefined;
  3. description?: string | undefined;
  4. expirationDate?: string | undefined;
  5. }

使用错误类型2,您可以忘记键。但使用第一个类型则不行。

您可能需要的是:

  1. const errors: Partial<Record<keyof FormValues, string>> = {}
英文:

Take a look at this (your type) :

  1. type errorType1 = Record&lt;keyof FormValues, string | undefined&gt;

wich is equivalent to

  1. type errorType1 = {
  2. name: string | undefined;
  3. description: string | undefined;
  4. expirationDate: string | undefined;
  5. }

Then look at

  1. type errorType2 = Partial&lt;Record&lt;keyof FormValues, string&gt;&gt;

which is equivalent to

  1. type errorType2 = {
  2. name?: string | undefined;
  3. description?: string | undefined;
  4. expirationDate?: string | undefined;
  5. }

With error type 2, you are allowed to forget keys. With the first you can't.

What you need is probably :

  1. const errors: Partial&lt;Record&lt;keyof FormValues, string&gt;&gt; = {}

答案2

得分: 1

您可以使用类型实用程序Partial<Type>来使类型的键变为可选的:

  1. const errors: Partial<Record<keyof FormValues, string>> = {};

以下是一个完整的示例,基于您问题中的代码:

TS Playground

  1. const formValueKeys = [
  2. "name",
  3. "description",
  4. "expirationDate",
  5. ] as const satisfies readonly (keyof FormValues)[];
  6. const validate = (values: FormValues) => {
  7. const errors: Partial<Record<keyof FormValues, string>> = {};
  8. for (const key of formValueKeys) {
  9. if (!values[key]) errors[key] = "Required";
  10. }
  11. return errors;
  12. };
英文:

You can use the type utility Partial&lt;Type&gt; to make the keys of a type optional:

  1. const errors: Partial&lt;Record&lt;keyof FormValues, string&gt;&gt; = {};

Here's a complete example based on the code from your question:

TS Playground

  1. const formValueKeys = [
  2. &quot;name&quot;,
  3. &quot;description&quot;,
  4. &quot;expirationDate&quot;,
  5. ] as const satisfies readonly (keyof FormValues)[];
  6. const validate = (values: FormValues) =&gt; {
  7. const errors: Partial&lt;Record&lt;keyof FormValues, string&gt;&gt; = {};
  8. for (const key of formValueKeys) {
  9. if (!values[key]) errors[key] = &quot;Required&quot;;
  10. }
  11. return errors;
  12. };

huangapple
  • 本文由 发表于 2023年2月8日 22:41:25
  • 转载请务必保留本文链接:https://go.coder-hub.com/75387390.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定