美文网首页
final关键字

final关键字

作者: 云烟渐成雨 | 来源:发表于2019-08-01 18:34 被阅读0次

    1.1 final数据

    • 1.一个永不改变的编译时常量
    • 2.一个在运行时被初始化的值,而你不希望它被改变

    在Java中,这类常量必须是基本数据类型,并且以关键字final表示。在这个常量进行定义的时候,必须对其进行赋值

    一个既是static又是final的域只占据一段不能改变的储存空间

    当对对象引用而不是基本数据类型运用final时。对于基本类型,final使数值恒定不变;而用于对象引用,final使引用恒定不变。这一限制同样适用于数组,它也是对象。

    class Value {
        int i;
    
        public Value(int i) {
            this.i = i;
        }
    }
    
    public class FinalData {
        private static Random random = new Random(47);
        private String id;
    
        public FinalData(String id) {
            this.id = id;
        }
    
        private final int valueOne = 9;
        private static final int VALUE_TWO = 29;
        public static final int VALUE_THREE = 39;
        private final int ir = random.nextInt(20);
        static final int INT = random.nextInt(20);
        private Value v1 = new Value(1);
        private final Value v2 = new Value(2);
        private static final Value V3 = new Value(3);
        private final int[] a = {1, 2, 3, 4, 5, 6};
    
        @Override
        public String toString() {
            return "FinalData{" +
                    "id='" + id + '\'' +
                    ", ir=" + ir + ", INT =" + INT +
                    '}';
        }
    
        public static void main(String[] args) {
            FinalData fd1 = new FinalData("fd1");
    //        fd1.valueOne++;   //Error:valueOne为final,值不能改变
            fd1.v2.i++;
            fd1.v1 = new Value(9);
            for (int i = 0; i < fd1.a.length; i++)
                fd1.a[i]++;
    
    //        fd1.v2 = new Value(0);    //Error
    //        fd1.a = new int[3];       //Error
            System.out.println(fd1);
            System.out.println("Creating new FinalData");
            FinalData fd2 = new FinalData("fd2");
            System.out.println(fd1);
            System.out.println(fd2);
        }
    }
    

    运行结果:

    FinalData{id='fd1', ir=15, INT =18}
    Creating new FinalData
    FinalData{id='fd1', ir=15, INT =18}
    FinalData{id='fd2', ir=13, INT =18}
    

    1.我们不能因为数据是final的就认为在编译时可以知道它的值,在运行时使用随机生成的数值来初始化ir和INT就说明了这点。

    2.示例部分展示了将final数值定义为静态和非静态的区别。在fd1和fd2中,ir的值是唯一的(这里的唯一是对于对应的引用),但INT的值是不可以通过创建第二个FinalData对象来改变的,因为它是static的,在装载时已被初始化,而不是每次创建新对象时都会初始化。(静态初始化动作,这段代码仅执行一次)

    1.2 空白final

    Java允许生成“空白final”,空白final指被声明为final但又未给定初值的域。无论在何种情况下,编译器都要确保空白final使用前必须被初始化。

    class Poppet {
        public int i;
    
        Poppet(int ii) {
            i = ii;
        }
    }
    
    public class BlankFinal {
        private final int i = 0;
        private final int j;
        private final Poppet p;
    
        public BlankFinal() {
            j = 1;
            p = new Poppet(1);
            System.out.println("j = " + j + ",p.i =" + p.i);
        }
    
        public BlankFinal(int x) {
            j = x;
            p = new Poppet(x);
            System.out.println("j = " + j + ",p.i =" + p.i);
        }
    
        public static void main(String[] args) {
            new BlankFinal();
            new BlankFinal(2);
        }
    }
    
    

    输出结果:

    j = 1,p.i =1
    j = 2,p.i =2
    

    必须在域的定义外或者每个构造器用表达式对final进行赋值,这正是final域在使用前总是被初始化的原因所在。

    1.2.1 域的定义

    class bike{ 
      static int bikes;
      int gear;
      int cadence;
    
      void create( int newGear, int newCadence ){
        bikes = bikes + 1;
        gear = newGear;
        cadence = newCadence;
      }
      
      int getSpeed(){
        int speed = gear*cadence*5*3.141;
        return speed;
      }
    }
    
    • bikes是一个类变量(静态域)
    • gear 和 cadence 是对象变量(实例变量)(非静态域)

    这里有一点点小矛盾,其实这样照百科这样说,那么bikes、gear和cadence都是类变量,bikes是类变量中的静态变量,而gear和cadence是类变量中的实例变量

    • speed是对象方法的变量(局部变量)。

    java没有出现local variable->局部变量,gobal variable->全局变量,要说的话类变量的作用范围和全局变量一样,只不过不那样叫。中文翻译的问题

    • newGear和newCadence是函数(方法)的参数(参数)。

    1.3 final参数

    Java允许在参数的列表中以声明的方式将参数指名为final。意味着我们无法在方法中更改参数引用所指向的对象。

    class Args{
        public void arg(){
    
        }
    }
    
    public class FinalArguments {
        void with(final Args args){
    //        args = new Args();       Error:args is final
        }
    
        void without(Args args){
            args = new Args();
            args.arg();
        }
    
        void f(final int i){
    //        i++;              Error:can't change,i is final
        }
    
        //final类型的参数只可以读,无法修改参数
        int g(final int i){
            return i + 1;
        }
    
        public static void main(String[] args) {
            FinalArguments finalArguments = new FinalArguments();
            finalArguments.without(null);
            finalArguments.with(null);
            System.out.println(finalArguments.g(1));
        }
    }
    

    方法f()和g()展示了当基本类型的参数被指明为fianl时:可以读参数,无法修改参数。这一特性主要用来向匿名内部类传递数据。

    1.4 final方法

    将方法指定为fianl,可以将方法锁定,以防止任何继承类修改,并且不会被覆盖。

    1.5 final类

    当某个类整体被定义为final时,表明你不希望继承该类,不希望它有子类。

    相关文章

      网友评论

          本文标题:final关键字

          本文链接:https://www.haomeiwen.com/subject/dkwcdctx.html