先通过例子看一看:

package com.sotaof.testfinal;

public class Value {
int i;
public Value(int i){
this.i = i;
}
}
package com.sotaof.testfinal;

import java.util.Random;

public class FinalData {

	private static Random rand = new Random(47);
private String id;
public FinalData(String id) {
this.id = id;
}
private final int valueOne = 9;
private static final int VALUE_TWO = 99;
public static final int VALUE_THREE = 39; private final int i4 = rand.nextInt(20);
static final int INT_5 = rand.nextInt(20);
private Value v1 = new Value(11);
private final Value v2 = new Value(22);
private static final Value VAL_3 = new Value(33); private final int[] a = {1,2,3,4,5,6};
public String toString() {
return id + ": " + "i4 = " + i4 + ", INT_5 = " + INT_5;
}
public static void main(String[] args) {
FinalData fd1 = new FinalData("fd1");
//fd1.valueOne++; 这句代码会报错,因为final修饰的基本类型数值是不可以变化的
fd1.v2.i++; //这句代码不会报错,因为final修饰的v2对象自身是可以变化的
fd1.v1 = new Value(9); //这个代码是不会报错的,就是创建一个对象,v1没有用final修饰
for(int i = 0; i < fd1.a.length; i++) {
fd1.a[i]++; //这句代码也是不会报错的,因为数组也是对象,而final修饰的对象自身可以变化
}
//fd1.v2 = new Value(0); 这句代码会报错 ,因为v2是用final修饰的,final修饰的对象自身是可以变化的,
//但是final修饰的对象的引用是不可以变化的,所以再用v2创建对象,就说明v2的引用变化了,所以报错
//fd1.VAL_3 = new Value(1);和上边一样的道理
//fd1.a = new int[3];和上边一样的道理
//刚才那三段代码都是一个意思,就是说final修饰的对象,这个对象的引用是不可以变化的
System.out.println(fd1);
System.out.println("Creating new FinalData");
FinalData fd2 = new FinalData("fd2");
System.out.println(fd1);
System.out.println(fd2);
} }

1、我们不能因为某数据是final的就认为在编译时可以知道它的值。在运行时使用随机生成的数值来初始化i4和INT_5就说明了这一点。

2、这个实例也向我们展示了将final数值定义为静态和非静态的区别,这个区别只有当数值 在运行时内 被初始化时 才会显示(这句话不容易理解,我将其用" "分开了,以方便阅读理解,细细品读),这是因为编译器对编译时数值一视同仁,当运行程序时就会看到这个区别。

3、在fd1对象1中i4的值是唯一的,在fd2对象中i4的值也是唯一的,但是fd1和fd2中的i4值就不一定相同了,因为如果只有final修饰的话,在每次创建对象时都会被初始化(网友解释:i4仅仅是final类型的,分别在fd1和fd2两个对象中各开辟一块空间);

4、不管是在fd1中,还是在fd2中,INT_5的值都是一样的,是不可以通过创建第二个FinalData对象而加以改变的,这是因为它是static的,在装载的时候,就被初始化,而不是每次创建新对象时都初始化(网友解释:static final 仅使用一段不能改变的空间,所以INT_5值不变)。

