final、static、this、super关键字总结

final关键字

final关键字,用来修饰类、方法和变量,具有以下特点:

  • final修饰的类不能被继承,final类中的所有成员方法都会被隐式的指定为final方法
  • final修饰的方法不能被重写
  • final修饰的变量是常量,如果是基本数据类型的变量,则其数值一旦在初始化之后就不能更改;如果是引用类型的变量,则在对其初始化之后便不能让其指向另一个对象

说明:使用final方法的原因有两个。

  1. 把方法锁定,以防任何继承类修改它的含义
  2. 效率。

早期Java实现版本中,会final方法转为内嵌调用,但如果方法过于庞大,可能看不到内嵌调用带来的性能提升。(现在的Java版本已经不使用final方法进行优化了)。类中所有的private方法都隐式的指定为final。

static关键字

static关键字主要有以下四种使用场景

  1. 修饰成员变量和成员方法:被static修饰的成员属于类,不属于单个这个类的某个对象,被类中所有对象共享,可以并且建议通过类名调用。被static声明的成员变量属于静态成员变量,静态变量存放在Java内存区域的方法区。调用格式:类名.静态变量名 类名.静态方法ming()
  2. 静态代码块:静态代码块定义在类中方法外,静态代码块在非静态代码块之前执行(静态代码块->非静态代码块->构造方法),该类不管创建多少对象,静态代码块只执行一次
  3. 静态内部类(static修饰类的话只能修饰内部类):静态内部类与非静态内部类之间存在一个最大的区别:非静态内部类在编译完成之后会隐含的保存一个引用,该引用个指向创建它的外围类,但是静态内部类却没有,没有这个引用就意味着:1.它的创建是不需要依赖外围类的创建 2.它不能使用任何外围类的非static成员变量和方法
  4. 静态导包(用来导入类中的静态资源,1.5之后的新特性):格式为:import static这两个关键字连用可以指定导入某个类中的指定静态资源,并且不需要使用类名调用类中静态成员,可以直接使用类中静态成员和成员方法。

this关键字

this关键字用于引用类的当前实例。

class Manager{
    Employees[] employees;

    void manageEmployees(){
        int totalEmp = this.employees.length;
        System.out.println("Total employess: " + totalEmp);
        this.report();
    }
    void report(){

    }
}
  • this.employees.length:访问类Manager的当前实例的变量
  • this.report():调用类Manager的当前实例的方法

super关键字

super关键字用于从子类访问父类的变量和方法

public class Super{
    protected int number;
    protected showNumber(){
        System.out.println("number = " + number);
    }
}
//Sub 类访问父类成员变量 number 并调用其父类 Super 的 showNumber() 方法。
public class Sub extends Super{
    void bar(){
        super.number = 10;
        super.showNumber();
    }
}

使用this和super要注意的问题

  • 在构造器中使用super()调用父类中的其他构造方法时,该语句必须处于构造器的首行,否则编译器会报错。另外,this调用本类中的其他构造方法时,也要放在首行。
  • this、super不能在static方法中

被static修饰的成员属于类,不属于单个这个类的某个对象,被类中所有对象共享。而this代表对本类对象的引用,指向本类对象;super代表对父类对象的引用。所以,this和super是属于对象范畴的东西,而静态方法属于类范畴的东西

static关键字详解

static关键字主要用于以下四种场景

  1. 修饰成员变量和成员方法
  2. 静态代码块
  3. 修饰类(只能修饰内部类)
  4. 静态导包(用来导入类中的静态资源,1.5之后的新特性)

修饰成员变量和成员方法
被static修饰的成员属于类,不属于单个这个类的某个对象,被类中所有的对象共享,可以并且建议通过类名调用。被static声明的成员变量属于静态成员变量,静态变量存放在Java内存区域的方法区

方法区与Java堆一样,是各个线程共享的内存区域,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。虽然Java虚拟机规范把方法区面熟为堆的一个逻辑部分,但是它却有一个别名叫做Non-Heap(非堆),目的应该是与Java堆区分开来.

HotSpot虚拟机中方法区也常被称为”永久代”.本质上两者并不等价.仅仅因为HotSpot虚拟机设计团队用永久代来实现方法区而已,这样HotSpot虚拟机的垃圾收集器就可以像管理Java堆一样管理这部分内存了,但这不是一个好主意,更容易出现内存溢出问题

调用格式

  • 类名.静态变量名
  • 类名.静态方法名()

如果变量或者方法被private则代表该属性或该方法只能在类的内部被访问不能在类的外部被访问

public class StaticBean{
    String name;
    static int age;
    public StaticBean(String name){
        this.name = name;
    }
    //静态方法
    static void sayHello(){
        System.out.println("hello i am java");
    }

    @Override
    public String toString(){
        return "StaticBean{" + "name="+name + ",age="+age+"}";
    }
}

