静态变量与实例变量的定义与区别:
静态变量:指的是被static修饰的类的变量,被所有类实例对象所共享,在内存中只有一个副本,当且仅当在类初次加载时会被初始化。
实例变量:定义在类中但在任何方法之外的变量,未使用static关键字修饰
静态变量与实例变量的主要区别:
(1)静态变量被所有类实例对象所共享,在内存中只有一个副本,当且仅当在类初次加载时会被初始化。
(2)实例变量是实例对象所拥有的,在创建对象的时候被初始化,存在多个副本,各个实例对象拥有的副本互不影响
静态变量需要关注的点:
1. static关键字在java中不允许用来修饰局部变量
2. this可以访问静态变量,但往往不建议访问,在Java语法上是没有错的,但由于静态变量是属于整个类的,是所有类实例共享的,用类名去访问静态变量更加直观的表示出静态变量是整个类的,而this指的是类的一个实例变量,所以不建议使用this访问静态变量。
3. static不会改变类成员的访问权限,在Java中能够影响到访问权限的只有private、public、protected(包括包访问权限)这几个关键字。
静态方法与实例方法的定义与区别:
静态方法(或称为类方法):指被static修饰的成员方法,不需要通过它所属的类的任何实例就可以被调用,因此在静态方法中不能使用this关键字,也不能直接访问所属类的实例变量和实例方法,但是可以直接访问所属类的静态变量和静态方法。
实例方法(或称为对象方法):是属于类的某个对象的方法,可以调用静态方法(类方法),未使用static修饰。
静态方法与实例方法的区别:
1. 静态方法不需要通过它所属的类的任何实例就可以被调用,因此在静态方法中不能使用 this 关键字,也不能直接访问所属类的实例变量和实例方法,但是可以直接访问所属类的静态变量和静态方法。另外,和 this 关键字一样,super 关键字也与类的特定实例相关,所以在静态方法中也不能使用 super 关键字。
2. 在实例方法中可以直接访问所属类的静态变量、静态方法、实例变量和实例方法。
简述一下类的加载机制
类并不是在一开始就全部加载好,而是在需要时才会去加载(提升速度)
以下几种常见的情况会加载类:
- 访问类的静态变量,或者为静态变量赋值
- new 创建类的实例(隐式加载)
- 调用类的静态方法
- 子类初始化时
所有被标记为静态的内容,会在类刚加载的时候就分配,而不是在对象创建的时候分配,所以说静态内容一定会在第一个对象初始化之前完成加载。
public class Student {
static int a = test(); //直接调用静态方法,只能调用静态方法
Student(){
System.out.println("构造类对象");
}
static int test(){ //静态方法刚加载时就有了
System.out.println("初始化变量a");
return 1;
}
}
思考:下面这种情况下,程序能正常运行吗?如果能,会输出什么内容?
public class Student {
static int a = test();
static int test(){
return a;
}
public static void main(String[] args) {
System.out.println(Student.a);
}
}
输出结果为0。
定义和赋值是两个阶段,在定义时会使用默认值(上面讲的,类的成员变量会有默认值)定义出来之后,如果发现有赋值语句,再进行赋值,而这时,调用了静态方法,所以说会先去加载静态方法,静态方法调用时拿到a,而a这时仅仅是刚定义,所以说还是初始值,最后得到0
代码块
1.普通代码块:
类中方法的方法体
2.构造代码块:
构造块会在创建对象时被调用,每次创建时都会被调用,优先于类构造函数执行。
3.静态代码块:
用static{}包裹起来的代码片段,只会执行一次。静态代码块优先于构造块执行。
4.同步代码块:
使用synchronized(){}包裹起来的代码块,在多线程环境下,对共享数据的读写操作是需要互斥进行的,否则会导致数据的不一致性。同步代码块需要写在方法中。
静态代码块和构造代码块的异同点
相同点:都是JVM加载类后且在构造函数执行之前执行,在类中可定义多个,一般在代码块中对一些static变量进行赋值。
不同点:静态代码块在非静态代码块之前执行。静态代码块只在第一次new时执行一次,之后不在执行。而非静态代码块每new一次就执行一次。
public class Student {
{
System.out.println("我是代码块");
}
Student(){
System.out.println("我是构造方法");
}
}
静态代码块和上面的静态方法和静态变量一样,在类刚加载时就会调用;
public class Student {
static int a;
static {
a = 10;
}
public static void main(String[] args) {
System.out.println(Student.a);
}
}