- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
在您可能需要修改可序列化类并保持向后兼容性的情况下,我有一个关于 Java 序列化的问题。
我有深厚的 C# 经验,所以请允许我将 Java 与 .NET 进行比较。
在我的 Java 场景中,我需要使用 Java 的运行时序列化机制序列化一个对象,并将二进制数据存储在永久存储中以供将来重用这些对象。 问题是,在未来,类可能会发生变化。可以添加或删除字段。
除了这篇关于如何 not to program in Java 的精彩文章外,我对 Java 序列化一无所知。在处理序列化时。正如我想象的那样 (d),serialVersionUID 在 Java 序列化中起着关键作用,这就是我需要您帮助的地方。
除了文章的示例(我知道这是错误的编码),当我修改类后 Eclipse 要求更新它时,该字段是否不被修改?
我记得在 .NET 世界中,当我添加新字段时,我必须将 [OptionalField]
属性添加到该字段以获得向后兼容性,因此 CLR 在旧的序列化中不需要它数据。此外,当我需要弃用某个字段时,我必须只删除公共(public)方法而不是私有(private)字段。
最佳连载指南是什么?
谢谢。
[添加] 这是一个例子。假设我有 Foo 类
public class Foo {
private String bar;
}
然后我改成:
public class Foo {
private String bar;
private Integer eggs;
}
这两个版本之间是否存在兼容性问题?如果我在编译“newFoo”时反序列化“oldFoo”,eggs 是否等于 null 或抛出异常?显然,我更喜欢第一个!!
最佳答案
假设您有一个类 MyClass
,并且您希望确保后续的序列化兼容性,或者至少确保您不会无意中更改其序列化形式。您可以使用 GS Collections test utilities 中的 Verify.assertSerializedForm()
在大多数情况下。
首先编写一个测试,断言您的类的 serialVersionUID
为 0L
并且序列形式为空字符串。
@Test
public void serialized_form()
{
Verify.assertSerializedForm(
0L,
"",
new MyClass());
}
运行测试。它会失败,因为字符串表示 Base64 编码并且永远不会为空。
org.junit.ComparisonFailure: Serialization was broken. <Click to see difference>
当您单击以查看差异时,您将看到实际的 Base64 编码。将其粘贴到空字符串中。
@Test
public void serialized_form()
{
Verify.assertSerializedForm(
0L,
"rO0ABXNyAC9jYXJhbWVsa2F0YS5zaHVrbmlfZ29lbHZhLkV4ZXJjaXNlOVRlc3QkTXlDbGFzc56U\n"
+ "hVp0q+1aAgAAeHA=",
new MyClass());
}
重新运行测试。它很可能会再次失败并显示类似这样的错误消息。
java.lang.AssertionError: serialVersionUID's differ expected:<0> but was:<-7019839295612785318>
将新的 serialVersionUID 粘贴到测试中以代替 0L。
@Test
public void serialized_form()
{
Verify.assertSerializedForm(
-7019839295612785318L,
"rO0ABXNyAC9jYXJhbWVsa2F0YS5zaHVrbmlfZ29lbHZhLkV4ZXJjaXNlOVRlc3QkTXlDbGFzc56U\n"
+ "hVp0q+1aAgAAeHA=",
new MyClass());
}
测试现在将通过,直到您更改序列化表单。如果您不小心破坏了测试(更改序列化形式),首先要做的是检查您是否在 Serializable 类中指定了 serialVerionUID
。如果您将其遗漏,JVM 会为您生成它并且它非常脆弱。
public class MyClass implements Serializable
{
private static final long serialVersionUID = -7019839295612785318L;
}
如果测试仍然失败,您可以尝试通过将新字段标记为 transient 、使用 writeObject() 等完全控制序列化表单来恢复序列化表单。
如果测试仍然失败,您必须决定是查找并还原破坏序列化的更改,还是将您的更改视为对序列化形式的有意更改。
当您故意更改序列化形式时,您将需要更新 Base64 字符串以使测试通过。当您这样做时,至关重要同时更改 serialVersionUID
是很重要的。选择什么数字并不重要,只要它是您以前从未在类里面使用过的数字即可。惯例是将其更改为 2L
,然后是 3L
,依此类推。如果您从随机生成的 serialVersionUID
(如 -7019839295612785318L
在示例中),您仍应将数字更改为 2L
,因为它仍然是序列化形式的第二个版本。
注意:我是 GS 集合的开发人员。
关于java - 字段添加或删除等类更改是否保持 Serializable 的向后兼容性?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6374646/
这不是我的专业领域,所以我希望问的是正确的问题。 我们有一台滚动租用的服务器。旧服务器是32位windows服务器,新服务器是64位windows 2008 R2 SP1。 其中一个 Web 应用程序
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我已将 Oracle 数据库从 10g 迁移到 12c。从 12c 开始,oracle 不支持 PLSQL_V2_COMPATIBILITY 参数。 该参数用于: https://www.safari
开发环境db server为SqlServer 2005(开发版) 有什么方法可以确保我的 SQL 查询将在 SqlServer 2000 中运行? 此数据库设置为兼容级别“SQL Server 20
我有一个这种形式的类: public class Foo implements Serializable { private static final long serialVersionUI
我有以下代码来隐藏状态栏,取自 http://developer.android.com/training/system-ui/status.html和 Hide status bar android
我正在尝试测试 .prop() 是否方法存在于当前包含的 jQuery 中(出于兼容性原因): if(typeof $.prop === 'function') 我期望上面的条件是 true对于 jQ
当收到新消息时,我在项目中使用BroadcastChannel更改所有选项卡的标题。 问题在于它仅适用于chrome和firefox。因此,我决定使用localStorage创建一个Broadcast
我正在使用一个函数通过 FTP 将一个文件上传到我的服务器。这是我的代码并且工作正常但是创建的文件 example.json 不兼容 UTF8,因为它有 Atlético 而不是 Atlético 例
我正在使用兼容性类来构建用户代理字符串: public abstract class Compatibility { private static int sdkInt = 0; pr
我需要实现一个 C 例程来(解)压缩 gzip 格式的文件。 谁能举个例子? 我试过 zlib,但它似乎不兼容。 谢谢。 最佳答案 zlib 与 gzip 文件完全兼容,但您需要确保您使用的是面向 g
我正在使用以下 CSS 代码,它与 Chrome 完美兼容,但与 IE 浏览器不兼容 .collapse{ display:block; } .collapse + input[type="c
我的应用程序以 android Sdk 的 v10 为目标,但具有 minSdkVersion 的 v6。默认情况下,“match_parent”属性将用于宽度或高度。我应该为 fill_parent
我正在阅读有关 dynamic_cast 的内容,然后我遇到了以下语句 ( from cplusplus.com ): Compatibility note: This type of dynamic
我正在尝试在 Linux 下使用 QtCreator 构建一个用 VS 2008 编写的项目,但我遇到了很多错误: /home/ga/dev/CppGroup/MonteCarlo/main.cpp:
因此,我正在构建一个网站,用户可以在该网站上上传观看视频。我正在使用标准的 HTML5 视频播放器 ( ... )目前,我使用多个来源:MP4、OGG 和 WEBM,以实现跨浏览器兼容性 由于维护三
mozilla 和其他浏览器是否有类似-webkit-box-reflect 的属性?我无法在谷歌上找到哪些其他浏览器支持这个。因此,如果有人可以告诉我或给我链接,那就太好了。 最佳答案 这不仅可以使
我定义了一个自定义的 ValidateSet 参数属性,如下所示: Class MyValidValuesAttribute : System.Management.Automation.IValid
我使用 .net 4.0、linq 等编写 winforms 应用程序。它可以在带有 .net 2.0 的机器上运行吗? 最佳答案 不,不会。为 Framework 4.0 版编译的应用程序将要求该框
我如何专门检查 @keyframes translate3d 动画 与浏览器的兼容性? 请不要关闭这个问题,因为在问这个问题之前我已经尝试了很多 stackoverflow 解决方案。 我想检查我的网
我是一名优秀的程序员,十分优秀!