当前位置: 首页 > 面试题库 >

为什么JSON.stringify不序列化原型值?

袁致远
2023-03-14
问题内容

最近,我一直在进行相当多的JSON解析,并在Node.js和浏览器中传递Javascript,并遇到了这个难题。

我使用构造函数创建的任何对象都无法通过JSON.stringify进行完全序列化,除非我单独初始化构造函数中的所有值!这意味着我的原型在设计这些类时实际上变得毫无用处。

有人可以阐明为什么以下内容没有按我期望的那样进行序列化吗?

var ClassA = function () { this.initialisedValue = "You can see me!" };
ClassA.prototype = { initialisedValue : "You can't see me!", uninitialisedValue : "You can't see me!" };
var a = new ClassA();
var a_string = JSON.stringify(a);

怎么了:

a_string == {“ initializedValue”:“您可以看到我!” }

我期望:

a_string == {“ initializedValue”:“你可以看到我!”,“ uninitializedValue”:“你不能看到我!” }

更新(01-10-2019):

终于注意到@ncardeli的答案,它确实使我们可以做以下事情来达到我的上述要求(在2019年!):

更换 var a_string = JSON.stringify(a);

var a_string = JSON.stringify(a, Object.keys(ClassA.prototype));

完整代码:

var ClassA = function () { this.initialisedValue = "You can see me!" };
ClassA.prototype = { initialisedValue : "You can't see me!", uninitialisedValue : "You can't see me!" };
var a = new ClassA();
var a_string = JSON.stringify(a, Object.keys(ClassA.prototype));

console.log(a_string)

问题答案:

仅仅是因为这就是JSON的工作方式。根据ES5规范:

令K为内部字符串列表,该字符串列表由[[Enumerable]]属性为true的value 的所有 自身属性 的名称组成。

这是有道理的,因为JSON规范中没有用于保留信息的机制,如果包含继承的属性,则该信息将需要将JSON字符串解析回JavaScript对象中。在您的示例中,该如何解析:

{“ initializedValue”:“你可以看到我!”,“ uninitializedValue”:“你不能看到我!” }

除了具有2个键值对的平面对象之外,没有任何信息可将其解析为其他任何内容。

而且,如果您考虑一下,JSON并非旨在直接映射到JavaScript对象。其他语言必须能够将JSON字符串解析为简单的名称/值对结构。如果JSON字符串包含序列化完整JavaScript作用域链所需的所有信息,则其他语言可能无法将其解析为有用的内容。用Douglas
Crockford在json.org上的话说:

这些[哈希表和数组]是通用数据结构。几乎所有现代编程语言都以一种或另一种形式支持它们。可以与编程语言互换的数据格式也应基于这些结构,这是有道理的。



 类似资料:
  • 问题内容: 我正在使用JavaScript将对象序列化为JSON字符串, 我注意到只有可枚举的对象属性才被序列化: [ 笔 ] 我想知道为什么会这样?我已经通过搜索MDN页面中,json2解析器文档。我找不到任何地方记录此行为。 我怀疑这是使用仅通过[[enumerable]]属性的循环(至少在情况下)的结果。可以使用类似的方法同时返回可枚举和不可枚举的属性。不过,这可能会导致序列化(由于反序列化

  • 问题内容: 我正在使用TypeScript定义一些类,当我创建属性时,它会生成与以下plunkr中的等效项: http://plnkr.co/edit/NXUo7zjJJZaUuyv54TD9i?p=preview 序列化时,它不会输出我刚刚定义的属性。 并通过序列化定义的属性来达到我的期望。(请参阅plunkr输出)。 我的实际问题是:这正常吗? 如果是这样,我如何以最有效的方式解决它? 问题答

  • 我试图在Java中序列化,但在运行我的程序时,我收到一个。 查看类,我注意到它没有实现。 为什么不实现?

  • 如果接口只是一个标记接口,用于在 java 中传递有关类的某种元数据 - 我有点困惑: 在阅读了java的序列化算法(元数据从下到上,然后从上到下的实际实例数据)的过程之后,我无法真正理解哪些数据不能通过该算法进行处理。 简而言之: 哪些数据可能导致? 我怎么知道我不应该为我的类添加子句?

  • 问题内容: 如下代码: 引发以下异常: 我猜内部类具有一个允许对其私有访问的字段和方法的字段。声明内部类static 可以解决它,但是如果需要此访问权限怎么办?有没有一种方法可以在不包含封闭类的情况下序列化非静态内部类,例如通过引用外部类? 编辑:例如,仅在序列化之前才需要访问外部类。好的,编译器不知道这一点,但是我认为这就是关键字存在的原因。 问题答案: 如果InnerClass需要此访问权限怎

  • 在我的代码中,我有以下行: 在野外,它生活在一个枚举中: 我从sonarqube那里得到以下提及: 使“形状”暂时或可序列化。可序列化类中的字段本身必须是可序列化的或瞬态的,即使该类从未显式序列化或反序列化。这是因为在负载下,大多数J2EE应用程序框架都会将对象刷新到磁盘,而一个据称具有非瞬态、非序列化数据成员的可序列化对象可能会导致程序崩溃,并为攻击者打开大门。 据我所知,int[](和int[