转载自 https://blog.csdn.net/hikvision_java_gyh/article/details/8964541

1、如果一个数据既是static又是final,那么它会拥有一块无法改变的存储空间。

2、final data: 当final用于基本数据类型时,final让其值(value)保持不变,但是当用于object reference时,final仅让reference保持不变。也就是说当reference一旦被初始化用于代表某个对象时,便再也不能改变指向另一个对象,但对象本身的内容确实可以改变的。final对array的作用和对reference的作用一样。参考以下例子:

public class Test1{
    private final int li_int=12;
    private final InClass inClass1=new InClass(5);
    private final InClass inClass2=new InClass(8);
    public void modifiedFinal(int a){
    //下面语句出现编译错误,不能修改final基本类型的值
    //li_int = a; 
    //下面语句出现编译错误,不能将已经初始化的final变量指向另一个对象
    //inClass1=inClass2;
    //下面语句成功,虽然引用不能改变但final变量引用的对象本身内容是可以改变的
    inClass1.mod(a);
 }
 class InClass{
  int li_a=0;
  public InClass(int a){
   li_a=a; 
  } 
  public int mod(int b){
   li_a=b; 
   return li_a;
  }
 }
 public static void main(String args[]){
  Test1 test1=new Test1();
  test1.modifiedFinal(100);
  System.out.println(test1.inClass1.li_a);
 }
}

3、blank finals:java允许将数据成员声明为final,却不赋初值。但是,blank finals必须在使用之前初始化,且必须在构造函数中初始化。请参考以下例子:

public class Test2{
 //final变量一开始允许不赋值
 private final int li_int;
 public Test2(int a){
  //下面语句编译通过,对定义为空的final变量的赋值必须在构造方法中进行,而且必须要赋值,不赋值也报错
  li_int = a; 
 }
 public int mod(int a){
  //下面语句编译出错,对定义为空的final变量的赋值必须在构造方法中进行
  //li_int = a; 
  return li_int;
 } 
}

4、final arguments: 声明arguments为final,可以保证该argument不能再被指向它处,当argment是基本数据类型时,就意味着值不能改变。参考以下例子:

public class Test3{
 private  int li_int=12;
 private  InClass inClass1=new InClass(5);
 private  InClass inClass2=new InClass(8);
 public void modifiedFinal(final int a,final InClass in){
  //下面语句出现编译错误,不能修改final基本类型的值
  //a = 15; 
  //下面语句出现编译错误,不能将已经初始化的final变量指向另一个对象
  //in=inClass2;
  //下面语句成功,虽然引用不能改变但final变量引用的对象本身内容是可以改变的
  in.mod(a);
 }
 class InClass{
  int li_a=0;
  public InClass(int a){
   li_a=a; 
  } 
  public int mod(int b){
   li_a=b; 
   return li_a;
  }
 }
 public static void main(String args[]){
  int a=100;
  //内部类初始化
  Test3 test3=new Test3();
  Test3.InClass in=test3.new InClass(30);
  System.out.println(in.li_a);
  test3.modifiedFinal(a,in);
  System.out.println(in.li_a);
 } 
}

5、final methods: 可以锁住该method,不让继承类改变其意义(不允许子类覆写);允许编译器对此method作为inline method调用。参考以下例子:

public class Test4{
 private final int li_int=0;
 public final int pub_fi_mod(){
  return li_int;
 }
 protected final int pro_fi_mod(){
  return li_int;
 } 
 private final int pri_fi_mod(){
  return  li_int; 
 }
 private int pri_mod(){
  return li_int; 
 }
}

public class Test5 extends Test4{
 private int li_i=100;
 //下面的方法编译出错,不能覆盖final方法,只针对public和protected,子类中方法与父类中private的方法名相同不是覆盖,与父类中同方法名的方法没有任何关系(除了名字相同)。
 /*
 public int pub_fi_mod(){
  return li_i; 
 }
 protected int pro_fi_mod(){
  return li_i; 
 }*/
 private final int pri_fi_mod(){
  return  li_i; 
 }
 private int pri_mod(){
  return li_i; 
 }
 public static void main(String args[]){
  System.out.println(new Test5().pri_mod()); 
 }
}

6、fianl(method) vs private(method): class所有的private methods自然而然都是final,private methods仅仅是隐藏class中的某段程序代码而已,不能被overrid,即使子类中恰好有同名的method,也不会产生什么效果;其中两者的区别是在子类中可以出现与private方法有相同签名的方法,而public或protected的final方法不能被重写,但允许方法名相同但参数列表不同的重构方法出现。借用以上例子,将Test5修改后编译通过:

