为什么Java有瞬时字段?


当前回答

因为并非所有变量都具有可序列化的性质

其他回答

在我回答这个问题之前,我需要解释序列化,因为如果你理解了它在科学计算机中的含义,你可以很容易地理解这个关键词。

当对象通过网络传输/保存在物理介质(文件,…)上时,该对象必须“序列化”。序列化转换字节状态对象序列。这些字节在网络上发送/保存,并根据这些字节重新创建对象。

例子:

public class Foo implements Serializable 
{
 private String attr1;
 private String attr2;
 ...
}

现在,如果这个类中有一个字段不想传输或保存,可以使用transient关键字

private transient attr2;

这防止在序列化类时包含字段表单。

Java中的transient关键字用于指示字段不应该是序列化(这意味着保存,就像保存到文件)过程的一部分。

摘自Java语言规范,Java SE 7版,第8.3.1.3节。瞬态场:

变量可以标记为瞬态表明它们不是对象的持久状态。

例如,您可能有从其他字段派生的字段,并且只能以编程方式进行,而不是通过序列化来持久化状态。

这里有一个GalleryImage类,它包含一个图像和从该图像派生的缩略图:

class GalleryImage implements Serializable
{
    private Image image;
    private transient Image thumbnailImage;

    private void generateThumbnail()
    {
        // Generate thumbnail.
    }

    private void readObject(ObjectInputStream inputStream)
            throws IOException, ClassNotFoundException
    {
        inputStream.defaultReadObject();
        generateThumbnail();
    }    
}

在本例中,thumbnailImage是通过调用generateThumbnail方法生成的缩略图。

thumbnailImage字段被标记为瞬态,因此只序列化原始图像,而不是同时保存原始图像和缩略图图像。这意味着需要更少的存储空间来保存序列化对象。(当然,这可能是可取的,也可能不是可取的,这取决于系统的要求——这只是一个示例。)

在反序列化时,调用readObject方法以执行将对象状态还原回序列化发生时的状态所需的任何操作。在这里,需要生成缩略图,因此readObject方法被重写,以便通过调用generateThumbnail方法生成缩略图。

有关更多信息,《发现Java序列化API的秘密》一文(最初在Sun开发者网络上提供)中有一节讨论了transient关键字的使用,并介绍了使用transient关键字阻止某些字段序列化的场景。

当您不想共享一些与序列化相关的敏感数据时,需要使用它。

transient用于指示类字段不需要序列化。最好的例子可能是线程字段。通常没有理由序列化线程,因为它的状态非常“特定于流”。

为什么Java中需要瞬态字段?

transient关键字为您提供了对序列化过程的一些控制,并允许您从此过程中排除一些对象财产。序列化过程用于持久化Java对象,主要是为了在传输或不活动时保留它们的状态。有时,不序列化对象的某些属性是有意义的。

应标记哪些字段为瞬态?

既然我们知道了transient关键字和transient字段的用途,那么知道要标记哪些字段是非常重要的。静态字段也不会序列化,因此相应的关键字也会起作用。但这可能会破坏你的课堂设计;这就是瞬时关键字的作用所在。我尽量不允许其值可以从其他字段派生的字段被序列化,所以我将它们标记为暂时的。如果您有一个名为interest的字段,其值可以从其他字段(本金、利率和时间)计算,则无需对其进行序列化。

另一个很好的例子是文章字数统计。若要保存整篇文章,那个么真的不需要保存字数,因为可以在文章“反序列化”时计算字数;记录器实例几乎不需要序列化,因此可以使它们成为暂时的。