首页 > 编程语言 >[Java] Java 关键字 : transient

[Java] Java 关键字 : transient

时间:2024-07-05 13:57:58浏览次数:22  
标签:username 序列化 Java 变量 关键字 transient user password

0 序

Java中的transient关键字,transient是短暂的意思。对于transient修饰的成员变量,在类的实例对象的序列化处理过程中会被忽略。
因此,transient变量不会贯穿对象的序列化和反序列化,生命周期仅存于调用者的内存中而不会写到磁盘里进行持久化。

1 序列化

  • Java中对象的序列化指的是将对象转换成以字节序列的形式来表示,这些字节序列包含了对象的数据和信息,一个序列化后的对象可以被写到数据库或文件中,也可用于网络传输。
  • 一般地,当我们使用缓存cache内存空间不够有可能会本地存储到硬盘)或远程调用rpc网络传输)的时候,经常需要让实体类实现Serializable接口,目的就是为了让其可序列化
  • 当然,序列化后的最终目的是为了反序列化,恢复成原先的Java对象实例。所以序列化后的字节序列都是可以恢复成Java对象的,这个过程就是反序列化。

2 为什么要用transient关键字?

  • 持久化对象时,对于一些特殊的数据成员(如用户的密码,银行卡号等),我们不想用序列化机制来保存它。为了在一个特定对象的一个成员变量关闭序列化,可以在这个成员变量前加上关键字transient

3 transient的作用

作用

  • transient是Java语言的关键字,用来表示一个成员变量不是该对象序列化的一部分。
  • 当一个对象被序列化的时候,transient型变量的值不包括在序列化的结果中。而非transient型的变量是被包括进去的。

即:一旦变量被transient修饰,变量将不再是对象持久化的一部分,该变量内容在序列化后无法被访问。

  • transient关键字只能修饰变量,而不能修饰方法
  • 注意,本地变量是不能被transient关键字修饰的。变量如果是用户自定义类变量,则该类需要实现Serializable接口。
  • 一个静态变量不管是否被transient修饰,均不能被序列化
  • 如果反序列化后类中static变量还有值,则值为当前JVM中对应static变量的值。
  • 序列化保存的是对象状态静态变量保存的是类状态。因此,序列化并不保存静态变量
  • 典型应用:如果类中使用了Logger实例(private final static Logger logger = ...),那么Logger实例也是不需要序列化的。
    即:static修饰的静态变量天然地就是不可序列化的。

案例:序列化与反序列化

package org.example;

import lombok.extern.slf4j.Slf4j;
import org.example.entity.User;
import org.springframework.util.ObjectUtils;

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

public class TransientTest {
    public static void main(String[] args) {
        User user = new User();
        user.setUsername("Jack");
        user.setPassword("123456");

        System.out.println("read before serializable :");
        System.out.println("username : " + user.getUsername());//username : Jack
        System.out.println("password : " + user.getPassword());//password : 123456
        System.out.println("toString : " + user.toString());//toString : User{username='Jack', password='123456'}

        //序列化数据
        try {
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(
                new FileOutputStream("E:\\tmp_data\\user.serial-data.txt")
            );
            objectOutputStream.writeObject( user );
            objectOutputStream.flush();
            objectOutputStream.close();
        } catch (Exception exception) {
            System.err.println("Fail to write serial data to file! exception : " + exception);
        }

        //反序列化数据
        try {
            ObjectInputStream objectInputStream = new ObjectInputStream(
                    new FileInputStream("E:\\tmp_data\\user.serial-data.txt")
            );
            user = (User) objectInputStream.readObject();
            objectInputStream.close();
            System.out.println("read after serializable :");
            System.out.println("username : " + user.getUsername());//username : Jack
            System.out.println("password : " + user.getPassword());//password : null
            System.out.println("toString : " + user.toString());//toString : User{username='Jack', password='null'}
        } catch (Exception exception) {
            System.err.println("Fail to read serial data from file! exception : " + exception);
        }
    }
}

package org.example.entity;

import java.io.Serializable;

public class User implements Serializable {
    private static final Long serialVersionUID = 3535352442213124L;

    private String username;
    private transient String password;

    public User() {
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    @Override
    public String toString() {
        return "User{" +
                "username='" + username + '\'' +
                ", password='" + password + '\'' +
                '}';
    }
}

X 参考文献

标签:username,序列化,Java,变量,关键字,transient,user,password
From: https://www.cnblogs.com/johnnyzen/p/18285687

相关文章

  • JAVA API营业执照识别、企业工商信息查询
    现如今,随着互联网技术的不断成熟与智能应用的不断普及,人们对于营业执照识别技术的需求越来越大。像工商、税务、银行、网上注册等一些应用场景在进行经营活动和业务中,都需要对营业执照的信息进行采集和录入,无论是个人业务还是企业对公业务都在逐步的互联网化。对于平台......
  • 2.2 实验三、自动生成语法分析程序(JavaCUP)
    help-assignment2.3实验三、自动生成语法分析程序(JavaCUP)实验三要求你下载一个语法分析程序自动生成工具JavaCUP,利用该工具自动产生一个Oberon-0语言的语法分析和语法制导翻译程序;生成的程序源代码是以Java语言编写的。2.3.1实验步骤3.1、下载自动生成工具Java......
  • 微信小程序源码-基于Java后端的餐厅点餐系统毕业设计(附源码+论文)
    大家好!我是程序员一帆,感谢您阅读本文,欢迎一键三连哦。......
  • 微信小程序源码-基于Java后端的球馆预约系统毕业设计(附源码+论文)
    大家好!我是程序员一帆,感谢您阅读本文,欢迎一键三连哦。......
  • 微信小程序源码-基于Java后端的英语互助系统毕业设计(附源码+论文)
    大家好!我是程序猿老A,感谢您阅读本文,欢迎一键三连哦。......
  • Java 方法中循环调用具有事务的方法
    在Java中,循环调用一个具有事务的方法时,需要特别注意事务的边界和管理。通常,事务的边界是由框架(如Spring)来控制的,确保方法执行时数据的完整性和一致性。然而,在循环中调用事务方法时,每个调用都可以被视为独立的事务,除非特别配置以允许跨多个方法调用共享同一事务。1.Java方法中循......
  • Java定时任务CRON表达式
    CRON表达式@PostMapping("/schedule")publicStringscheduleTaskWithCron(@RequestParamintsecond,@RequestParamintminute,@RequestParaminthour,@RequestParamintdayOfMonth,......
  • 18. JAVA 多线程锁介绍
    1.前言本节内容主要是对Java多线程锁进行介绍,是对锁的一个全方位的概述,为我们对后续深入学习不同的锁的使用方法奠定一个良好的基础。本节内容的知识点如下:乐观锁与悲观锁的概念,以及两种锁之间的区别,这是并发编程中经常涉及到的知识点,这是本节课程的核心知识点,是热度很高......
  • Java泛型
    Java泛型泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。比如我们要写一个排序方法,能够对整型数组、字符串数组甚至其他任何类型的数据进行排序,我们就可以使用Java泛型。1......
  • Java 空间和时间高效的二项式系数(Space and time efficient Binomial Coefficient)
    这里函数采用两个参数n和k,并返回二项式系数C(n,k)的值。 例子: 输入:n=4和k=2输出:6解释:4C2等于4!/(2!*2!)=6输入:n=5和k=2输出:10解释:5C2等于5!/(3!*2!)=10        在本文中,我们讨论了O(n*k)时间和O(k)额外空间算法。C(n,......