【问题标题】:How to parse JSON string in Typescript如何在 Typescript 中解析 JSON 字符串
【发布时间】:2016-12-05 22:36:40
【问题描述】:

有没有办法在 Typescript 中将字符串解析为 JSON。
示例:在 JS 中,我们可以使用JSON.parse()。 Typescript中是否有类似的功能?

我有一个 JSON 对象字符串如下:

{"name": "Bob", "error": false}

【问题讨论】:

  • 在其主页上,它说“TypeScript 是 JavaScript 的类型化超集,可编译为纯 JavaScript”。 JSON.parse() 函数应该可以正常使用。
  • 我正在使用 Atom 文本编辑器,当我执行 JSON.parse 时,我收到错误:“{}”类型的参数不可分配给“字符串”类型的参数
  • 这是一个非常基本的问题,对某些人来说可能看起来微不足道,但它仍然是一个有效的问题,并且在 SO(我没有)中找不到等效的问题,所以没有不让问题继续运行的真正原因,在我看来也不应该被否决。
  • @SanketDeshpande 当您使用JSON.parse 时,您会得到一个对象,而不是string(有关更多信息,请参阅我的答案)。如果你想把一个对象变成一个字符串,那么你需要使用JSON.stringify
  • 实际上这不是一个简单的问题,原因有两个。首先, JSON.parse() 不会返回相同类型的对象 - 它会匹配某些接口,但不会出现任何智能,例如访问器。此外,我们肯定希望 SO 成为人们在谷歌搜索时去的地方吗?

标签: javascript json string typescript


【解决方案1】:

Typescript 是 javascript 的(超集),因此您只需像在 javascript 中一样使用 JSON.parse

let obj = JSON.parse(jsonString);

只有在打字稿中,您才能为结果对象指定类型:

interface MyObj {
    myString: string;
    myNumber: number;
}

let obj: MyObj = JSON.parse('{ "myString": "string", "myNumber": 4 }');
console.log(obj.myString);
console.log(obj.myNumber);

(code in playground)

【讨论】:

  • 如何验证输入是否有效(类型检查,打字稿的目的之一)?将输入 '{ "myString": "string", "myNumber": 4 }' 替换为 '{ "myString": "string", "myNumberBAD": 4 }' 不会失败,并且 obj.myNumber 将返回 undefined。
  • @DavidPortabella 您不能对字符串的内容进行类型检查。这是一个运行时问题,类型检查是针对编译时间的
  • 好的。我如何验证 typescript obj 在运行时满足其接口?也就是说,myNumber 在此示例中不是未定义的。例如,在 Scala Play 中,您将使用 Json.parse(text).validate[MyObj]playframework.com/documentation/2.6.x/ScalaJson你怎么能在打字稿中做同样的事情(也许有一个外部库可以这样做?)?
  • @DavidPortabella 没有办法做到这一点,不容易,因为在运行时 MyObj 不存在。 SO中有很多关于这个主题的其他线程,例如:Check if an object implements an interface at runtime with TypeScript
  • 好的,谢谢。每天我都更相信使用 scalajs。
【解决方案2】:

类型安全JSON.parse

您可以继续使用JSON.parse,因为 TS 是 JS 超集。还有一个问题:JSON.parse 返回any,这破坏了类型安全。以下是更强类型的两个选项:

1。用户定义的类型保护 (playground)

自定义type guards 是最简单的解决方案,通常足以进行外部数据验证:

// For example, you expect to parse a given value with `MyType` shape
type MyType = { name: string; description: string; }

// Validate this value with a custom type guard
function isMyType(o: any): o is MyType {
  return "name" in o && "description" in o
}

扩展

围绕JSON.parse 创建一个通用包装器,它将一个类型保护作为输入并返回解析后的类型值或错误结果:

const safeJsonParse = <T>(guard: (o: any) => o is T) => 
  (text: string): ParseResult<T> => {
    const parsed = JSON.parse(text)
    return guard(parsed) ? { parsed, hasError: false } : { hasError: true }
  }

type ParseResult<T> =
  | { parsed: T; hasError: false; error?: undefined }
  | { parsed?: undefined; hasError: true; error?: unknown }

使用示例:

const json = '{ "name": "Foo", "description": "Bar" }';
const result = safeJsonParse(isMyType)(json) // result: ParseResult<MyType>
if (result.hasError) {
  console.log("error :/")  // further error handling here
} else {
  console.log(result.parsed.description) // result.parsed now has type `MyType`
}

safeJsonParse 可能会扩展为 fail fast 或 try/catch JSON.parse 错误。

2。外部库

