说一说java关键字final和transient

所属分类: 软件编程 / java 阅读数: 47
收藏 0 赞 0 分享

首先,说说final。
final关键字可以修饰变量,方法,类
final变量:
需求:
1 需要一个永不改变的编译时常量
2 一个运行时被初始化的值,不希望被更改
好处:编译时就执行的计算,减轻运行时的负担
扩展:
可以修饰基本类型和引用对象。修饰基本类型的时候,表示数值很定不变。修饰对象引用的时候,一旦引用被初始化指向一个对象,就无法再将它更改指向另一个对象(该对象本身是可以修改的)
空白final
final修饰但又没有给出初始值的域
必须在域的的定义或构造器内用表达式给final赋值(final使用前必须初始化)
 注意:
如果一个对象被static和final同时修饰(编译期常量),一般用大写表示,下划线链接单词
修饰参数:
如果final修饰参数,表示该参数可读,但无法修改。
用法示例:

private Random rand=new Random();
 private static Random random=new Random();
 private final int n1=12;
 private final int number=rand.nextInt(30);
 private static final int NUMBER2=random.nextInt(40);
 @Test
 public void finalDataTest(){
  System.out.println(n1);
  System.out.println("--------------------");
  System.out.println(rand.nextInt(30));
  System.out.println("--------------------");
  System.out.println("编译初始之后,不会改变:"+number);
  System.out.println("--------------------");
  System.out.println("编译初始之后,不会改变:"+NUMBER2);  
 }
 /**
  * final修饰参数:该参数可读,但无法修改。
  * @param sk
  * @return

  */
 public String finalParam(final String sk){

  //sk="jeyson"; final参数不能被修改

  return sk;
 } 

final方法:
final也可以修饰方法,表示该方法不能被子类继承。
使用final的好处:
1 JDK1.5以前,效率更高,JDK1.5以后可以忽略
 2 方法锁定,确保子类中该方法含义不变,不能被覆盖
 用法示例:    

public final String finalMethod(){

   return "Hello World" ;

  }

final类:
不希望被任何类继承,可以使用final修饰类
用法示例:  

 public final class FinalClassTx {
 private int k ; 
 public void getMyWord(){
   System. out .println("这是一个final类,k的值是" +getK());
 } 
public int getK() {
  return k ;

}
public void setK( int k) {

  this .k = k;

} 

} 

然后transient关键字:
transient只能修饰变量,表示该变量不能被序列化。
一般我们继承Serializable接口的类,序列化会自动进行,使用transient修饰的变量在该类被序列化的时候,不会序列化到指定目的地。
 所以,
1 被transient修饰的变量不再是对象持久化的一部分,该变量内容序列化无法获得访问
2 transient只能修饰变量,不能修饰方法和类
 3 一个静态变量无论是否被transient修饰,都不能被序列化
用法示例:  

public class TransientEx { 
 public static void main(String[] args) {
  User user=new User();
  user.setUsername("jeyson");
  user.setPassword("123456");
  System.out.println("序列化前:");
  System.out.println(" username="+user.getUsername());
   System.out.println(" password="+user.getPassword());
  //序列化
  try {
   ObjectOutputStream os=new ObjectOutputStream(new FileOutputStream("C://MyResource//test1.txt"));
   os.writeObject(user);
   os.flush();
   os.close();
  } catch (Exception e) {
   e.printStackTrace();
  }
  //反序列化

  try {
   
   ObjectInputStream is=new ObjectInputStream(new FileInputStream("C://MyResource//test1.txt"));

   user=(User) is.readObject();

   is.close();

   System.out.println("序列化后:");

   System.out.println(" username="+user.getUsername());

   System.out.println(" password="+user.getPassword());
   
  } catch (Exception e) {
   e.printStackTrace();
  }
  System.out.println("--------------------------------");
  

 }

}
class User implements Serializable{ 
 private static final long serialVersionUID = 1L; 

 private String username;

 //使用 transient

 private transient String password;

 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;

 } 

 

}

扩展:Externalizable
实现了serializable接口的类,所以序列化会自动进行
实现了Externaliazble接口的类,没有任何东西可以自动序列化,无论是否使用transient对结果都没有影响。
此时如果需要序列化,需要在writeExternal方法中上进行手动指定所要序列化的变量。
使用示例:     

