final关键字

final代表最终的,不可改变的。其可以修饰类,方法,变量(变量的范围要比属性的范围的大)。

       1  final戏称断子绝孙类,也就是子类无法继承。final的方法不能够被重写,final的变量只能赋值一次(这里有一点需要强调的是,fanal修饰的是变量而非常量),即在初始化的时候,而且还必须赋值。

        2final修饰的类中没有set属性,由于final修饰的属性只能够赋值一次,所以不可能有set方法。如果有的话,试想当你new出一个该类的对象的时候,这个对象是能够调用set方法无数多次的,显然与final修饰的属性赋值一次相矛盾。因此,final的属性是没有set方法的。

final修饰的属性为什么不是通常意义下的常量呢?也许你会有这样的疑问,既然final修饰的属性只能赋值一次,有不能把其值给改变,为什么还不能把他当作常量?又这样的疑问是正确的。但是你可以转换一个角度想一下。针对一个对象来说,其值的确无法改变,但是当你重新new一个对象的时候,final修饰的变量一定还是可以在赋值的啊!这也就是说final在一个类中只能赋值一次,当你创建对象的时候,需要调用这个类的构造,而final修饰的变量是可以放在构造当中的,创建出来的对象,调用final修饰的属性的时候,这个属性的值不能总是相同。在这样情况下,的确是不能够把final修饰的东西当作常量来看的。试想一下如果一个常量 2,不过其对任何一个对象来说,其值永远是2,绝对是不会改变的,这样的才算是常量。

      3你也许看到过Java中在定义常量的时候,有final的影子,的确没错。你想一想,定义的常量一定是这样出现的啊:public static final   A_B = 4;这里需要解释一下的是,是否会问为什么要用public修饰而不用private或者protected来修饰呢?因为是常量,没必要隐藏,索性公开算了,这才符合常量的真正的意义。
public class Custom {
	private final int id;
	private static int i;
	public Custom() {
		id = ++i;
	}
	public int getId(){
		return id;
	}
}
public class Test {
	public static void main(String[] args){		
		for(int i=0;i<10;i++){
			Custom s = new Custom();
			System.out.println(s.getId());
		}
	}
}

这段代码编译和运行都能过去。我们分析一下,id是用final来修饰的,而i是用static来修饰的,我们看到了,id的确只能在创建对象的时候赋值一次,这没错,这就告诉我们id的值只有在创建对象的时候才能改变,也就印证了上面的说法,final修饰的属性不是常量。而static修饰的属性i,识别new出来对象所共享的,放在静态池中。放在静态池中的变量,我以前也提到过其特点:常驻内存,反复利用。这两个关键在上面的代码中同时出现,分别用来修饰两个变量,只是为了更好地证明final修饰的属性id只能赋值一次,而且是可以改变的。

补充一点:在引用类型当中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(); } } ``` ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值