検索
ホームページJava&#&チュートリアルJavaでトランジェントを使用する方法

Javaでトランジェントを使用する方法

Feb 20, 2017 am 10:13 AM
javatransient

Java 言語の Transient は、クラス、同期、その他のよく知られたキーワードほどよく知られていないため、面接の質問でいくつか登場します。この記事では、一時的なものについて説明します。

transient の使用

Q: transient キーワードは何を達成できますか?

A: オブジェクトがシリアル化される (ターゲット ファイルにバイト シーケンスを書き込む) と、オブジェクトが逆シリアル化される (ソース ファイルから読み取る) ときに、インスタンス内でこのキーワードで宣言された変数が永続化されなくなります。 )、そのようなインスタンス変数の値は永続化されず、復元されません。たとえば、オブジェクトを逆シリアル化する場合、データ ストリーム (ファイルなど) が存在しない可能性があります。その理由は、オブジェクト内に java.io.InputStream 型の変数があり、これらの変数によって参照される入力ストリームをデシリアライズできないためです。連載中に開封しました。

トランジェントの使い方の紹介

Q: トランジェントの使い方は?

A: インスタンス変数宣言に transient 修飾子が含まれています。スニペット 1 は簡単なデモンストレーションを提供します。

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

class ClassLib implements Serializable {
    private transient InputStream is;
    private int majorVer;
    private int minorVer;
    ClassLib(InputStream is) throws IOException {
        System.out.println("ClassLib(InputStream) called");
        this.is = is;
        DataInputStream dis;
        if (is instanceof DataInputStream)
            dis = (DataInputStream) is;
        else
            dis = new DataInputStream(is);
        if (dis.readInt() != 0xcafebabe)
            throw new IOException("not a .class file");
        minorVer = dis.readShort();
        majorVer = dis.readShort();
    }
    int getMajorVer() {
        return majorVer;
    }
    int getMinorVer() {
        return minorVer;
    }
    void showIS() {
        System.out.println(is);
    }
}
public class TransDemo {
    public static void main(String[] args) throws IOException {
        if (args.length != 1) {
            System.err.println("usage: java TransDemo classfile");
            return;
        }
        ClassLib cl = new ClassLib(new FileInputStream(args[0]));
        System.out.printf("Minor version number: %d%n", cl.getMinorVer());
        System.out.printf("Major version number: %d%n", cl.getMajorVer());
        cl.showIS();
        try (FileOutputStream fos = new FileOutputStream("x.ser");
                ObjectOutputStream oos = new ObjectOutputStream(fos)) {
            oos.writeObject(cl);
        }
        cl = null;
        try (FileInputStream fis = new FileInputStream("x.ser");
                ObjectInputStream ois = new ObjectInputStream(fis)) {
            System.out.println();
            cl = (ClassLib) ois.readObject();
            System.out.printf("Minor version number: %d%n", cl.getMinorVer());
            System.out.printf("Major version number: %d%n", cl.getMajorVer());
            cl.showIS();
        } catch (ClassNotFoundException cnfe) {
            System.err.println(cnfe.getMessage());
        }
    }
}

フラグメント 1: ClassLib オブジェクトのシリアル化と逆シリアル化

ClassLib クラスと TransDemo クラスはフラグメント 1 で宣言されています。 ClassLib は、Java クラス ファイルを読み取り、これらのインスタンスをシリアル化および逆シリアル化できるように java.io.Serializable インターフェイスを実装するライブラリです。 TransDemo は、ClassLib インスタンスのシリアル化と逆シリアル化に使用されるアプリケーション クラスです。

ClassLib は、入力ストリームを無意味にシリアル化する可​​能性があるため、そのインスタンス変数を一時変数として宣言します (前述のように)。実際、この変数が一時的でない場合、InputStream は Serializable インターフェイスを実装していないため、x.ser の内容を逆シリアル化するときに java.io.NotSerializableException がスローされます。

フラグメント 1 をコンパイルします: javac TransDemo.java; 1 つのパラメーター TransDemo.class を使用してアプリケーションを実行します: java TransDemo TransDemo.class。次のような出力が表示される場合があります。

ClassLib(InputStream) called
Minor version number: 0
Major version number: 51
java.io.FileInputStream@79f1e0e0
Minor version number: 0
Major version number: 51
null

上記の出力は、オブジェクトが再構築されるときにコンストラクター メソッドが呼び出されないことを示しています。さらに、ClassLib オブジェクトがシリアル化されるときに値を持つ MajorVer およびminorVer とは対照的に、 is はデフォルトで null であるとみなされます。

クラスのメンバー変数と一時変数

Q: クラスのメンバー変数で一時変数を使用できますか?

