Сериализировать вложенные классы машинописного текста в json

Я не могу сериализовать вложенный набор классов в json:

import { readFileSync  } from 'fs'

class Address {
    constructor(
        public readonly street: string,
        public readonly city: string){}
}

class Person {
    constructor(
        public readonly address: Address,
        public readonly name: string){}
}

class School {
    private students = new Map<string, Person>();
    public add(student: Person): School {
        this.students.set(student.name, student);
        return this;
    }
    public toJson(filename: string) {        
    }
}

const p1 = new Person(new Address("5th Ave.", "NYC"), "moish");
const p2 = new Person(new Address("remi", "Boston"), "dave");
const p3 = new Person(new Address("Dart", "Boston"), "uzi");

let school = new School();
school.add(p1).add(p2).add(p3)
console.info(school); // <--- good ! but wrong format 🤓🤓🤓
console.info(JSON.stringify(school)); // doesn't work: {"students":{}} 🥶🥶🥶

И вот как я его компилирую и запускаю:

% npx tsc --target es2022 --moduleResolution node main.ts
% node main.js                                           
School {
  students: Map(3) {
    'moish' => Person { address: [Address], name: 'moish' },
    'dave' => Person { address: [Address], name: 'dave' },
    'uzi' => Person { address: [Address], name: 'uzi' }
  }
}
{"students":{}}
Стоит ли изучать PHP в 2023-2024 годах?
Стоит ли изучать PHP в 2023-2024 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
0
0
62
2
Перейти к ответу Данный вопрос помечен как решенный

Ответы 2

Нет встроенной поддержки сериализации Map в JSON. Но вы можете переопределить метод toJSON.

class School {
  students = new Map();

  add(student) {
    this.students.set(student.name, student)
    return this;
  }

  toJSON() {
    return {
      students: Object.fromEntries(this.students)
    }
  }
}

class Student {
  constructor(name) {
    this.name = name
  }
}

const school = new School().add(new Student('A')).add(new Student('B')).add(new Student('C'))

console.info(JSON.stringify(school))
Ответ принят как подходящий

Правильное имя метода для настройки значения, используемого во время сериализации JSON объекта, — toJSON().

В вашем случае вам просто нужно предоставить сериализуемый тип данных вместо Map , например, объект. Вы можете использовать метод Object.fromEntries() в методе итератора entry() для создания такого объекта:

toJSON (): { students: Record<string, Person> } {
  return { students: Object.fromEntries(this.students.entries()) };
}

Вот полный пример, основанный на коде, который вы показали в своем вопросе:

Игровая площадка ТС

class Address {
  constructor (
    public readonly street: string,
    public readonly city: string,
  ) {}
}

class Person {
  constructor (
    public readonly address: Address,
    public readonly name: string,
  ) {}
}

class School {
  private students = new Map<string, Person>();

  public add (student: Person): School {
    this.students.set(student.name, student);
    return this;
  }

  public toJSON (): { students: Record<string, Person> } {
    return { students: Object.fromEntries(this.students.entries()) };
  }
}

console.clear();
const p1 = new Person(new Address("5th Ave.", "NYC"), "moish");
const p2 = new Person(new Address("remi", "Boston"), "dave");
const p3 = new Person(new Address("Dart", "Boston"), "uzi");

const school = new School();
school.add(p1).add(p2).add(p3);
console.info(JSON.stringify(school)); // {"students":{"moish":{"address":{"street":"5th Ave.","city":"NYC"},"name":"moish"},"dave":{"address":{"street":"remi","city":"Boston"},"name":"dave"},"uzi":{"address":{"street":"Dart","city":"Boston"},"name":"uzi"}}}

Скомпилированный JS из TS Playground:

"use strict";
class Address {
    constructor(street, city) {
        this.street = street;
        this.city = city;
    }
}
class Person {
    constructor(address, name) {
        this.address = address;
        this.name = name;
    }
}
class School {
    constructor() {
        this.students = new Map();
    }
    add(student) {
        this.students.set(student.name, student);
        return this;
    }
    toJSON() {
        return { students: Object.fromEntries(this.students.entries()) };
    }
}
console.clear();
const p1 = new Person(new Address("5th Ave.", "NYC"), "moish");
const p2 = new Person(new Address("remi", "Boston"), "dave");
const p3 = new Person(new Address("Dart", "Boston"), "uzi");
const school = new School();
school.add(p1).add(p2).add(p3);
console.info(JSON.stringify(school)); // {"students":{"moish":{"address":{"street":"5th Ave.","city":"NYC"},"name":"moish"},"dave":{"address":{"street":"remi","city":"Boston"},"name":"dave"},"uzi":{"address":{"street":"Dart","city":"Boston"},"name":"uzi"}}}

Ссылка на тип утилиты Record<Keys, Type>: https://www.typescriptlang.org/docs/handbook/utility-types.html#recordkeys-type

можете ли вы объяснить возвращаемый тип toJSON? я не уверен, что понимаю это

OrenIshShalom 23.11.2022 10:48
^ @OrenIshShalom Я обновил ответ ссылкой на используемый тип утилиты. Тип возвращаемого значения — это объект со свойством students, который представляет собой объект со строковыми ключами и Person значениями.
jsejcksn 23.11.2022 10:51

Другие вопросы по теме