在这里为了方便大家更多的了解final的数据,这里附上一位网友的例子,觉得很不错(http://chihom.iteye.com/blog/233035)

下面的两个初始化例子

class InitalizedClass {
static {
System.out.println("You have initalized InitalizedClass!");
}
public static int inititalize_varible = 1; } public class TestInitializeClass {
public static void main(String[] args) {
System.out.println(InitalizedClass.inititalize_varible); } }

上面的结果是: 


You have initalized nitalizedClass! 


1

再看下面的例子

class InitalizedClass {
static {
System.out.println("You have initalized InitalizedClass!");
}
//和上面的例子唯一的差异就是此处的变量INITIALIZED_VARIBLE被声明为final
public final static int INITIALIZED_VARIBLE = 1; } public class TestInitializeClass {
public static void main(String[] args) {
System.out.println(InitalizedClass.INITIALIZED_VARIBLE); } }

上面的结果是: 


1

这里大家会问:为什么两个例子执行结果不一样?

原因是第二个例子中的INITIALIZED_VARIBLE为编译期常量,它不会导致类的初始化,也就是说,我第一个例子中,"InitalizedClass.inititalize_varible"这句代码会进入InitalizedClass这个类中,并执行static那段儿代码;而第二个例子中,"InitalizedClass.INITIALIZED_VARIBLE"这句代码并不会进入InitalizedClass这个类中,那有人会说,不进入这个类中怎么可能调用到INITIALIZED_VARIBLE这个值呢?因为我这个程序在编译的时候就初始化了,内存中已经存在INITIALIZED_VARIBLE这个值了,所以不再需要调用InitalizedClass类,这样大家应该理解了吧。

我也是今天才有了一点儿的理解,不过还是有一个疑问,希望各位网友能帮忙解决:

private final int valueOne = 9;

private final int i = rand.nextInt(20);

这两个代码,分别是在什么时候初始化的,我的理解是,valueOne和 i 都是编译时常量,但valueOne在编译的时候就初始化了,编译的时候就知道valueOne的值了,而 i 则是在运行时进行初始化的,运行的时候才会知道它的值。

final的深入理解 - final数据的更多相关文章

  1. 深入理解final和static关键字

    深入理解final和static关键字 参考:http://blog.csdn.net/qq1028951741/article/details/53418852 final关键字 final关键字可 ...

  2. 面试题系列:工作5年,第一次这么清醒的理解final关键字?

    面试题:用过final关键字吗?它有什么作用 面试考察点 考察目的: 了解面试者对Java基础知识的理解 考察人群: 工作1-5年,工作年限越高,对于基础知识理解的深度就越高. 背景知识 final关 ...

  3. 深入理解final关键字

    在了解了final关键字的基本用法之后,这一节我们来看一下final关键字容易混淆的地方. 1.类的final变量和普通变量有什么区别? 当用final作用于类的成员变量时,成员变量(注意是类的成员变 ...

  4. java基础4:深入理解final关键字

    本文主要介绍了final关键字的使用方法及原理 具体代码在我的GitHub中可以找到 https://github.com/h2pl/MyTech 文章首发于我的个人博客: https://h2pl. ...

  5. Java面试题,深入理解final关键字

    final关键字 final的简介 final可以修饰变量,方法和类,用于表示所修饰的内容一旦赋值之后就不会再被改变,比如String类就是一个final类型的类. final的具体使用场景 fina ...

  6. 深入理解final关键字以及一些建议

    引子:一说到final关键字,相信大家都会立刻想起一些基本的作用,那么我们先稍微用寥寥数行来回顾一下. 一.final关键字的含义 final是Java中的一个保留关键字,它可以标记在成员变量.方法. ...

  7. Java中的内存处理机制和final、static、final static总结

    Java中的内存处理机制和final.static.final static总结   装载自:http://blog.csdn.net/wqthaha/article/details/20923579 ...

  8. Java第11次作业:什么是继承?继承的好处?什么是覆写?super()?构造代码块?子父类初始化顺序? 抽象类能用final声明吗?final关键字声明类 方法 变量以及全局常量?抽象类的构造方法?

    什么是继承? 继承是以父类为基础,子类可以增加新的数据或新的功能.子类不能选择性地继承父类.这种技术使得复用以前的代码非常容易. JAVA不支持多继承,单继承使JAVA的继承关系很简单,一个类只能有一 ...

  9. final修饰符(5)-final方法

    final修饰的类的方法不能被重写,例如如果父类不希望子类重写某个方法,则可以使用final修饰符修饰该方法 在java的Object类里面有一个final方法:getClass(),因为Java不希 ...

随机推荐

  1. 解决IE11只能用管理员身份运行的问题

    解决IE11只能用管理员身份运行的问题 IE11 打不开,必须要用管理员身份运行才可以打开,而且重置浏览器这个方法也不奏效. 今天本人也遇到了,上网查找发现是注册表权限的问题,原因尚不明确,安装了或被 ...

  2. Socket编程(摘抄)

    http://www.blogjava.net 例子代码就在我的博客中,包括六个UDP和TCP发送接受的cpp文件,一个基于MFC的局域网聊天小工具工程,和此小工具的所有运行时库.资源和执行程序.代码 ...

  3. YUI之数组操作

    YUI的构建数组,将类数组转换成真正的数组,从而可以使用数组的所有方法   数组构建 //真正的数组返回1,类数组返回2,其余的返回0 YArray.test = function (obj) { v ...

  4. postman接口测试工具3.0版本的坑

    今天用postman接口测试工具3.0版本被坑,找了半天,原来postman这个新版本有个坑啊 下面的get参数,第一行不管你填不填,都是无效的,可能是postman的一个bug吧

  5. HTML5 canvas 在线画笔绘图工具(一)

    HTML5 canvas 在线画笔绘图工具(一) 功能介绍 这是我用Javascript写的第一个程序,在写的过程中走了很多弯路,所以写完之后想分享出来,给与我一样的初学者做为学习的参考,同时在编写这 ...

  6. linux下安装pdf

    官方下载地址:http://www.foxitsoftware.cn/downloads/ 问题:下载官方包以后解压,双击不能打开,也没有任何提示. 用teminal 来打开foxitreader,t ...

  7. 蜘蛛牌(hdu 1584 DFS)

    蜘蛛牌 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  8. 海量数据处理利器greenplum——初识

    简介及适用场景 如果想在数据仓库中快速查询结果,可以使用greenplum. Greenplum数据库也简称GPDB.它拥有丰富的特性: 第一,完善的标准支持:GPDB完全支持ANSI SQL 200 ...

  9. Bootstrap 模态对话框只加载一次 remote 数据的解决办法

    原文: https://my.oschina.net/qczhang/blog/190215?p=1

  10. C#安装程序制作让安装后的程序开机自动运行

    1.创建安装项目后要在自己的解决方案是添加一个新的类库项目(ClassLibrary1),并在新类库中添加一下安装程序类(Installer1),在Installer1类中添加如下代码: string ...