A: 質問の答えについてはスニペット 2 を参照してください

public class TransDemo {
    public static void main(String[] args) throws IOException {
        Foo foo = new Foo();
        System.out.printf("w: %d%n", Foo.w);
        System.out.printf("x: %d%n", Foo.x);
        System.out.printf("y: %d%n", foo.y);
        System.out.printf("z: %d%n", foo.z);
        try (FileOutputStream fos = new FileOutputStream("x.ser");
                ObjectOutputStream oos = new ObjectOutputStream(fos)) {
            oos.writeObject(foo);
        }
        foo = null;
        try (FileInputStream fis = new FileInputStream("x.ser");
                ObjectInputStream ois = new ObjectInputStream(fis)) {
            System.out.println();
            foo = (Foo) ois.readObject();
            System.out.printf("w: %d%n", Foo.w);
            System.out.printf("x: %d%n", Foo.x);
            System.out.printf("y: %d%n", foo.y);
            System.out.printf("z: %d%n", foo.z);
        } catch (ClassNotFoundException cnfe) {
            System.err.println(cnfe.getMessage());
        }
    }
}

フラグメント 2: Foo オブジェクトのシリアル化と逆シリアル化

フラグメント 2 はスニペット 1 と多少似ています。ただし、違いは、シリアル化および逆シリアル化されるのは ClassLib ではなく Foo オブジェクトであることです。さらに、Foo には変数のペア、w と x、およびインスタンス変数 y と z が含まれています。

フラグメント 2 (javac TransDemo.java) をコンパイルし、アプリケーション (java TransDemo) を実行します。次の出力が表示されます。

w: 1
x: 2
y: 3
z: 4
w: 1
x: 2
y: 3
z: 0

この出力は、インスタンス変数 y がシリアル化されているが、z はシリアル化されておらず、一時的としてマークされていることがわかります。ただし、Foo がシリアル化されると、変数 w と x がシリアル化および逆シリアル化されるのか、それとも通常のクラス初期化方法で初期化されるだけなのかはわかりません。答えを得るには、x.ser の内容を確認する必要があります。

x.ser の 16 進数を以下に示します:

00000000 AC ED 00 05 73 72 00 03 46 6F 6F FC 7A 5D 82 1D ....sr..Foo.z]..
00000010 D2 9D 3F 02 00 01 49 00 01 79 78 70 00 00 00 03 ..?...I..yxp....

JavaWorld の記事「明らかになった Java シリアル化アルゴリズム」のおかげで、次の出力の意味が分かりました:

AC ED シリアル化プロトコル識別

00 05ストリームのバージョン番号

73は、これが新しいオブジェクトであることを意味します

72は、これが新しいクラスであることを意味します

00 03は、クラス名の長さ(3)を意味します

46 6F 6Fは、クラス名(Foo)を意味します

FC 7A 5D 82 1D D2 9D 3F はクラスのシリアル バージョン識別子を表します

02 はオブジェクトがシリアル化をサポートしていることを表します

00 01 はこのクラスの変数の数 (1) を表します

49 変数の型コード (0x49、または I、を表します) int )

00 01 は変数名の長さ (1) を表します

79 変数名 (y)

78 はオブジェクトのオプションのデータブロックの終わりを表します

70 はクラス階層の最上位に到達したことを表します

00 00 00 03 y の値を表す (3)

明らかに、インスタンス変数 y のみがシリアル化されます。 z は一時的なものであるため、シリアル化できません。さらに、一時的とマークされていても、w と x はシリアル化できないクラス変数であるため、シリアル化できません。

上記は Java での transient の使用方法です。その他の関連コンテンツについては、PHP 中国語 Web サイト (www.php.cn) に注目してください。

声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
JVMのクラスローダーサブシステムは、プラットフォームの独立性にどのように貢献していますか?JVMのクラスローダーサブシステムは、プラットフォームの独立性にどのように貢献していますか?Apr 23, 2025 am 12:14 AM

クラスローダーは、統一されたクラスファイル形式、動的読み込み、親代表団モデル、プラットフォーム非依存バイトコードを通じて、さまざまなプラットフォーム上のJavaプログラムの一貫性と互換性を保証し、プラットフォームの独立性を実現します。

Javaコンパイラはプラットフォーム固有のコードを作成しますか?説明する。Javaコンパイラはプラットフォーム固有のコードを作成しますか?説明する。Apr 23, 2025 am 12:09 AM

Javaコンパイラによって生成されたコードはプラットフォームに依存しませんが、最終的に実行されるコードはプラットフォーム固有です。 1。Javaソースコードは、プラットフォームに依存しないバイトコードにコンパイルされます。 2。JVMは、特定のプラットフォームのバイトコードをマシンコードに変換し、クロスプラットフォーム操作を保証しますが、パフォーマンスは異なる場合があります。

