Почему у Java есть временные поля?


Ответы:


1673

transientКлючевое слово в Java используется для указания того, что поле не должно быть частью сериализации (спасшая средства, как в файл) процесс.

Из спецификации языка Java, Java SE 7 Edition , раздел 8.3.1.3. transientПоля :

Переменные могут быть помечены, transientчтобы указать, что они не являются частью постоянного состояния объекта.

Например, у вас могут быть поля, которые получены из других полей, и должны выполняться только программно, а не сохраняться в состоянии посредством сериализации.

Вот 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Поле помечается как transient, поэтому только в оригиналеimage сериализация , а не сохраняющиеся как исходное изображения и уменьшенное изображение. Это означает, что для сохранения сериализованного объекта потребуется меньше места. (Конечно, это может или не может быть желательно в зависимости от требований системы - это только пример.)

Во время десериализации readObjectметод вызывается для выполнения любых операций, необходимых для восстановления состояния объекта обратно в состояние, в котором произошла сериализация. Здесь, миниатюра должна быть сгенерирована, поэтому readObjectметод переопределяется, чтобы миниатюра была сгенерирована путем вызова generateThumbnailметода.

Для получения дополнительной информации статья « Раскройте секреты API сериализации Java» (которая первоначально была доступна в Sun Developer Network) содержит раздел, в котором обсуждается использование и представлен сценарий, в котором используется transientключевое слово для предотвращения сериализации определенных полей.


221
Но почему это ключевое слово, а не аннотация @DoNotSerialize?
Элазар Лейбович

333
Я думаю, это относится ко времени, когда в Java не было аннотаций.
Питер Випперманн

46
Я нахожу странным, что сериализуемый является внутренним для Java. Он может быть реализован как интерфейс или абстрактный класс, который требует от пользователей переопределения методов чтения и записи.
Калеб

7
@MJafar: readObject обычно объединен в механизмы десериализации и поэтому вызывается автоматически. Кроме того, во многих случаях вам не нужно переопределять его - реализация по умолчанию делает свое дело.
Майк Адлер

13
@caleb, вероятно, потому что работа с двоичными форматами самостоятельно невероятно болезненна в Java из-за отсутствия целых чисел без знака.
правостороннее

435

Прежде чем понять transientключевое слово, нужно понять концепцию сериализации. Если читатель знает о сериализации, пропустите первый пункт.

Что такое сериализация?

Сериализация - это процесс сохранения состояния объекта. Это означает, что состояние объекта преобразуется в поток байтов, который используется для сохранения (например, хранения байтов в файле) или передачи (например, отправки байтов по сети). Таким же образом, мы можем использовать десериализацию, чтобы вернуть состояние объекта из байтов. Это одна из важных концепций в Java-программировании, потому что сериализация в основном используется в сетевом программировании. Объекты, которые должны быть переданы через сеть, должны быть преобразованы в байты. Для этого каждый класс или интерфейс должен реализовывать Serializableинтерфейс. Это маркерный интерфейс без каких-либо методов.

Теперь, каково transientключевое слово и его цель?

По умолчанию все переменные объекта преобразуются в постоянное состояние. В некоторых случаях вы можете избежать сохранения некоторых переменных, потому что вам не нужно сохранять эти переменные. Таким образом, вы можете объявить эти переменные как transient. Если переменная объявлена ​​как transient, она не будет сохранена. Это основная цель transientключевого слова.

Я хочу объяснить два приведенных выше пункта следующим примером:

package javabeat.samples;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

class NameStore implements Serializable{
    private String firstName;
    private transient String middleName;
    private String lastName;

    public NameStore (String fName, String mName, String lName){
        this.firstName = fName;
        this.middleName = mName;
        this.lastName = lName;
    }

    public String toString(){
        StringBuffer sb = new StringBuffer(40);
        sb.append("First Name : ");
        sb.append(this.firstName);
        sb.append("Middle Name : ");
        sb.append(this.middleName);
        sb.append("Last Name : ");
        sb.append(this.lastName);
        return sb.toString();
    }
}

public class TransientExample{
    public static void main(String args[]) throws Exception {
        NameStore nameStore = new NameStore("Steve", "Middle","Jobs");
        ObjectOutputStream o = new ObjectOutputStream(new FileOutputStream("nameStore"));
        // writing to object
        o.writeObject(nameStore);
        o.close();

        // reading from object
        ObjectInputStream in = new ObjectInputStream(new FileInputStream("nameStore"));
        NameStore nameStore1 = (NameStore)in.readObject();
        System.out.println(nameStore1);
    }
}

И вывод будет следующим:

First Name : Steve
Middle Name : null
Last Name : Jobs

