欢迎来到51自学网!

51自学网

当前位置: 主页 > 脚本专题 >

JAVA中的final关键字用法实例详解

时间:2018-07-27 15:45来源:网络整理 作者:51自学网
本文实例讲述了JAVA中的final关键字用法。分享给大家供大家参考,具体如下: 根据上下文环境,java的关键字final也存在着细微的区别,但通常指的是“这是无法改变的。”不想改变的理

本文实例讲述了JAVA中的final关键字用法。分享给大家供大家参考,具体如下:

根据上下文环境,java的关键字final也存在着细微的区别,但通常指的是“这是无法改变的。”不想改变的理由有两种:一种是效率,另一种是设计。由于两个原因相差很远,所以关键子final可能被误用。

接下来介绍一下使用到final的三中情况:数据,方法,类

final数据

许多编程语言都有某种方法,来向编译器告知一块数据是恒定不变的。有时数据的恒定不变是很有用的,例如:

1. 一个编译时恒定不变的常量
2. 一个在运行时初始化,而你不希望它被改变。

对于编译期常量的这种情况,编译器可以将该常量值代入任何可能用到它的计算式中,也就是说,可以在编译期就执行计算式,这减轻了一些运行时的负担。在java中,这类常量必须是基本类型,并且以final表示。在对这个常量定义时,必须进行赋值。

一个即是static又是final的域只占一段不能改变的存储空间。

当final应用于对象引用时,而不是基本类型时,其含义有些让人疑惑。对基本类型使用final不能改变的是他的数值。而对于对象引用,不能改变的是他的引用,而对象本身是可以修改的。一旦一个final引用被初始化指向一个对象,这个引用将不能在指向其他对象。java并未提供对任何对象恒定不变的支持。这一限制也通用适用于数组,它也是对象。例如:

package finalPackage; import java.util.*; class Value { int i; public Value(int i) { this.i = i; } } /** * final数据常量 * @author Administrator * 对基本类型使用final不能改变的是它的数值。 * 而对于对象引用,不能改变的是他的引用,而对象本身是可以修改的。 * 一旦一个final引用被初始化指向一个对象,这个引用将不能在指向其他对象。 * 注意,根据惯例,即是static又是final的域(即编译器常量)将用大写表示,并用下划分割个单词。 */ public class FinalData { private static Random rand = new Random(47); private String id; public FinalData(String id) { this.id = id; } // 编译时常量 Can be compile-time constants: private final int valueOne = 9; private static final int VALUE_TWO = 99; // 典型的公共常量 Typical public constant: public static final int VALUE_THREE = 39; // 运行时常量 Cannot be compile-time constants: 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); // 数组 Arrays: 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++; // Error: can't change value fd1.v2.i++; // Object isn't constant! fd1.v1 = new Value(9); // OK -- not final for (int i = 0; i < fd1.a.length; i++) fd1.a[i]++; // Object isn't constant! // ! fd1.v2 = new Value(0); // Error: Can't // ! fd1.VAL_3 = new Value(1); // change reference // ! fd1.a = new int[3]; System.out.println(fd1); System.out.println("Creating new FinalData"); FinalData fd2 = new FinalData("fd2"); System.out.println(fd1); System.out.println(fd2); } /** * 输出结果: * fd1: i4 = 15, INT_5 = 18 * Creating new FinalData * fd1: i4 = 15, INT_5 = 18 * fd2: i4 = 13, INT_5 = 18 */ }

由于valueOne和VALUE_TWO都是带有编译时数值的final基本类型,所以它们二者均可以用作编译期常量,并且没有重大区别。VALUE_THREE是一种更加典型的对常量进行定义的方式:定义为public,可以被任何人访问;定义为static,则强调只有一份;定义为final,这说明它是个常量。请注意带有恒定初始值的final static基本类型全用大写字母命名,并且字母与字母之间用下划线隔开。

我们不能因为某些数据是final的就认为在编译时可以知道它的值。在运行时使用随机数来初始化i4和INT_5的值说明了这一点。实例中fd1和fd2中i4的值是唯一的,每次都会被初始化为15,13。INT_5的值是不可以通过创建第二个FinalData对象加以改变的。这是因为他是static的,在装载类时(也就是第一次创建这个类对象时)已经被初始化,而不是每次创建都初始化。

java也许生成"空白final",所谓空白final是指被声明为final但又未给初值的域。无论什么情况下编译器都会保证final域在使用前初始化。但空白final在final的使用上提供了很大的灵活性,为此,一个final域可以根据某些对象有所不同,却又保持恒定不变的特性。下面的事例说明了一点:

(责任编辑:admin)

织梦二维码生成器
顶一下
(0)
0%
踩一下
(0)
0%
------分隔线----------------------------
栏目列表
推荐内容