JVMは、さまざまなオペレーティングシステムでマルチスレッドをどのように処理しますか?JVMは、さまざまなオペレーティングシステムでマルチスレッドをどのように処理しますか?Apr 23, 2025 am 12:07 AM

マルチスレッドは、プログラムの応答性とリソースの利用を改善し、複雑な同時タスクを処理できるため、最新のプログラミングで重要です。 JVMは、スレッドマッピング、スケジューリングメカニズム、同期ロックメカニズムを介して、異なるオペレーティングシステム上のマルチスレッドの一貫性と効率を保証します。

Javaの文脈では、「プラットフォームの独立」とはどういう意味ですか?Javaの文脈では、「プラットフォームの独立」とはどういう意味ですか?Apr 23, 2025 am 12:05 AM

Javaのプラットフォームの独立性とは、書かれたコードがJVMが変更なしでインストールされた任意のプラットフォームで実行できることを意味します。 1)JavaソースコードはBytecodeにコンパイルされ、2)BytecodeはJVMによって解釈および実行されます、3)JVMは、プログラムが異なるオペレーティングシステムで実行されることを確認するために、メモリ管理とガベージコレクション機能を提供します。

Javaアプリケーションは、プラットフォーム固有のバグや問題に遭遇する可能性がありますか?Javaアプリケーションは、プラットフォーム固有のバグや問題に遭遇する可能性がありますか?Apr 23, 2025 am 12:03 AM

JavaApplicationScanIndEDENCOUNTIONPLATFORM-SPECISTESUESUSESEJVM'SABSTRACTION.REASONSINCLUDE:1)NativeCodeandLibraries、2)OperatingSystemDifferences、3)JVMimplementationVariations、および4)HardweardePencies.TomiteTETETETESES、DEVELAPERSHOULD:1)

クラウドコンピューティングは、Javaのプラットフォーム独立の重要性にどのような影響を与えますか?クラウドコンピューティングは、Javaのプラットフォーム独立の重要性にどのような影響を与えますか?Apr 22, 2025 pm 07:05 PM

クラウドコンピューティングにより、Javaのプラットフォームの独立性が大幅に向上します。 1)JavaコードはBytecodeにコンパイルされ、異なるオペレーティングシステムでJVMによって実行され、クロスプラットフォーム操作が確保されます。 2)DockerとKubernetesを使用してJavaアプリケーションを展開して、携帯性とスケーラビリティを向上させます。

Javaのプラットフォームの独立性は、その広範な採用においてどのような役割を果たしましたか?Javaのプラットフォームの独立性は、その広範な採用においてどのような役割を果たしましたか?Apr 22, 2025 pm 06:53 PM

java'splatformendenceallowsdevelopersowritecodeodeonceanceandonitondeviceoros withajvm.

コンテナ化テクノロジー(Dockerなど)は、Javaのプラットフォーム独立性の重要性にどのように影響しますか?コンテナ化テクノロジー(Dockerなど)は、Javaのプラットフォーム独立性の重要性にどのように影響しますか?Apr 22, 2025 pm 06:49 PM

Dockerなどのコンテナ化技術は、Javaのプラットフォームの独立性を置き換えるのではなく、強化します。 1)環境全体の一貫性を確保し、2)特定のJVMバージョンを含む依存関係を管理する、3)展開プロセスを簡素化して、Javaアプリケーションをより順応性と管理しやすくする。

See all articles

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

PhpStorm Mac バージョン

PhpStorm Mac バージョン

最新(2018.2.1)のプロフェッショナル向けPHP統合開発ツール

MantisBT

MantisBT

Mantis は、製品の欠陥追跡を支援するために設計された、導入が簡単な Web ベースの欠陥追跡ツールです。 PHP、MySQL、Web サーバーが必要です。デモおよびホスティング サービスをチェックしてください。

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

MinGW - Minimalist GNU for Windows

MinGW - Minimalist GNU for Windows

このプロジェクトは osdn.net/projects/mingw に移行中です。引き続きそこでフォローしていただけます。 MinGW: GNU Compiler Collection (GCC) のネイティブ Windows ポートであり、ネイティブ Windows アプリケーションを構築するための自由に配布可能なインポート ライブラリとヘッダー ファイルであり、C99 機能をサポートする MSVC ランタイムの拡張機能が含まれています。すべての MinGW ソフトウェアは 64 ビット Windows プラットフォームで実行できます。

WebStorm Mac版

WebStorm Mac版

便利なJavaScript開発ツール