public class Test5 extends Test4{
 private int li_i=100;
 //下面的方法编译出错,不能覆盖final方法
 /*
 public int pub_fi_mod(){
  return li_i; 
 }
 protected int pro_fi_mod(){
  return li_i; 
 }
 */
 //但允许参数列表不同的重构方法出现
 public int pub_fi_mod(int a){
  return li_i; 
 }
 protected int pro_fi_mod(int a){
  return li_i; 
 }
 private final int pri_fi_mod(){
  return  li_i; 
 }
 private int pri_mod(){
  return li_i; 
 }
 public static void main(String args[]){
  System.out.println(new Test5().pri_mod()); 
 }
}

7、final classes: 当把一个class声明为final时,也就决定了此class将不能被继承(比如String类,此类为final类,具体可以参见其实现java.lang.String)。final classes的methods可以是final,也可以是非final的;其中的数据成员可以是final的也可以不是,他们将服从final data的原则。参考以下例子:

public final class Test6{
 private final int li_int=0;
 public int li_a=123;
 public final int mod(){
  return li_int;
 } 
 public int pri_mod(){
  return li_a; 
 }
 public static void main(String args[]){
  System.out.println(new Test6().pri_mod()); 
 }
}

//Test6是final类,所以Test7不能继承
public class Test7 extends Test6{
 private int li_int=0;
}

PS:从以上可以看出,final是将一个对象的地址不变,对基本类型的值保持不变(因为基本类型变量指向的物理地址存放value而对象变量指向的物理地址存放对象内容的地址)。

PS:以前读书时老师说java中final定义常量,只说对了一半,对基本类型是对的,对String也是对的,因为String虽然是对象,但不会出现String变量地址不变而其内容发生改变的情况(String是一个整体不能只改变其中的一个字符),所以也是对的,但对其他的对象只能保持其引用地址不变不能保证其内容不变,所以是错的。

再补充一些内容:

1、对final属性在声明时就赋值,而且赋的值是常量的话,那编译器会将所有用到此属性的地方都替换成常量,这个请参考下面的代码:

package com.xx.dryr.test1;

import java.lang.reflect.Field;

public class Test1Class1{

public final int x = 100;

public int f(Test1Class1 t1c11,Test1Class1 t1c12) throws Exception{

int i = t1c11.x;

System.out.println("i's value is "+i);

changeX(t1c11);

int j = t1c12.x;

System.out.println("j's value is "+j);

return j - i;

}

public static void changeX(Test1Class1 t1c1) throws Exception{

Class clazz = t1c1.getClass();

Field fieldX = clazz.getDeclaredField("x");

fieldX.setAccessible(true);

fieldX.setInt(t1c1, 300);

System.out.println("fieldX's vlaue is "+fieldX.getInt(t1c1));

}

public int test() throws Exception{

return f(this,this);

}

public static void main(String[] args) throws Exception{

Test1Class1 t1c1 = new Test1Class1();

System.out.println(t1c1.test());

}

}

运行结果是:

i's value is 100

fieldX's vlaue is 300

j's value is 100

0

虽然在changeX方法中,已经将x的值修改为300,但因为编译时所有使用到x的地方都使用100替换了,所以在运行时再怎么修改x的值都不会对使用到x的地方产生影响。

2、否则,对不是在编译时确定final属性值的情况下,final属性的值是可以改变的。请参考如下代码,对上面的代码稍微做了修改,让final属性x在构造方法中初始化:

package com.xx.dryr.test1;

import java.lang.reflect.Field;

public class Test1Class1{

public final int x ;

public Test1Class1(){

x = 100;

}

public int f(Test1Class1 t1c11,Test1Class1 t1c12) throws Exception{

int i = t1c11.x;

System.out.println("i's value is "+i);

changeX(t1c11);

int j = t1c12.x;

System.out.println("j's value is "+j);

return j - i;

}

public static void changeX(Test1Class1 t1c1) throws Exception{

Class clazz = t1c1.getClass();

Field fieldX = clazz.getDeclaredField("x");

fieldX.setAccessible(true);

fieldX.setInt(t1c1, 300);

System.out.println("fieldX's vlaue is "+fieldX.getInt(t1c1));

}

public int test() throws Exception{

return f(this,this);

}

public static void main(String[] args) throws Exception{

Test1Class1 t1c1 = new Test1Class1();

System.out.println(t1c1.test());

}

}

运行结果是:

i's value is 100

