• 07. Java基础之final


    一. final关键字的基本用法

          final可以用来修饰类、方法、变量(包含成员变量和局部变量)

    1. final修饰类

          当用final修饰一个类时,表明这个类不能被继承。final类中的成员变量可以根据需要设为final,但是final类中的所有成员方法都会被隐式地指定为final方法,因为无法覆盖它们。再final类中可以给方法添加final修饰词,但这不会添加任何意义。

    public final class Base {

    }

    note: 在使用final修饰类的时候,要注意谨慎选择,除非这个类真的在以后不会用来继承或者出于安全的考虑,尽量不要将类设计为final类。

    2. final方法

         类中的所有的private方法都隐式的指定为final的。由于无法取用private方法,所以也就无法覆盖它。可以对private方法添加final修饰词,但这并不能给该方法增加任何额外的意义。(note:private的方法或变量不会被继承,也无从override之说,只是多了一个同名的方法而已)

    3. final变量

        对于基本类型,final使数值恒定不变;而对于对象引用,final使引用恒定不变。一旦引用被初始化指向一个对象,就无法再把它改为指向另外一个对象。

    二. 易错点

    1. final变量和普通变量的区别?

     1 public class Test {
     2     public static void main(String[] args)  {
     3         String a = "hello2"; 
     4         final String b = "hello";
     5         String d = "hello";
     6         String c = b + 2; 
     7         String e = d + 2;
     8         System.out.println((a == c));
     9         System.out.println((a == e));
    10     }
    11 }
    View Code
    1 true
    2 false
    View Code

           大家可以先想一下这道题的输出结果。为什么第一个比较结果为true,而第二个比较结果为fasle。这里面就是final变量和普通变量的区别了,当final变量是基本数据类型以及String类型时,如果在编译期间能知道它的确切值,则编译器会把它当做编译期常量使用。也就是说在用到该final变量的地方,相当于直接访问的这个常量,不需要在运行时确定。这种和C语言中的宏替换有点像。因此在上面的一段代码中,由于变量b被final修饰,因此会被当做编译器常量,所以在使用到b的地方会直接将变量b 替换为它的  值。而对于变量d的访问却需要在运行时通过链接来进行。想必其中的区别大家应该明白了,不过要注意,只有在编译期间能确切知道final变量值的情况下,编译器才会进行这样的优化,比如下面的这段代码就不会进行优化:

     1 public class Test {
     2     public static void main(String[] args)  {
     3         String a = "hello2"; 
     4         final String b = getHello();
     5         String c = b + 2; 
     6         System.out.println((a == c));
     7  
     8     }
     9      
    10     public static String getHello() {
    11         return "hello";
    12     }
    13 }
    View Code

    这段代码的输出结果为false。

    2. 被final修饰的引用变量指向的对象内容可变吗?

    在上面提到被final修饰的引用变量一旦初始化赋值之后就不能再指向其他的对象,那么该引用变量指向的对象的内容可变吗?看下面这个例子:

     1 public class Test {
     2     public static void main(String[] args)  {
     3         final MyClass myClass = new MyClass();
     4         System.out.println(++myClass.i);
     5  
     6     }
     7 }
     8  
     9 class MyClass {
    10     public int i = 0;
    11 }
    View Code

    1

    这段代码可以顺利编译通过并且有输出结果,输出结果为1。这说明引用变量被final修饰之后,虽然不能再指向其他对象,但是它指向的对象的内容是可变的。

    3. final和static

    很多时候会容易把static和final关键字混淆,static作用于成员变量用来表示只保存一份副本,而final的作用是用来保证变量不可变。看下面这个例子:

     1 public class Test {
     2     public static void main(String[] args)  {
     3         MyClass myClass1 = new MyClass();
     4         MyClass myClass2 = new MyClass();
     5         System.out.println(myClass1.i);
     6         System.out.println(myClass1.j);
     7         System.out.println(myClass2.i);
     8         System.out.println(myClass2.j);
     9  
    10     }
    11 }
    12  
    13 class MyClass {
    14     public final double i = Math.random();
    15     public static double j = Math.random();
    16 }
    View Code

    运行这段代码就会发现,每次打印的两个j值都是一样的,而i的值却是不同的。从这里就可以知道final和static变量的区别了。

    4. 匿名内部类中使用的外部局部变量为什么只能是final变量?

    想必这个问题也曾经困扰过很多人,在讨论这个问题之前,先看下面这段代码:

     1 public class Test {
     2     public static void main(String[] args)  {
     3          
     4     }
     5      
     6     public void test(final int b) {
     7         final int a = 10;
     8         new Thread(){
     9             public void run() {
    10                 System.out.println(a);
    11                 System.out.println(b);
    12             };
    13         }.start();
    14     }
    15 }
    View Code

    这段代码会被编译成两个class文件:Test.class和Test1.classOutter1.class。默认情况下,编译器会为匿名内部类和局部内部类起名为Outterx.class(x为正整数)。

      

      根据上图可知,test方法中的匿名内部类的名字被起为 Test$1。

      上段代码中,如果把变量a和b前面的任一个final去掉,这段代码都编译不过。我们先考虑这样一个问题:

      当test方法执行完毕之后,变量a的生命周期就结束了,而此时Thread对象的生命周期很可能还没有结束,那么在Thread的run方法中继续访问变量a就变成不可能了,但是又要实现这样的效果,怎么办呢?Java采用了 复制  的手段来解决这个问题。将这段代码的字节码反编译可以得到下面的内容:

      我们看到在run方法中有一条指令:

    bipush 10

      这条指令表示将操作数10压栈,表示使用的是一个本地局部变量。这个过程是在编译期间由编译器默认进行,如果这个变量的值在编译期间可以确定,则编译器默认会在匿名内部类(局部内部类)的常量池中添加一个内容相等的字面量或直接将相应的字节码嵌入到执行字节码中。这样一来,匿名内部类使用的变量是另一个局部变量,只不过值和方法中局部变量的值相等,因此和方法中的局部变量完全独立开。

      下面再看一个例子:

     1 public class Test {
     2     public static void main(String[] args)  {
     3          
     4     }
     5      
     6     public void test(final int a) {
     7         new Thread(){
     8             public void run() {
     9                 System.out.println(a);
    10             };
    11         }.start();
    12     }
    13 }
    View Code

    反编译得到:

      我们看到匿名内部类Test$1的构造器含有两个参数,一个是指向外部类对象的引用,一个是int型变量,很显然,这里是将变量test方法中的形参a以参数的形式传进来对匿名内部类中的拷贝(变量a的拷贝)进行赋值初始化。

      也就说如果局部变量的值在编译期间就可以确定,则直接在匿名内部里面创建一个拷贝。如果局部变量的值无法在编译期间确定,则通过构造器传参的方式来对拷贝进行初始化赋值。

      从上面可以看出,在run方法中访问的变量a根本就不是test方法中的局部变量a。这样一来就解决了前面所说的 生命周期不一致的问题。但是新的问题又来了,既然在run方法中访问的变量a和test方法中的变量a不是同一个变量,当在run方法中改变变量a的值的话,会出现什么情况?

      对,会造成数据不一致性,这样就达不到原本的意图和要求。为了解决这个问题,java编译器就限定必须将变量a限制为final变量,不允许对变量a进行更改(对于引用类型的变量,是不允许指向新的对象),这样数据不一致性的问题就得以解决了。

      到这里,想必大家应该清楚为何 方法中的局部变量和形参都必须用final进行限定了。

    参考文献:

    http://www.cnblogs.com/dolphin0520/p/3736238.html

  • 相关阅读:
    CF786E ALT
    CF704D Captain America
    [NOI2016]循环之美
    「PKUWC2018」猎人杀
    [HNOI2019]JOJO
    博客已转移
    $20200203$的数学作业
    20200202的数学作业
    NOIp 2016 选课 (DP)
    Luogu P2574 XOR的艺术 (线段树)
  • 原文地址:https://www.cnblogs.com/Hermioner/p/9571715.html
Copyright © 2020-2023  润新知