一、final修饰符概述

1. final可以修饰类、变量和方法

2. final修饰的类、变量和方法不可改变

3. 不允许为final变量重新赋值,子类不允许覆盖父类的final方法,final类不能派生子类

4. 通过使用final关键字,允许Java实现不可变类,不可变类会让系统更加安全

二、final成员变量

1. 对于final修饰的成员变量而言,一旦有了初始值,就不能被重新赋值

2. 由于成员变量不一定需要显式初始化,故那些既没有在定义时指定初始值,也没有在初始化块、构造器中指定初始值的final成员变量的值将一直是系统默认分配的0、'\u0000'、false或null,这些final成员变量也就完全失去了存在的意义,因此Java规定:final修饰的成员变量必须由程序员显式地指定初始值

  • final类变量:必须在静态初始化块中指定初始值或声明该类变量时指定初始值,而且只能在两个地方的其中之一指定
  • final实例变量:必须在非静态初始化块、声明该实例变量或构造器中指定初始值,而且只能在三个地方的其中之一指定
	// 定义final成员变量时指定默认值
final int a = 6; // 下面final成员变量将在构造器或初始化块中分配初始值
final String str;
final int c;
final static double d; // 下面定义的ch实例变量是不合法的,因为没有显式指定初始值
// final char ch; // 初始化块,可对没有指定默认值的实例变量指定初始值
{
str = "Hello";
} // 静态初始化块,可对没有指定默认值的类变量指定初始值
static
{
d = 3.14;
} // 构造器,可对既没有指定默认值,又没有在初始化块中指定初始值的实例变量指定初始值
public FinalVariableTest()
{
c = 5;
}

三、final局部变量

1. 相比于成员变量,系统不会对局部变量进行初始化,局部变量必须由程序员显式初始化,因此使用final修饰局部变量时,既可以在定义时指定默认值,也可以不指定默认值

  • 如果在定义时指定了默认值,则后面代码中不能再对该变量赋值
  • 如果在定义时未指定默认值,则可以在后面代码中对该final变量赋初始值,但只能一次,不能重复赋值

四、final变量→宏变量

1. 当某个final变量满足以下两个条件时,它就不再是一个变量,而是相当于一个直接量(即宏变量)

  • 在该final变量定义时指定了初始值
  • 该初始值可以在编译时就被确定下来

2. 除了为final变量赋值时赋直接量的情况外,如果被赋的表达式只是基本的算术表达式或字符串连接运算,没有访问普通变量,也没有调用方法,Java编译器同样会将这种final变量当成“宏变量”处理

		// ex:下面定义了4个final“宏变量”
final int a = 5;
final int b = 3 + 2;
final double d = 1.2 / 3;
final String str = "我爱" + "China"; // 下面的ss变量的值因为调用了方法,所以无法在编译时被确定下来,ss也就不会被当成“宏变量”处理
final String ss = "我爱China" + String.valueOf(1314);

  

五、final方法

1. final修饰的方法不可被重写,但完全可以被重载

2. 由于子类无法重写父类的private方法(如果子类中定义一个与父类private方法有相同方法名、相同形参列表、相同返回值类型的方法,也不是方法重写,只是重新定义了一个新方法),因此即使使用final修饰一个private方法,依然可以在其子类中定义与该方法具有相同方法名、相同形参列表、相同返回值类型的方法

public class PrivateFinalMethodTest
{
private final void test() {}
} class sub extends PrivateFinalMethodTest
{
// 下面的方法定义不会出现问题
public void test() {}
}

注:虽然子类和父类都包含了同名的void test()方法,但子类并不是重写父类的方法,因此即使父类的void test()方法使用了final修饰,子类中依然可以定义void test()方法。

六、final类

1. final修饰的类不可以有子类,所以若某个类不想被继承,则可以使用final修饰该类

public final class FinalClass
{
} // 下面的类定义将出现编译错误
class Sub extends FinalClass
{
}

七、不可变类

1. 不可变类是指创建该类的实例后,该实例的实例变量是不可改变的

2. Java提供的8个包装类和java.lang.String类都是不可变类

3. 如果要创建自定义的不可变类,可遵守如下规则:

  • 使用private和final修饰符来修饰该类的成员变量
  • 提供带参数构造器,用于根据传入参数来初始化类里的成员变量
  • 仅为该类的成员变量提供getter方法,不要为该类的成员变量提供setter方法
  • 如果有必要,重写Object类的hashCode()和equals()方法
/*
* 定义一个不可变类Address
*/
public class Address
{
// 不可变类的实例的实例变量不可改变
private final String detail;
private final String postCode; public Address()
{
detail = "";
postCode = "";
} public Address(String detail, String postCode)
{
this.detail = detail;
this.postCode = postCode;
} public String getDetail()
{
return this.detail;
}
public String getPostCode()
{
return this.postCode;
} // 重写equals()方法,判断两个对象是否相等
public boolean equals(Object obj)
{
// ...
}
public int hashCode()
{
return detail.hashCode() + postCode.hashCode() * 31;
}
}

补:对于上面的Address类,当程序创建了Address实例后,将无法修改该Address实例的detail和postCode实例变量。