如果您需要验证许多不同的值,手动编写类型保护函数会变得很麻烦。有一些库可以帮助完成这项任务 - 示例(没有完整列表):

更多信息

【讨论】:

    【解决方案3】:

    如果您希望 JSON 具有经过验证的 Typescript 类型,则需要自己进行验证工作。这不是什么新鲜事。在纯 Javascript 中,您也需要这样做。

    验证

    我喜欢将我的验证逻辑表达为一组“转换”。我将Descriptor 定义为转换图:

    type Descriptor<T> = {
      [P in keyof T]: (v: any) => T[P];
    };
    

    然后我可以创建一个函数,将这些转换应用于任意输入:

    function pick<T>(v: any, d: Descriptor<T>): T {
      const ret: any = {};
      for (let key in d) {
        try {
          const val = d[key](v[key]);
          if (typeof val !== "undefined") {
            ret[key] = val;
          }
        } catch (err) {
          const msg = err instanceof Error ? err.message : String(err);
          throw new Error(`could not pick ${key}: ${msg}`);
        }
      }
      return ret;
    }
    

    现在,我不仅在验证我的 JSON 输入,而且还在构建一个 Typescript 类型。上述泛型类型确保结果从您的“转换”中推断出类型。

    如果转换引发错误(这是您实现验证的方式),我喜欢用另一个错误包装它,显示哪个键导致错误。

    用法

    在您的示例中,我将按如下方式使用:

    const value = pick(JSON.parse('{"name": "Bob", "error": false}'), {
      name: String,
      error: Boolean,
    });
    

    现在value 将被输入,因为StringBoolean 在接受输入并返回输入输出的意义上都是“转换器”。

    此外,value实际上是那种类型。换句话说,如果name 实际上是123,它将被转换为"123",这样你就有了一个有效的字符串。这是因为我们在运行时使用了String,这是一个接受任意输入并返回string 的内置函数。

    你可以看到这个工作here。尝试以下方法来说服自己:

    • 将鼠标悬停在const value 定义上以查看弹出框显示正确的类型。
    • 尝试将"Bob" 更改为123 并重新运行示例。在您的控制台中,您将看到名称已正确转换为字符串 "123"

    【讨论】:

    • 你举了一个例子,“如果name实际上是123,它将被转换为"123"。这似乎是不正确的。我的value回来{name: 123..不是{name:"123".. 当我复制粘贴您的所有代码并进行更改时。
    • 很奇怪,它对我有用。在这里试试:typescriptlang.org/play/index.html(使用123 而不是"Bob")。
    • 我认为您不需要定义 Transformed 类型。你可以使用Objecttype Descriptor&lt;T extends Object&gt; = { ... };
    • 谢谢@lovasoa,你是对的。 Transformed 类型完全没有必要。我已经相应地更新了答案。
    • 如果您真的想验证 JSON 对象是否具有正确的类型,您将希望将 123 自动转换为字符串 "123",因为它是 JSON 对象中的一个数字。
    【解决方案4】:

    有一个很棒的库ts-json-object

    在您的情况下,您需要运行以下代码:

    import {JSONObject, required} from 'ts-json-object'
    
    class Response extends JSONObject {
        @required
        name: string;
    
        @required
        error: boolean;
    }
    
    let resp = new Response({"name": "Bob", "error": false});
    

    这个库会在解析前验证json

    【讨论】:

      【解决方案5】:

      使用 app.quicktype.io 在 TypeScript 中安全地解析 JSON。稍后将对此进行更多介绍。 JSON.parse() 返回类型 any 并且在“快乐路径”中就足够了,但可能导致与下游类型安全相关的错误,这违背了 TypeScript 的目的。例如:

      interface User {
        name: string,
        balance: number
      }
      
      const json = '{"name": "Bob", "balance": "100"}' //note the string "100"
      const user:User = JSON.parse(json)
      
      const newBalance = user.balance + user.balance * 0.05 //should be 105 after interest
      console.log(newBalance ) //but it ends up as 1005 which is clearly wrong
      

      所以让 quicktype 完成繁重的工作并生成代码。将下面的字符串复制并粘贴到 quicktype 中。

      {
        "name": "Bob",
        "balance": 100
      }
      

      确保选择 TypeScript 作为语言并启用“在运行时验证 JSON.parse 结果”

      现在我们可以在解析时防御性地处理异常(如果有),并防止下游发生错误。

      import { Convert, User } from "./user";
      
      const json =
        '{"firstName": "Kevin", "lastName": "Le", "accountBalance": "100"}';
      
      try {
        const user = Convert.toUser(json);
        console.log(user);
      } catch (e) {
        console.log("Handle error", e);
      }
      

      user.ts是quicktype生成的文件。

      // To parse this data:
      //
      //   import { Convert, User } from "./file";
      //
      //   const user = Convert.toUser(json);
      //
      // These functions will throw an error if the JSON doesn't
      // match the expected interface, even if the JSON is valid.
      
      export interface User {
          name:    string;
          balance: number;
      }
      
      // Converts JSON strings to/from your types
      // and asserts the results of JSON.parse at runtime
      export class Convert {
          public static toUser(json: string): User {
              return cast(JSON.parse(json), r("User"));
          }
      
          public static userToJson(value: User): string {
              return JSON.stringify(uncast(value, r("User")), null, 2);
          }
      }
      
      function invalidValue(typ: any, val: any, key: any = ''): never {
          if (key) {
              throw Error(`Invalid value for key "${key}". Expected type ${JSON.stringify(typ)} but got ${JSON.stringify(val)}`);
          }
          throw Error(`Invalid value ${JSON.stringify(val)} for type ${JSON.stringify(typ)}`, );
      }
      
      function jsonToJSProps(typ: any): any {
          if (typ.jsonToJS === undefined) {
              const map: any = {};
              typ.props.forEach((p: any) => map[p.json] = { key: p.js, typ: p.typ });
              typ.jsonToJS = map;
          }
          return typ.jsonToJS;
      }
      
      function jsToJSONProps(typ: any): any {
          if (typ.jsToJSON === undefined) {
              const map: any = {};
              typ.props.forEach((p: any) => map[p.js] = { key: p.json, typ: p.typ });
              typ.jsToJSON = map;
          }
          return typ.jsToJSON;
      }
      
      function transform(val: any, typ: any, getProps: any, key: any = ''): any {
          function transformPrimitive(typ: string, val: any): any {
              if (typeof typ === typeof val) return val;
              return invalidValue(typ, val, key);
          }
      
          function transformUnion(typs: any[], val: any): any {
              // val must validate against one typ in typs
              const l = typs.length;
              for (let i = 0; i < l; i++) {
                  const typ = typs[i];
                  try {
                      return transform(val, typ, getProps);
                  } catch (_) {}
              }
              return invalidValue(typs, val);
          }
      
          function transformEnum(cases: string[], val: any): any {
              if (cases.indexOf(val) !== -1) return val;
              return invalidValue(cases, val);
          }
      
          function transformArray(typ: any, val: any): any {
              // val must be an array with no invalid elements
              if (!Array.isArray(val)) return invalidValue("array", val);
              return val.map(el => transform(el, typ, getProps));
          }
      
          function transformDate(val: any): any {
              if (val === null) {
                  return null;
              }
              const d = new Date(val);
              if (isNaN(d.valueOf())) {
                  return invalidValue("Date", val);
              }
              return d;
          }
      
          function transformObject(props: { [k: string]: any }, additional: any, val: any): any {
              if (val === null || typeof val !== "object" || Array.isArray(val)) {
                  return invalidValue("object", val);
              }
              const result: any = {};
              Object.getOwnPropertyNames(props).forEach(key => {
                  const prop = props[key];
                  const v = Object.prototype.hasOwnProperty.call(val, key) ? val[key] : undefined;
                  result[prop.key] = transform(v, prop.typ, getProps, prop.key);
              });
              Object.getOwnPropertyNames(val).forEach(key => {
                  if (!Object.prototype.hasOwnProperty.call(props, key)) {
                      result[key] = transform(val[key], additional, getProps, key);
                  }
              });
              return result;
          }
      
          if (typ === "any") return val;
          if (typ === null) {
              if (val === null) return val;
              return invalidValue(typ, val);
          }
          if (typ === false) return invalidValue(typ, val);
          while (typeof typ === "object" && typ.ref !== undefined) {
              typ = typeMap[typ.ref];
          }
          if (Array.isArray(typ)) return transformEnum(typ, val);
          if (typeof typ === "object") {
              return typ.hasOwnProperty("unionMembers") ? transformUnion(typ.unionMembers, val)
                  : typ.hasOwnProperty("arrayItems")    ? transformArray(typ.arrayItems, val)
                  : typ.hasOwnProperty("props")         ? transformObject(getProps(typ), typ.additional, val)
                  : invalidValue(typ, val);
          }
          // Numbers can be parsed by Date but shouldn't be.
          if (typ === Date && typeof val !== "number") return transformDate(val);
          return transformPrimitive(typ, val);
      }
      
      function cast<T>(val: any, typ: any): T {
          return transform(val, typ, jsonToJSProps);
      }
      
      function uncast<T>(val: T, typ: any): any {
          return transform(val, typ, jsToJSONProps);
      }
      
      function a(typ: any) {
          return { arrayItems: typ };
      }
      
      function u(...typs: any[]) {
          return { unionMembers: typs };
      }
      
      function o(props: any[], additional: any) {
          return { props, additional };
      }
      
      function m(additional: any) {
          return { props: [], additional };
      }
      
      function r(name: string) {
          return { ref: name };
      }
      
      const typeMap: any = {
          "User": o([
              { json: "name", js: "name", typ: "" },
              { json: "balance", js: "balance", typ: 0 },
          ], false),
      };
      

      【讨论】:

        【解决方案6】:

        您还可以使用执行 json 类型验证的库,例如 Sparkson。它们允许你定义一个 TypeScript 类,你想解析你的响应,在你的情况下它可能是:

        import { Field } from "sparkson";
        class Response {
           constructor(
              @Field("name") public name: string,
              @Field("error") public error: boolean
           ) {}
        }
        

        该库将验证 JSON 负载中是否存在必填字段以及它们的类型是否正确。它还可以进行大量的验证和转换。

        【讨论】:

        • 您应该提到,您是上述库的主要贡献者。
        【解决方案7】:

        JSON.parse 在 TypeScript 中可用,所以你可以直接使用它:

        JSON.parse('{"name": "Bob", "error": false}') // Returns a value of type 'any'
        

        但是,您通常希望在解析 JSON 对象的同时确保它与特定类型匹配,而不是处理 any 类型的值。在这种情况下,您可以定义如下函数:

        function parse_json<TargetType extends Object>(
          json: string,
          type_definitions: { [Key in keyof TargetType]: (raw_value: any) => TargetType[Key] }
        ): TargetType {
          const raw = JSON.parse(json); 
          const result: any = {};
          for (const key in type_definitions) result[key] = type_definitions[key](raw[key]);
          return result;
        }
        

        此函数接受一个 JSON 字符串和一个对象,该对象包含加载您正在创建的对象的每个字段的各个函数。你可以这样使用它:

        const value = parse_json(
          '{"name": "Bob", "error": false}',
          { name: String, error: Boolean, }
        );
        

        【讨论】:

          【解决方案8】:

          TS 有一个 JavaScript 运行时

          Typescript 有一个 JavaScript 运行时,因为它被编译为 JS。这意味着作为语言的一部分内置的 JS 对象,例如 JSONObjectMath 在 TS 中也可用。因此我们可以直接使用JSON.parse方法来解析JSON字符串。

          示例:

          const JSONStr = '{"name": "Bob", "error": false}'
          
          // The JSON object is part of the runtime
          const parsedObj = JSON.parse(JSONStr);
          
          console.log(parsedObj);
          // [LOG]: {
          //   "name": "Bob",
          //   "error": false
          // } 
          
          // The Object object is also part of the runtime so we can use it in TS
          const objKeys = Object.keys(parsedObj);
          
          console.log(objKeys);
          // [LOG]: ["name", "error"] 
          

          现在唯一的问题是 parsedObj 的类型为any,这在 TS 中通常是一种不好的做法。如果我们使用类型保护,我们可以键入对象。这是一个例子:

          const JSONStr = '{"name": "Bob", "error": false}'
          const parsedObj = JSON.parse(JSONStr);
          
          interface nameErr {
            name: string;
            error: boolean;
          }
          
          function isNameErr(arg: any): arg is nameErr {
            if (typeof arg.name === 'string' && typeof arg.error === 'boolean') {
              return true;
            } else {
              return false;
            }
          }
          
          if (isNameErr(parsedObj)) {
            // Within this if statement parsedObj is type nameErr;
            parsedObj
          }
          

          【讨论】:

            【解决方案9】:

            嘿,如果你对你的 json 对象进行 typeof 操作,结果会是 typescript 的字符串。你可以在这里阅读更多内容:Typescript: difference between String and string

            所以只要试试这个方法就行了-

            JSON.parse(String({"name": "Bob", "error": false}))
            

            【讨论】:

              【解决方案10】:

              是的,这在 TypeScript 中有点棘手,但是您可以像下面这样执行以下示例

              let decodeData  =  JSON.parse(`${jsonResponse}`);

              【讨论】:

              • 没有正确回答问题,错误的代码/示例(${jsonResponse} 没有意义)
              猜你喜欢
              • 2011-05-20
              • 2014-09-10
              • 1970-01-01
              • 1970-01-01
              • 1970-01-01
              • 1970-01-01
              • 1970-01-01
              • 1970-01-01
              • 2023-03-09
              相关资源
              最近更新 更多