在java中,final的含义在不同的场景下有细微的差别,但总体上来说,它指的是“这是不可变的”。下面,我们来讲final的四种主要用法。
final数据
在编写程序时,我们经常需要说明一个数据是不可变的,我们称为常量。在java中,用final关键字修饰的变量,只能进行一次赋值操作,并且在生存期内不可以再次赋值。更重要的是,final会告诉编译器,这个数据是不会修改的,那么编译器就可能会在编译时期就对该数据进行替换甚至执行计算,这样可以对我们的程序起到一点优化。不过在针对基本类型和引用类型时,final关键字的效果存在细微差别。我们来看下面的例子:
package s1;
public class L {
public static void main(String[] args) {
final int i1 = 1; //定义变量并直接赋常量,此时为编译时常量
i1 = 11;//编译出错,因为final数据只能赋值一次
final int i2; //定义变量,但先不赋值;在后面需要的地方再赋值,此时不是编译时常量
i2 = 2;
i2 = 12;//编译出错,因为final数据只能赋值一次
//此处i3虽然是final并赋值,但是赋的并不是常量,所以i3不是编译时常量
final int i3 = (int) (Math.random() * 10);
i3 = 13;//编译出错,因为final数据只能赋值一次
//上面部分是基本类型的例子,下面看看有关对象
final Value value1 = new Value();
value1 = new Value();//编译出错,因为final数据只能赋值一次
//下面这样是可以的,因为value1并没有改变(没有再次赋值),改变的仅仅是value1对象里面的value值
value1.value = 1;
value1.value = 1;
}
}
class Value {
int value;
}
空白final
java允许“空白final”,所谓空白final是指被声明为final但又未赋初始值的域。无论什么情况,编译器都确保空白final在使用前必须初始化。但是空白final在关键字final的使用上提供了更大的灵活性,为此,一个类中的final域就可以做到根据不同对象而有所不同,却又保持其恒定不变的特性。下面举个列子:
package s1;
public class L {
public static void main(String[] args) {
Value value1 = new Value(10);
value1.maxValue = 20;//编译出错,因为final数据只能赋值一次
Value value2 = new Value(20);
}
}
class Value {
final int maxValue;
Value(int value) {
this.maxValue = value;
}
}
final方法
使用final方法的原因主要是把方法锁起来,以防止继承者修改它的含义。这是出于设计的考虑:想要确保在继承中使方法行为保持不变,并且不会被覆盖。例子:
package s1;
public class L {
public static void main(String[] args) {
}
}
class L1 {
final void calculate() {
System.out.println(getClass().getSimpleName() + "calculate");
}
}
class L2 extends L1 {
void calculate() { //编译出错,不能覆盖L1中的calculate方法,因为它是final的
}
}
final和private关键字
类中的所有private方法都隐式地指定为final方法。由于子类无法调用访问父类的private方法,所以也就无法覆盖它,也就无法实现多态(关于多态,可以参考https://blog.csdn.net/GracefulGuigui/article/details/103869327)。可以对private方法添加final修饰词,但这并不会给该方法增加任何额外的意义。例子:
package s1;
public class L {
public static void main(String[] args) {
L1 l1 = new L2();
l1.test();
}
}
class L1 {
void test() {
calculate1();
calculate2();
}
private final void calculate1() {
System.out.println("L1 calculate1");
}
void calculate2() {
System.out.println("L1 calculate2");
}
}
class L2 extends L1 {
// 不会报错,虽然L1中calculate1方法是final,但是private限制了L2无法访问L1的calculate1,所以可以起同名+同参数列表的方法
void calculate1() {
System.out.println("L2 calculate1");
}
void calculate2() {
System.out.println("L2 calculate2");
}
}
运行结果:
L1 calculate1 //由于是private方法,不支持多态
L2 calculate2 // 因为多态,调用的实际是L2的calculate2
final类
final类比较简单易懂,被final修饰的类,则表明该类无法被其它类继承。
陈贵贵
发布了64 篇原创文章 · 获赞 4 · 访问量 1万+
私信
关注
标签:Java,void,Value,class,详解,L1,final,赋值
来源: https://blog.csdn.net/GracefulGuigui/article/details/103935349