4. 不可变类的实例在整个生命周期中永远处于初始化状态,它的实例变量不可改变,因此对不可变类的实例的控制将更加简单

  • 不可变类的实例状态不可改变,可以很方便地被多个对象所共享

12. final修饰符的更多相关文章

  1. 类成员(static)和final修饰符

    在Java类里只能包含成员变量.方法.构造器.初始化块.内部类(包括接口.枚举)5种成员,类成员是用static来修饰的,其属于整个类. 当使用实例来访问类成员时,实际上依然是委托给该类来访问类成员, ...

  2. Java final 修饰符知识点总结

    final从字面上理解含义为“最后的,最终的”.在Java中也同样表示出此种含义. final可以用来修饰变量(包括类属性.对象属性.局部变量和形参).方法(包括类方法和对象方法)和类. 1. fin ...

  3. Java中的final修饰符

    1.什么时候可以选择final修饰符 如果想让一个类不被其他类继承,不允许在有子类,这时候就要考虑用到final来修饰. 2.用final修饰的类 首先大家要明白,用final修饰的类是不能被继承的, ...

  4. 对于形式参数只能用final修饰符,其它任何修饰符都会引起编译器错误

    在Java中修饰符总共有一下几种: 1.访问控制修饰符    分别有:public private protected,缺省 2.其它修饰符      分别有:abstract,final,stati ...

  5. private static final 修饰符

    java修饰符分类修饰符字段修饰符方法修饰符根据功能同主要分下几种 1.权限访问修饰符 public,protected,default,private,四种级别修饰符都用来修饰类.方法和字段 包外 ...

  6. JAVA基础-栈与堆,static、final修饰符、内部类和Java内存分配

    Java栈与堆 堆:顺序随意 栈:后进先出(Last-in/First-Out). Java的堆是一个运行时数据区,类的对象从中分配空间.这些对象通过new.newarray.anewarray和mu ...

  7. as3 中 final 修饰符

    现在,在ActionScript 3.0的修饰符中,只有final修饰符没有介绍.之所有放在这里介绍,是因为final修饰符只与继承有关,指定一个方法不能被重写或一个类不能被继承. 一般来说,当用fi ...

  8. final修饰符:

    知识点: 1.final关键字用于修饰类.变量和方法 2.有点类似C#里的 sealed 关键字,用于表示它修饰的方法.变量和类不可以再被改变 3.final修饰变量时,表示该变量一旦获取了初始值,就 ...

  9. Java中final修饰符深入研究

    一.开篇 本博客来自:http://www.cnblogs.com/yuananyun/ final修饰符是Java中比较简单常用的修饰符,同时也是一个被"误解"较多的修饰符.对很 ...

随机推荐

  1. C#&.Net干货分享- iTextSharp导出数据源到PDF

    namespace Frame.ITextSharp{    /// <summary>    /// iTextSharp导出数据源到PDF    /// </summary> ...

  2. JVM java内存区域的介绍

    jvm虚拟机在运行时需要用到的内存区域.广泛一点就是堆和栈,其实不然,堆和栈只是相对比较笼统的说法,真正区分有如下几个 先上图一: 总的就是 java的内存模型 内存模型又分堆内存(heap)和方法区 ...

  3. JVM 类的加载机制

    在对类的实例化之前.JVM 一般会先进行初始化 主要经过如下几个阶段: 1.加载                       类加载的第一阶段,类加载时机有两个: 1.预加载:当虚拟机启动时,会预加载 ...

  4. Codeforces Round #583 (Div. 1 + Div. 2, based on Olympiad of Metropolises)

    传送门 A. Optimal Currency Exchange 枚举一下就行了. Code #include <bits/stdc++.h> using namespace std; t ...

  5. HDL的三种描述方式

    结构化描述 结构化描述方式是最原始的描述方式,是抽象级别最低的描述方式,但同时也是最接近于实际的硬件结构的描述方式.结构化的描述方式,思路就像在面包板上搭建数字电路一样,唯一的不同点就是我们通过HDL ...

  6. LG5202 「USACO2019JAN」Redistricting 动态规划+堆/单调队列优化

    问题描述 LG5202 题解 \[opt[i]=xx+(cnt[i]-cnt[yy]<=0)\] 发现\(cnt[i]-cnt[yy] <= 0\)只能有两种取值 于是直接堆优化即可 \( ...

  7. Codeforces Round #599 (Div. 2) B1. Character Swap (Easy Version) 水题

    B1. Character Swap (Easy Version) This problem is different from the hard version. In this version U ...

  8. Python连载44-XML其他注意点

    一.XML文件注意点 1.内容中不能出现尖括号 例如:下面是不合法的 <grade>成绩<90</grade> 解决方案:使用实体引用<EntityReferenc ...

  9. PHP中查询指定时间范围内的所有日期,月份,季度,年份

    /** * 查询指定时间范围内的所有日期,月份,季度,年份 * * @param $startDate 指定开始时间,Y-m-d格式 * @param $endDate 指定结束时间,Y-m-d格式 ...

  10. Element类

    ElementTree API主要包含在Element类中,ElementTree API就是XML树相关的函数 追加子节点有两种方式,一种是使用append(),另一种是使用SubElement() ...