Ответы:
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
ключевое слово для предотвращения сериализации определенных полей.
Прежде чем понять 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
, поэтому оно не будет храниться в постоянном хранилище.
NameFormatter
подержанную машину toString()
. Или что-нибудь еще, связанное с настройкой, просмотром или бизнес-логикой, а не с данными.
Чтобы позволить вам определить переменные, которые вы не хотите сериализовать.
В объекте у вас может быть информация, которую вы не хотите сериализовать / сохранять (возможно, ссылка на родительский объект фабрики), или, возможно, нет смысла сериализовать. Отметив их как «переходные» означает, что механизм сериализации будет игнорировать эти поля.
Мой маленький вклад:
Что такое переходное поле?
По сути, любое поле, измененное с помощью transient
ключевого слова, является временным полем.
Зачем нужны переходные поля в Java? Ключевое слово дает некоторый контроль над процессом сериализации и позволяет исключить некоторые свойства объектов из этого процесса. Процесс сериализации используется для сохранения Java-объектов, главным образом для того, чтобы их состояния могли сохраняться во время их передачи или неактивности. Иногда имеет смысл не сериализовать определенные атрибуты объекта. Какие поля следует пометить как переходные?
Теперь мы знаем цельtransient
transient
ключевые слова и переходные поля, важно знать, какие поля помечать переходные. Статические поля также не сериализуются, поэтому соответствующее ключевое слово также поможет. Но это может испортить дизайн вашего класса; Вот где transient
ключевое слово приходит на помощь. Я стараюсь не допустить сериализации полей, значения которых могут быть получены из других, поэтому я отмечаю их как временные. Если у вас есть поле с именем interest
, значение которого можно вычислить из других полей ( principal
, rate
& time
), нет необходимости его сериализации.
Еще один хороший пример - количество слов в статье. Если вы сохраняете всю статью, вам не нужно сохранять количество слов, потому что оно может быть вычислено, когда статья будет «десериализована». Или подумайте о регистраторах;Logger
экземпляры почти никогда не нужно сериализовать, поэтому их можно сделать временными.
transient
transient
transient
Переменная является переменной , которая не может быть сериализации.
Один из примеров того, когда это может оказаться полезным, - это переменные, которые имеют смысл только в контексте конкретного экземпляра объекта и становятся недействительными после сериализации и десериализации объекта. В этом случае полезно, чтобы эти переменные стали null
вместо них, чтобы при необходимости вы могли повторно инициализировать их с полезными данными.
transient
используется, чтобы указать, что поле класса не должно быть сериализовано. Вероятно, лучший пример - это Thread
поле. Обычно нет причин для сериализации a Thread
, так как его состояние очень специфично для потока.
A
ссылается на не сериализуемый класс B
(как Thread
в вашем примере), то A
должен либо пометить ссылку, так как transient
XOR должен переопределить процесс сериализации по умолчанию, чтобы сделать что-то разумное с B
XOR, предполагая, что только сериализуемые подклассы B
фактически ссылаются (поэтому фактический подкласс должен заботиться об их «плохом» родителе B
) XOR признает, что сериализация не удастся. Только в одном случае (помеченном как переходный процесс) B
автоматически и тихо пропускается.
Системы сериализации, кроме нативной Java, также могут использовать этот модификатор. Например, Hibernate не сохранит поля, помеченные либо @Transient, либо модификатором transient . Терракота также уважает этот модификатор.
Я полагаю, что переносное значение модификатора таково: «Это поле предназначено только для использования в памяти. Не сохраняйте и не перемещайте его вне этой конкретной ВМ. Оно непереносимо». то есть вы не можете полагаться на его значение в другой области памяти виртуальной машины. Как и volatile, вы не можете полагаться на определенную семантику памяти и потоков.
transient
это не было бы ключевое слово, если бы оно было разработано в это время. Возможно, они будут использовать аннотацию.
Потому что не все переменные имеют сериализуемую природу
Прежде чем ответить на этот вопрос, я должен объяснить вам СЕРИАЛИЗАЦИЮ , потому что, если вы понимаете, что означает сериализация в научном компьютере, вы легко поймете это ключевое слово.
Сериализация
Когда объект передается через сеть / сохраняется на физическом носителе (файл, ...), объект должен быть «сериализован». Сериализация преобразует байты серии объектов статуса. Эти байты отправляются по сети / сохраняются, и объект воссоздается из этих байтов.
пример
public class Foo implements Serializable
{
private String attr1;
private String attr2;
...
}
Теперь, ЕСЛИ ВЫ ХОТИТЕ НЕ ПЕРЕДАТЬ / СОХРАНИТЬ поле этого объекта ТАК , вы можете использовать ключевое слово transient
private transient attr2;
Это необходимо, когда вы не хотите делиться некоторыми конфиденциальными данными, которые идут с сериализацией.
Thread
(например, кредит @AH), и в этом случае вы отметите его как временный. Однако поток сам по себе не является конфиденциальными данными, просто нет логического смысла его сериализации (и он не сериализуем).
Согласно переходному значению Google == длительность только в течение короткого времени; непостоянно.
Теперь, если вы хотите сделать что-то временное в Java, используйте ключевое слово transient.
Q: где использовать переходный процесс?
A: Как правило, в Java мы можем сохранять данные в файлы, собирая их в переменные и записывая эти переменные в файлы, этот процесс известен как сериализация. Теперь, если мы хотим избежать записи переменных данных в файл, мы бы сделали эту переменную переходной.
transient int result=10;
Примечание: временные переменные не могут быть локальными.
Проще говоря, временное ключевое слово 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;
}
}
Упрощенный пример кода для переходного ключевого слова.
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);
}
}
Поле, которое объявляется с помощью временного модификатора, не будет участвовать в сериализованном процессе. Когда объект сериализуется (сохраняется в любом состоянии), значения его переходных полей игнорируются в последовательном представлении, в то время как поле, отличное от переходных полей, будет участвовать в процессе сериализации. Это основная цель переходного ключевого слова.
@DoNotSerialize
?