1.Static关键字
1.当static修饰成员变量时
如果一个成员变量使用了static关键字,那么这个变量不再属于对象自己,而是属于所在的类。多个对象共享同一份数据。
public class Student {
private int id; // 学号
private String name; // 姓名
private int age; // 年龄
static String room; // 所在教室
private static int idCounter = 0; // 学号计数器,每当new了一个新对象的时候,计数器++
public Student() {
this.id = ++idCounter;
}
public Student(String name, int age) {
this.name = name;
this.age = age;
this.id = ++idCounter;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
public class Demo01StaticField {
public static void main(String[] args) {
Student two = new Student("黄蓉", 16);
two.room = "101教室";
System.out.println("姓名:" + two.getName()
+ ",年龄:" + two.getAge() + ",教室:" + two.room
+ ",学号:" + two.getId());
Student one = new Student("郭靖", 19);
System.out.println("姓名:" + one.getName()
+ ",年龄:" + one.getAge() + ",教室:" + one.room
+ ",学号:" + one.getId());
}
}
2.当static修饰方法时
一旦使用static修饰成员方法,那么这就成为了静态方法。静态方法不属于对象,而是属于类的。
如果没有static关键字,那么必须首先创建对象,然后通过对象才能使用它。(MyClass obj = new MyClass(); obj.method();)
如果有了static关键字,那么不需要创建对象,直接就能通过类名称来使用它。(MyClass.methodStatic();)
无论是成员变量,还是成员方法。如果有了static,都推荐使用类名称进行调用。这样可读性行好,别人一看就知道它是静态的。
静态变量:类名称.静态变量
静态方法:类名称.静态方法()
注意事项:
1. 静态不能直接访问非静态。
原因:因为在内存当中是【先】有的静态内容,【后】有的非静态内容。
“先人不知道后人,但是后人知道先人。”
2. 静态方法当中不能用this。
原因:this代表当前对象,通过谁调用的方法,谁就是当前对象。
public class MyClass {
int num; // 成员变量
static int numStatic; // 静态变量
// 成员方法
public void method() {
System.out.println("这是一个成员方法。");
// 成员方法可以访问成员变量
System.out.println(num);
// 成员方法可以访问静态变量
System.out.println(numStatic);
}
// 静态方法
public static void methodStatic() {
System.out.println("这是一个静态方法。");
// 静态方法可以访问静态变量
System.out.println(numStatic);
// 静态不能直接访问非静态【重点】
// System.out.println(num); // 错误写法!
// 静态方法中不能使用this关键字。
// System.out.println(this); // 错误写法!
}
}
public class Demo02StaticMethod {
public static void main(String[] args) {
MyClass obj = new MyClass();
obj.method();// 首先创建对象,然后才能使用没有static关键字的内容
// 对于静态方法来说,可以通过对象名进行调用,也可以直接通过类名称来调用。
obj.methodStatic(); // 正确,不推荐
MyClass.methodStatic(); // 正确,推荐,你一这样写所有人都知道methodStatic是一个静态方法
// 对于本类当中的静态方法,可以省略类名称
myMethod();//他就等价于Demo02StaticMethod.myMethod();
}
public static void myMethod() {
System.out.println("自己的方法!");
}
}
3.静态代码块的优先执行
静态内容总是优先于非静态,所以静态代码块比构造方法先执行。
静态代码块的典型用途:
当第一次用到本类时,本类中的静态代码块执行唯一的一次,一次性地对静态成员变量进行赋值。
public class Person {
static {
System.out.println("静态代码块执行!");
}
public Person() {
System.out.println("构造方法执行!");
}
}
public class Demo04Static {
public static void main(String[] args) {
Person one = new Person();
Person two = new Person();
//每次new一个对象就肯定立刻生成并执行该对象的构造方法,但静态方法比构造方法还要优先执行
}
}
2.super关键字
1.子类一定会调用父类的无参数构造方法(如果父类只有有参数的构造方法子类就不调用了)
子类构造方法当中有一个默认隐含的“super()”调用,所以一定是先调用的父类构造,后执行的子类构造。
而且super的父类构造调用,必须是子类构造方法的第一个语句。
public class Fu {
public Fu() {
System.out.println("父类的无参数构造方法!");
}
}
public class Zi extends Fu {
public Zi() {
super(); // 不管你写不写这行代码,系统都默认给你带上了
System.out.println("子类构造方法!");
}
}
public class Demo01Constructor {
public static void main(String[] args) {
Zi zi = new Zi();
}
}
输出:
父类的无参数构造方法!
子类构造方法!
2.子类必须调用父类构造方法,不写则赠送super();写了则用写的指定的super调用,super只能有一个,还必须是第一个。
public class Fu {
public Fu(int num) {
System.out.println("父类的有参数构造方法!");
}
}
public class Zi extends Fu {
public Zi() {
super(); // 不管你写不写这行代码,系统都默认给你带上了
System.out.println("子类构造方法!");
}
}
public class Demo01Constructor {
public static void main(String[] args) {
Zi zi = new Zi();
}
}
会报错,当你在子类不写super()时系统默认给你赠送了一个super()来继承父类**无参数的**构造方法,而父类只有一个**有参数**的构造方法,而你子类没有写有参数的构造方法。你改错的方法就是自己写一个super(10)来抵消掉系统给你的super()
3.super关键字的用法有三种:
- 在子类的成员方法中,访问父类的成员变量。
- 在子类的成员方法中,访问父类的成员方法。
- 在子类的构造方法中,访问父类的构造方法。
public class Zi extends Fu {
@Override
public void method() {
super.method(); // 在子类的成员方法中,访问父类的成员方法。
System.out.println("子类方法");
}
3.this关键字
this关键字用来访问本类内容。用法也有三种:
1. 在本类的成员方法中,访问本类的成员变量。
2. 在本类的成员方法中,访问本类的另一个成员方法。
3. 在本类的构造方法中,访问本类的另一个构造方法。
在第三种用法当中要注意:
A. this(...)调用也必须是构造方法的第一个语句,唯一一个。
B. super和this两种构造调用,不能同时使用。
- 在本类的成员方法中,访问本类的成员变量。
局部变量: 直接写成员变量名
本类的成员变量: this.成员变量名
父类的成员变量: super.成员变量名
public class Zi extends Fu {
int num = 20;
public void method() {
int num = 30;
System.out.println(num); // 30,局部变量
System.out.println(this.num); // 20,本类的成员变量
System.out.println(super.num); // 10,父类的成员变量
}
}
- 在本类的成员方法中,访问本类的另一个成员方法。
public class Zi extends Fu {
public void methodA() {
System.out.println("AAA");
}
public void methodB() {
this.methodA(); //在本类的成员方法中,访问本类的另一个成员方法。
System.out.println("BBB");
}
}
- 在本类的构造方法中,访问本类的另一个构造方法。
在第三种用法当中要注意:
A. this(…)调用也必须是构造方法的第一个语句,唯一一个。
B. super和this两种构造调用,不能同时使用。
public class Zi extends Fu {
public Zi() {
this(123); // 本类的无参构造,调用本类的有参构造
}
public Zi(int n) {
this(1, 2); //在本类的构造方法中,访问本类的另一个构造方法。
}
public Zi(int n, int m) {
}
}
public class Zi extends Fu {
public Zi() {
super(); // 错误,super和this不能同时使用,因为 this(...)调用必须是构造方法的第一个语句,super(...)调用也必须是构造方法的第一个语句,到底谁是第一个?
this(123); // 本类的无参构造,调用本类的有参构造
}
public Zi(int n) {
this(1, 2);
}
public Zi(int n, int m) {
}
}