public class ExternalizableEx implements Externalizable {
 private transient String name="ssss"; 
 @Override
 public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
  name=(String) in.readObject();
  
 }
 @Override
 public void writeExternal(ObjectOutput out) throws IOException {
  out.writeObject(name);  

 }
 public String getName() {
  return name;

 }

 public void setName(String name) {
  this.name = name;

 }
 
 public static void main(String[] args) {
   ExternalizableEx ex=new ExternalizableEx();
   ex.setName("jeyson");
   System.out.println("Externalizable序列化前:");
   System.out.println(ex.getName());
   //序列化
   try {

    ObjectOutputStream os=new ObjectOutputStream(new FileOutputStream(new File("C://MyResource//test2.txt")));
    os.writeObject(ex);
    os.flush();
    os.close();
  } catch (Exception e) {

   e.printStackTrace();
  }
   //反序列化

   try {
    ObjectInputStream is=new ObjectInputStream(new FileInputStream(new File("C://MyResource//test2.txt")));
    ex=(ExternalizableEx) is.readObject();
    is.close();

    System.out.println("Externalizable序列化后:");

    System.out.println(ex.getName());

  } catch (Exception e) {

   e.printStackTrace();
  }

 }

}

声明:
final大部分来自《java编程思想》第四版

参考文章:https://www.jb51.net/article/86996.htm

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

更多精彩内容其他人还在看

Java的面向对象编程基本概念学习笔记整理

这篇文章主要介绍了Java的面向对象编程基本概念学习笔记整理,包括类与方法以及多态等支持面向对象语言中的重要特点,需要的朋友可以参考下
收藏 0 赞 0 分享

Eclipse下编写java程序突然不会自动生成R.java文件和包的解决办法

这篇文章主要介绍了Eclipse下编写java程序突然不会自动生成R.java文件和包的解决办法 的相关资料,需要的朋友可以参考下
收藏 0 赞 0 分享

基于Java实现杨辉三角 LeetCode Pascal's Triangle

这篇文章主要介绍了基于Java实现杨辉三角 LeetCode Pascal's Triangle的相关资料,需要的朋友可以参考下
收藏 0 赞 0 分享

Java中Spring获取bean方法小结

Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架,如何在程序中获取Spring配置的bean呢?下面通过本文给大家介绍Java中Spring获取bean方法小结,对spring获取bean方法相关知识感兴趣的朋友一起学习吧
收藏 0 赞 0 分享

如何计算Java对象占用了多少空间?

在Java中没有sizeof运算符,所以没办法知道一个对象到底占用了多大的空间,但是在分配对象的时候会有一些基本的规则,我们根据这些规则大致能判断出来对象大小,需要的朋友可以参考下
收藏 0 赞 0 分享

剖析Java中的事件处理与异常处理机制

这篇文章主要介绍了Java中的事件处理与异常处理机制,讲解Java是如何对事件或者异常作出响应以及定义异常的一些方法,需要的朋友可以参考下
收藏 0 赞 0 分享

详解Java的Struts2框架的结构及其数据转移方式

这篇文章主要介绍了详解Java的Struts2框架的结构及其数据转移方式,Struts框架是Java的SSH三大web开发框架之一,需要的朋友可以参考下
收藏 0 赞 0 分享

Java封装好的mail包发送电子邮件的类

本文给大家分享了2个java封装好的mail包发送电子邮件的类,并附上使用方法,小伙伴们可以根据自己的需求自由选择。
收藏 0 赞 0 分享

在Java的Struts中判断是否调用AJAX及用拦截器对其优化

这篇文章主要介绍了在Java的Struts中判断是否调用AJAX及用拦截器对其优化的方法,Struts框架是Java的SSH三大web开发框架之一,需要的朋友可以参考下
收藏 0 赞 0 分享

java多线程Future和Callable类示例分享

JAVA多线程实现方式主要有三种:继承Thread类、实现Runnable接口、使用ExecutorService、Callable、Future实现有返回结果的多线程。其中前两种方式线程执行完后都没有返回值,只有最后一种是带返回值的。今天我们就来研究下Future和Callab
收藏 0 赞 0 分享
查看更多