Второе имя объявляется как transient, поэтому оно не будет храниться в постоянном хранилище.

Источник


26
Этот пример взят из этого кода, вы можете прочитать его здесь: javabeat.net/2009/02/what-is-transient-keyword-in-java
Кришна

11
Эта часть кажется мне странной и, возможно, сбивающей с толку: «Это означает, что состояние объекта преобразуется в поток байтов и сохраняется в файле ». Мне кажется, что в большинстве случаев сериализация не связана с записью в файл (наглядный пример: примеры сетевых подключений)
Garcia Hurtado

5
Пример плохой, так как отчество явно не является временным свойством.
Рафаэль

1
@ Рафаэль Для меня пример полезен и, по крайней мере, объясняет концепцию. Не могли бы вы привести какой-нибудь лучший пример, если вы в курсе?
Chaklader Asfak Arefe

@Yoda У нас может быть ссылка на NameFormatterподержанную машину toString(). Или что-нибудь еще, связанное с настройкой, просмотром или бизнес-логикой, а не с данными.
Рафаэль

84

Чтобы позволить вам определить переменные, которые вы не хотите сериализовать.

В объекте у вас может быть информация, которую вы не хотите сериализовать / сохранять (возможно, ссылка на родительский объект фабрики), или, возможно, нет смысла сериализовать. Отметив их как «переходные» означает, что механизм сериализации будет игнорировать эти поля.


36

Мой маленький вклад:

Что такое переходное поле?
По сути, любое поле, измененное с помощью transientключевого слова, является временным полем.

Зачем нужны переходные поля в Java? Ключевое слово дает некоторый контроль над процессом сериализации и позволяет исключить некоторые свойства объектов из этого процесса. Процесс сериализации используется для сохранения Java-объектов, главным образом для того, чтобы их состояния могли сохраняться во время их передачи или неактивности. Иногда имеет смысл не сериализовать определенные атрибуты объекта. Какие поля следует пометить как переходные? Теперь мы знаем цель
transient


transientключевые слова и переходные поля, важно знать, какие поля помечать переходные. Статические поля также не сериализуются, поэтому соответствующее ключевое слово также поможет. Но это может испортить дизайн вашего класса; Вот где transientключевое слово приходит на помощь. Я стараюсь не допустить сериализации полей, значения которых могут быть получены из других, поэтому я отмечаю их как временные. Если у вас есть поле с именем interest, значение которого можно вычислить из других полей ( principal, rate& time), нет необходимости его сериализации.

Еще один хороший пример - количество слов в статье. Если вы сохраняете всю статью, вам не нужно сохранять количество слов, потому что оно может быть вычислено, когда статья будет «десериализована». Или подумайте о регистраторах;Logger экземпляры почти никогда не нужно сериализовать, поэтому их можно сделать временными.


63
Ваше «простое предложение» - это просто тавтология. Это ничего не объясняет. Тебе было бы лучше без этого.
маркиз Лорн

1
Это хорошее объяснение, где поле должно бытьtransient
Chaklader Asfak Arefe

1
поле интереса и количество слов являются хорошими примерами переходных полей.
Тарун

1
Еще один хороший пример использования: если у вашего объекта есть такие компоненты, как сокеты, и если вы хотите сериализовать, то что происходит с сокетом? Если бы это продолжалось, после десериализации, что бы держал сокет? Имеет смысл сделать этот объект сокета какtransient
Мохаммед Сиддик

28

transientПеременная является переменной , которая не может быть сериализации.

Один из примеров того, когда это может оказаться полезным, - это переменные, которые имеют смысл только в контексте конкретного экземпляра объекта и становятся недействительными после сериализации и десериализации объекта. В этом случае полезно, чтобы эти переменные стали nullвместо них, чтобы при необходимости вы могли повторно инициализировать их с полезными данными.


да, что-то вроде членов поля «password or crediCardPin».
Mateen

17

transientиспользуется, чтобы указать, что поле класса не должно быть сериализовано. Вероятно, лучший пример - это Threadполе. Обычно нет причин для сериализации a Thread, так как его состояние очень специфично для потока.


Поправьте меня, если я ошибаюсь, но Thread не сериализуем, так что он все равно будет пропущен?
TFennis

3
@TFennis: Если сериализуемый класс Aссылается на не сериализуемый класс B(как Threadв вашем примере), то Aдолжен либо пометить ссылку, так как transientXOR должен переопределить процесс сериализации по умолчанию, чтобы сделать что-то разумное с BXOR, предполагая, что только сериализуемые подклассы Bфактически ссылаются (поэтому фактический подкласс должен заботиться об их «плохом» родителе B) XOR признает, что сериализация не удастся. Только в одном случае (помеченном как переходный процесс) Bавтоматически и тихо пропускается.
AH

