为什么Java有瞬时字段?


当前回答

根据谷歌短暂的意义==只持续很短的时间;短暂的

现在,如果您想在java中实现任何瞬态,请使用瞬态关键字。

Q: 在哪里使用瞬态?

A: 通常在java中,我们可以通过在变量中获取数据并将这些变量写入文件来将数据保存到文件中,这个过程称为序列化。现在,如果我们想避免将变量数据写入文件,我们会将该变量设置为瞬时变量。

transient int result=10;

注意:瞬态变量不能是局部变量。

其他回答

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关键字阻止某些字段序列化的场景。

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

序列化和反序列化是对称过程,如果不是,那么就不能期望结果被确定,在大多数情况下,未确定的值是没有意义的;串行化和反串行化是幂等的,这意味着您可以根据需要多次进行串行化,结果是相同的。

因此,如果Object可以存在于内存上,但不存在于磁盘上,那么Object就不能串行化,因为反序列化时计算机无法还原内存映射。例如,不能序列化Stream对象。

不能序列化Connection对象,因为它的状态也依赖于远程计算机。

允许您定义不希望序列化的变量。

在对象中,您可能有不希望序列化/持久化的信息(可能是对父工厂对象的引用),或者序列化没有意义。将这些标记为“瞬时”意味着序列化机制将忽略这些字段。

本机java系统以外的序列化系统也可以使用此修饰符。例如,Hibernate不会持久化标记为@Transient或Transient修饰符的字段。Terracotta也尊重这个修饰语。

我认为修饰语的比喻意义是“此字段仅供内存使用。不要以任何方式将其持久化或移出此特定VM。它不可移植”。也就是说,你不能依赖它在另一个VM内存空间中的值。与volatile非常类似,意味着您不能依赖特定的内存和线程语义。

简单地说,transient java关键字保护字段不被序列化为其非transient字段计数器部分。

在这个代码片段中,我们的抽象类BaseJob实现了Serializable接口,我们从BaseJob扩展,但不需要序列化远程和本地数据源;仅序列化organizationName和isSynced字段。

public abstract class BaseJob implements Serializable{
   public void ShouldRetryRun(){}
}

public class SyncOrganizationJob extends BaseJob {

   public String organizationName;
   public Boolean isSynced

   @Inject transient RemoteDataSource remoteDataSource;
   @Inject transient LocalDaoSource localDataSource;

   public SyncOrganizationJob(String organizationName) {
     super(new 
         Params(BACKGROUND).groupBy(GROUP).requireNetwork().persist());

      this.organizationName = organizationName;
      this.isSynced=isSynced;

   }
}