public class staticDemo{
    public static void main(String[]args){
        StaticBean staticBean = new StaticBean("1");
        StaticBean staticBean2 = new StaticBean("2");
        StaticBean staticBean3 = new StaticBean("3");
        StaticBean staticBean4 = new StaticBean("4");
        staticBean.age = 33;
        System.out.println(staticBean + " " + staticBean2 + " " + staticBean3 + " " + staticBean4);
        //StaticBean{name=1,age=33} StaticBean{name=2,age=33} StaticBean{name=3,age=33} StaticBean{name=4,age=33}
        StaticBean.sayHello();//Hello i am java
    }
}

静态方法和非静态方法
静态方法属于类本身,非静态方法属于从该类生成的对象。如果方法执行的操作不依赖其类的各个变量和方法,请将其设置为静态(这将使程序的占用空间更小),否则。它应该是非静态的。

class Foo{
    int i;
    public Foo(int i){
        this.i = i;
    }

    public static String method1(){
        return "An example string that doesn't depend on i (an instance variable)";
    }

    public int method2(){
        return this.i +1;
    }
}

可以Foo.method1()调用静态方法。但如果用这种方法调用method2()将失败。可这样调用Foo f = new Foo(); f.method2();

  • 在外部调用静态方法时,可以使用类名.方法名的方法,也可以使用对象名.方法名的方式。而实例方法只能用后面的方式。调用静态方法可以无需创建对象
  • 静态方法在访问本类的成员时,只允许访问静态成员(即静态成员变量和静态方法),不允许访问实例成员变量和实例方法,实例方法则无此限制。

静态代码块

静态代码块定义在类中方法外,静态代码块在非静态代码之前执行(静态代码块 —> 非静态代码块 —> 构造方法),该类不管创建多少对象,静态代码块只执行一次

静态代码块格式

static{
    xxx;
}

一个类中的静态代码块可以有多个,位置可以随便放,它不在任何的方法体内,JVM加载类时会执行这些静态代码块,如果静态代码块有多个,JVM会按照他们在类中出现的先后顺序依次执行它们.每个静态代码块只会被执行一次

public class Demo{
    static{
        i = 3;
        System.out.println(i);  //此处报错,不能在字段定义前进行访问
    }
    private static int i;
}

静态代码块对于定义在它之后的静态变量,可以赋值,但是不能访问.

静态代码块和非静态代码块(构造代码块)
相同点:都是在jvm加载类时且在构造方法执行之前执行,在类中都可以定义多个,定义多个时按定义顺序执行,一般在代码块中对一些static变量进行赋值。

不同点:静态代码块在非静态代码之前执行(静态代码块 —> 非静态代码块 —> 构造方法)静态代码块只在第一次new执行一次,之后不再执行,而非静态代码块在每new一次就执行一次。非静态代码块可在普通方法中定义(不过作用不大);而静态代码不行。

修正:静态代码块可能在第一次new对象的时候执行,但不一定只在第一次new的时候执行,比如通过Class.forName(“ClassDemo”)创建class对象的时候也会执行,即new和Class.forName()都回执行静态代码块

一般情况下,如果有些代码比如一些项目最常用的变量或对象必须在项目骑动的时候就执行,这个时候就需要静态代码块,这种代码是主动执行的。如果想要设计不需要创建对象就可以调用类中的方法,例如,Arrays类,Character类,String类等,就需要使用静态方法,两者的区别是静态代码块是自动执行的,而静态方法是被调用才执行的。

public class Test{
    public Test(){
        System.out.println("默认构造方法!");    //3
    }

    //非静态代码块
    {
        System.out.println("非静态代码块!");    //2
    }

    static {
        System.out.println("静态代码块!"); //1
    }

    private static void test(){
        System.out.println("静态方法中的内容!");    //4
        {
            System.out.println("静态方法中的代码块");//5
        }
    }

    public static void main(String[]args){
        Test test = new Test();
        Test.test();    //静态代码块 静态方法中的内容
    }
}

输出结果:1-2-3-4-5
当只执行Test test = new Test()时,输出 1-2-3

非静态代码块与构造函数的区别是:非静态代码块是给所有对象进行统一初始化,而构造函数是给对应的对象初始化,因为构造函数是可以有多个的,运行哪个构造函数就会建立什么样的对象,但无论建立哪个对象,都会先执行相同的构造代码块,也就是说,构造代码块中定义的是不同对象共性的初始化内容。

静态内部类

静态内部类与非静态内部类之间存在一个最大的区别,我们知道非静态内部类在编译完成之后会隐含的保存一个引用,该引用是指向创建它的外围类,但是静态内部类没有,没有这个引用意味着:

  1. 它的创建不需要依赖外围类的创建
  2. 它不能使用任何外围类的非static成员变量和方法

静态导包

格式:import static

这两个关键字连用可以指定导入某个类中的指定静态资源,并且不需要使用类名调用类中静态成员,可以直接使用该类中静态成员变量和成员方法

//将Math中的所有静态资源导入,这时可以直接使用里面的静态方法,而不用通过类名进行调用
//如果只想导入单一某个静态方法,只需将*换成对应的方法名即可
import static java.lang.Math.*; //换成import static java.lang.Math.max;
public class Demo{
    public static void main(String[]args){
        int max = max(1,2);
        System.out.println(max);
    }
}