final关键字

本文详细解析了Java中的final关键字,包括其在类、方法和变量上的作用。final修饰类能防止继承,修饰方法禁止重写,修饰变量确保不可变性。在匿名内部类中,final关键字用于确保局部变量在内部类中的可见性和一致性。虽然在JDK8之后,final修饰局部变量的显式要求放宽,但其保证数据一致性的核心作用仍然重要。

final的作用

静态方法可以修饰类、静态成员变量、普通成员变量、普通方法、局部变量
(1)final修饰类的作用:阻止当前类被继承
(2)final修饰的方法的作用:阻止子类继承和重写【注意:子类可以重载该方法】
(3)final修饰的成员变量初始化后便不可用修改,基本数据类型不能修改值,引用数据类型实际是引用不能修改修改,但引用指向存储对象空间中的内存可以被修改

final不能修饰抽象类

抽象类是用于被子类继承的,和final修饰类的作用冲突

静态方法没有必要修饰静态方法

静态方法可以修饰静态方法,但没有必要,因为final也无法阻止子类申明相同签名的的静态方法。父类的静态方法可以被子类继承,不能被子类重写,但子类可以申明相同同名的方法,父类的静态方法只是被隐藏了而不是被重写了。

匿名内部类访问方法中的局部变量为什么必须被final修饰

  • 在JDK8之前,如果在匿名内部类中需要访问局部变量,那么这个局部变量必须用final修饰符显示修饰;
  • 在JDK8之前,如果在匿名内部类中需要访问局部变量,那么这个局部变量一定是final修饰的,但final关键字可以省略

分析final修饰的变量如何传入匿名内部类

public class Demo {
    public static void main(String[] args) {
        int a = 10;
        String b = "abc";

        Thread thread = new Thread(() -> {
            System.out.println(a + b);
        });
        thread.start();
    }
}

反编译后的关键代码如下:
在这里插入图片描述
发现方法中的两个局部变量,作为参数传入了内部类中,事实上最符合反编译代码表述的是:

public class Demo.lambda$main$0 extends Thread {
	
	private int a;
	private String b;
	
	Demo.lambda$main$0(int a,String b) {
		this.a=a;
		this.b=b;
	}
 
	public void run() {
		System.out.println( a + b );
	}
}

final修饰的局部变量是作为匿名内部类构造器的参数传入匿名内部类内部的

为啥需要final修饰局部变量

final修饰局部变量还是防止数据被修改,保证数据的一致性,对引用变量来说是引用地址的一致性,对基本类型来说就是值的一致性。

  • 局部变量通过匿名内部类的构造器的参数参入匿名类中,如果是基本数据,实际传递的是值,如果是引用数据类型,实际传递的是引用,匿名内部类中实际是存了局部变量的副本。
  • 匿名内部类中和方法中任意一处修改了,另一方都不受影响,这样就会导致
    数据不一致,从而影响运行结果,所以需要final修饰。【注意:如果是引用数据类型,这里修改是指的引用,如果是修改引用数据类型的成员变量,实际上双方是一致的,修改的是堆内存中的数据,而不是虚拟机栈中的】
    在这里插入图片描述
### Final 关键字的用法及作用 #### 1. Final 修饰变量 当 `final` 用于修饰变量时,表示该变量一旦被初始化后就不能再更改其值。如果 `final` 修饰的是基本数据型的变量,则其值不可改变;如果 `final` 修饰的是引用型的变量,则该引用不能指向其他对象,但对象本身的内容是可以修改的[^4]。 ```java public class TestFinal { final int x = 10; final StringBuilder sb = new StringBuilder("Hello"); public void modify() { // x = 20; // 编译错误:无法为 final 变量赋值 sb.append(" World"); // 合法:可以修改对象内容 } } ``` #### 2. Final 修饰方法 当 `final` 用于修饰方法时,表示该方法不能被子重写。这通常用于确保某些方法的行为在继承体系中保持不变。例如,如果一个方法的逻辑是关键且不应被修改,则可以将其声明为 `final`[^3]。 ```java class SuperClass { final void display() { System.out.println("This is a final method."); } } class SubClass extends SuperClass { // void display() { } // 编译错误:无法重写 final 方法 } ``` #### 3. Final 修饰 当 `final` 用于修饰时,表示该不能被继承。这种用法通常用于定义一些工具或核心,防止其被扩展或篡改[^3]。 ```java final class UtilityClass { public void performAction() { System.out.println("Utility class action."); } } // class DerivedUtility extends UtilityClass { } // 编译错误:无法继承 final ``` #### 4. Final 在多线程环境中的作用 `final` 关键字在 Java 内存模型(JMM)中具有特殊的意义。它确保了 `final` 域的可见性,即在构造函数内对 `final` 域的写入与随后将被构造对象的引用赋值给引用变量之间不会发生重排序。此外,初次读取包含 `final` 域的对象引用与随后读取该 `final` 域之间也不会发生重排序[^2]。 ```java class FinalExample { final int value; FinalExample(int val) { this.value = val; } } ``` #### 5. Final 与代码块 `final` 还可以用于局部变量或代码块中。局部变量被声明为 `final` 后,其值只能被赋值一次。此外,`final` 修饰的变量可以在匿名内部中使用[^4]。 ```java public class FinalCodeBlock { public static void main(String[] args) { final int number = 10; Runnable runnable = new Runnable() { public void run() { System.out.println(number); // 合法:final 变量可以在匿名内部中使用 } }; runnable.run(); } } ``` ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值