fieldX's vlaue is 300

j's value is 300

200

从上面的例子中可见,final属性的值还是可以被改变的,但只有在特殊情况下(没有在编译时被替换),使用特殊的方式(像反射这样的方式),final属性的值才可以被改变。所以说一般情况下说final属性的值是不允许被修改的还是可以说的,但必须得知道这些例外情况的。

java中final的意义相关推荐

  1. java中常量final的用法_详解Java中final的用法

    本文主要介绍了Java中final的使用方法,final是java的关键字,本文就详细说明一下它的使用方法,需要的朋友可以参考下 概念 final 具有"不可改变的"的含义,可以修 ...

  2. Java中final的三种用法

    在java中final的三种用法: 1. final成员变量 2. final函数 3. final类 final成员变量 当你在类中定义变量时,在其前面加上final关键字,那便是说这个变量一旦被初 ...

  3. Java中final关键字的简介说明

    下文笔者讲述java中final关键字的功能简介说明,如下所示 final关键字功能 final关键字修饰的类.方法.变量都会产生特殊的意义 如:final关键字修饰过后的类不能被子类继承final关 ...

  4. java中final关键字的使用

    final 中文翻译为 最终的,在java中也是较为常用的关键字之一. 在java 中 final 关键字可以修饰  类.方法.变量 final 修饰在类上,则表示该类不能被继承,如果里面的成员变量没 ...

  5. JAVA中Final的用法

    JAVA中Final的用法 1. 修饰基础数据成员的final 这是final的主要用途,其含义相当于C/C++的const,即该成员被修饰为常量,意味着不可修改.如java.lang.Math类中的 ...

  6. 转 Java中final、finally、finalize的区别与用法

    Java中final.finally.finalize的区别与用法 1.简单区别: final用于声明属性,方法和类,分别表示属性不可交变,方法不可覆盖,类不可继承. finally是异常处理语句结构 ...

  7. [转载] java中final,finally,finalize三者的作用和区别

    参考链接: Java中final,finally和finalize Final是一个修饰符: 当final修饰一个变量的时候,变量变成一个常量,它不能被二次赋值 当final修饰的变量为静态变量(即由 ...

  8. [转载] Java中final关键字

    参考链接: Java中的final关键字 文章目录 1 final关键字1.1 final修饰类1.2 final修饰方法1.3 final修饰属性---常量1.3.1 final修饰普通数据类型的成 ...

  9. ACAC java中final关键字

    java中final关键字/*1.final关键字:java 中的一个关键字,最终的,不可变的可以修饰变量以及方法,还有类等.1.修饰的方法,修饰的类无法被覆盖,无法被重写,无法被继承写的方法不希望被 ...

最新文章

  1. python 图像分析自然纹理方向与粗细代码_python skimage图像处理(二)
  2. Java学习第三天160818 表单 框架 下拉列表等
  3. 震惊!Canvas原来还能这么搞!代码画一个时钟出来
  4. OpenGL生成的法线贴图并增加光照
  5. SpringMvc执行过程
  6. Sqlite 管理工具收藏
  7. 基于狄利克雷-多项式分布做文档聚类代码(dirichlet multinomial mixture model)
  8. java技术不行有复试怎么办_复试答不上来怎么办?记住这四点不慌!
  9. 来自Unix/Linux的编程启示录
  10. 2022最新RTMP+HTTP直播地址汇总(亲测可用)
  11. SpringBoot+RabbitMQ 实现 RPC 调用
  12. 这个小众副业,一次200,有人月入3万!
  13. Java集合源码剖析——基于JDK1.8中LinkedList的实现原理
  14. opengl 知识点2
  15. 如何用Python画滑稽笑脸
  16. 联想主板bios设置u盘启动项的方法怎么操作
  17. 搭建自己的简易服务器(公网)
  18. curl 实现qq挂号登录
  19. 5分钟使用ssl证书免费配置任意域名的 https
  20. C Primer Plus 学习打卡之第二章(含课后编程答案)

热门文章

  1. python爬虫 爬取有道翻译详解
  2. Python TCP聊天器
  3. Java判断类和实例的关系
  4. JAVA线程间协作:Condition
  5. C++98C++11的区别
  6. 模拟实现mapset
  7. 音视频技术开发周刊 | 221
  8. 闭关休养or趁火打劫:疫情之下焦点行业网络威胁分析
  9. 透过新硬件环境下的存储技术,看未来数据库系统崛起(附PPT)
  10. PMP 之程序性计划、实体性计划、综合计划