3
@TFennis Нет, это вызовет исключение.
маркиз Лорн

1
@AH: почему XOR? Я думаю, что код, который выполнял любую комбинацию этих вещей, работал бы, и некоторые комбинации могли бы быть полезными (например, переопределение процесса сериализации по умолчанию может быть полезным, даже если упоминаются только сериализуемые подклассы B, и наоборот).
суперкат

15

Системы сериализации, кроме нативной Java, также могут использовать этот модификатор. Например, Hibernate не сохранит поля, помеченные либо @Transient, либо модификатором transient . Терракота также уважает этот модификатор.

Я полагаю, что переносное значение модификатора таково: «Это поле предназначено только для использования в памяти. Не сохраняйте и не перемещайте его вне этой конкретной ВМ. Оно непереносимо». то есть вы не можете полагаться на его значение в другой области памяти виртуальной машины. Как и volatile, вы не можете полагаться на определенную семантику памяти и потоков.


14
Я думаю, что transientэто не было бы ключевое слово, если бы оно было разработано в это время. Возможно, они будут использовать аннотацию.
Йоахим Зауэр


7

Прежде чем ответить на этот вопрос, я должен объяснить вам СЕРИАЛИЗАЦИЮ , потому что, если вы понимаете, что означает сериализация в научном компьютере, вы легко поймете это ключевое слово.

Сериализация Когда объект передается через сеть / сохраняется на физическом носителе (файл, ...), объект должен быть «сериализован». Сериализация преобразует байты серии объектов статуса. Эти байты отправляются по сети / сохраняются, и объект воссоздается из этих байтов.
пример

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

Теперь, ЕСЛИ ВЫ ХОТИТЕ НЕ ПЕРЕДАТЬ / СОХРАНИТЬ поле этого объекта ТАК , вы можете использовать ключевое слово transient

private transient attr2;

пример


6

Это необходимо, когда вы не хотите делиться некоторыми конфиденциальными данными, которые идут с сериализацией.


7
Существуют варианты использования, отличные от конфиденциальных данных, в которых вы можете не захотеть сериализовать поле. Например, вы, вероятно, никогда не захотите сериализовать Thread(например, кредит @AH), и в этом случае вы отметите его как временный. Однако поток сам по себе не является конфиденциальными данными, просто нет логического смысла его сериализации (и он не сериализуем).
glen3b

1
@ glen3b Этот случай не исключен этим ответом. Это, безусловно, необходимо, так как дела обстоят в случае, когда упоминается постер.
маркиз Лорн

3

Согласно переходному значению Google == длительность только в течение короткого времени; непостоянно.

Теперь, если вы хотите сделать что-то временное в Java, используйте ключевое слово transient.

Q: где использовать переходный процесс?

A: Как правило, в Java мы можем сохранять данные в файлы, собирая их в переменные и записывая эти переменные в файлы, этот процесс известен как сериализация. Теперь, если мы хотим избежать записи переменных данных в файл, мы бы сделали эту переменную переходной.

transient int result=10;

Примечание: временные переменные не могут быть локальными.


0

Проще говоря, временное ключевое слово java защищает поля от бывшего Serialize в качестве их непереходных полей.

В этом фрагменте кода наш абстрактный класс 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;

   }
}

0

Упрощенный пример кода для переходного ключевого слова.

import java.io.*;

class NameStore implements Serializable {
    private String firstName, lastName;
    private transient String fullName;

    public NameStore (String fName, String lName){
        this.firstName = fName;
        this.lastName = lName;
        buildFullName();
    }

    private void buildFullName() {
        // assume building fullName is compuational/memory intensive!
        this.fullName = this.firstName + " " + this.lastName;
    }

    public String toString(){
        return "First Name : " + this.firstName
            + "\nLast Name : " + this.lastName
            + "\nFull Name : " + this.fullName;
    }

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

public class TransientExample{
    public static void main(String args[]) throws Exception {
        ObjectOutputStream o = new ObjectOutputStream(new FileOutputStream("ns"));
        o.writeObject(new NameStore("Steve", "Jobs"));
        o.close();

        ObjectInputStream in = new ObjectInputStream(new FileInputStream("ns"));
        NameStore ns = (NameStore)in.readObject();
        System.out.println(ns);
    }
}

0

Поле, которое объявляется с помощью временного модификатора, не будет участвовать в сериализованном процессе. Когда объект сериализуется (сохраняется в любом состоянии), значения его переходных полей игнорируются в последовательном представлении, в то время как поле, отличное от переходных полей, будет участвовать в процессе сериализации. Это основная цель переходного ключевого слова.

Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.