Java学习笔记7——import、静态初始化块、构造方法、继承、方法的重写、Object类的用法、equals方法、super关键字、继承树

1.导入类import

如果我们要使用其他包的类,需要使用import导入,从而可以在本类中直接通过类名来调用,否则就需要书写类的完整包名和类名。import后,便于编写代码,提高可维护性。

注意要点:

1. Java会默认导入java.lang包下所有的类,因此这些类我们可以直接使用。
2. 如果导入两个同名的类,只能用包名+类名来显示调用相关类。

2.静态初始化块

构造方法用于对象的初始化!静态初始化块,用于类的初始化操作!在静态初始化块中不能直接访问非static成员。

注意事项:
  静态初始化块执行顺序:

1. 上溯到Object类,先执行Object的静态初始化块,再向下执行子类的静态初始化块,直到我们的类的静态初始化块为止。
 2. 构造方法执行顺序和上面顺序一样!!

3.构造方法

java的构造方法中this()或者super()要放在第一行

java的构造方法中如果自己显性的调用super()的时候一定要放在第一行,如不是的话就会报错。

为什么一定要在第一行?

super()在第一行的原因就是: 子类有可能访问了父类对象, 比如在构造函数中使用父类对象的成员函数和变量, 在成员初始化使用了父类, 在代码块中使用了父类等, 所以为保证在子类可以访问父类对象之前要完成对父类对象的初始化

this()在第一行的原因就是: 为保证父类对象初始化的唯一性. 我们假设一种情况, 类B是类A的子类, 如果this()可以在构造函数的任意行使用, 那么会出现什么情况呢? 首先程序运行到构造函数B()的第一行, 发现没有调用this()和super(), 就自动在第一行补齐了super() , 完成了对父类对象的初始化, 然后返回子类的构造函数继续执行, 当运行到构造函数B()的"this(2) ;"时, 调用B类对象的B(int) 构造函数, 在B(int)中, 还会对父类对象再次初始化! 这就造成了对资源的浪费, 当然也有可能造成某些意想不到的结果, 不管怎样, 总之是不合理的, 所以this() 不能出现在除第一行以外的其他行!

4.继承和instanceof运算符

public class TestExtends {
	public static void main(String[] args) {
		Student stu = new Student();
		stu.name = "hp";
		stu.height = 173;
		stu.rest();
		
		Student stu2 = new Student("hp",6,"计算机");
		
		System.out.println(stu2 instanceof Student);
		System.out.println(stu2 instanceof Person);
		System.out.println(stu2 instanceof Object);
		System.out.println(new Person() instanceof Student);
	}
}

class Person {
	String name;
	int height;
	
	public void rest() {
		System.out.println("休息");
	}
}

class Student extends Person {
	String major;
	
	public void study() {
		System.out.println("学习两小时");
	}
	public Student(String name,int height,String major) {
		this.name = name;
		this.height = height;
		this.major = major;
	}
	public Student() {
        
	}
}

1.父类也称作超类、基类等。

2.Java中只有单继承,没有像C++那样的多继承。多继承会引起混乱,使得继承链过于复杂,系统难于维护。

3.Java中类没有多继承,接口有多继承。

4.子类继承父类,可以得到父类的全部属性和方法 (除了父类的构造方法),但不见得可以直接访问(比如,父类私有的属性和方法)。

5.如果定义一个类时,没有调用extends,则它的父类是:java.lang.Object。

instanceof是二元运算符,左边是对象,右边是类;当对象是右面类或子类所创建对象时,返回true;否则,返回false。

5.方法的重写override(覆盖)

子类通过重写父类的方法,可以用自身的行为替换父类的行为方法的重写是实现多态的必要条件

方法的重写需要符合下面的三个要点:

​ 1.“==”: 方法名、形参列表相同

​ 2.“≤”:返回值类型和声明异常类型,子类小于等于父类。

​ 3.“≥”: 访问权限,子类大于等于父类。

public class TestOverride {
	public static void main(String[] args) {
		Horse h = new Horse();
		h.run();
	}
}

class Vehicle{
	public void run() {
		System.out.println("跑");
	}
	public void stop() {
		System.out.println("停止");
	}
	public Person whoisps() {
        return new Person();
	}
}
class Horse extends Vehicle{
	public void run() {
		System.out.println("飞起来");
	}
	
	public Student whoisps() {//返回值类型小于等于父类的类型
        return new Student();
	}
}

子类如果没有重写父类的方法,子类对象在调用父类的该方法时,考虑到对象是个引用类型,此时调用的子类方法会指向父类对应的方法,如果重写了父类的方法,那么指向的是子类重写的方法。

6.Object类的用法

Object类是所有Java类的根基类,也就意味着所有的Java对象都拥有Object类的属性和方法。如果在类的声明中未使用extends关键字指明其父类,则默认继承Object类

Object类中定义有public String toString()方法,其返回值是 String 类型。Object类中toString方法的源码为:

public String toString() {
    return getClass().getName() + "@" + Integer.toHexString(hashCode());
}

根据如上源码得知,默认会返回“类名+@+16进制的hashcode”。在打印输出或者用字符串连接对象时,会自动调用该对象的toString()方法。

toString方法也可以重写

public String toString() {
	return name+",年龄:"+age;
}

7.==和equals方法

“==”代表比较双方是否相同。如果是基本类型则表示相等,如果是引用类型则表示地址相等即是同一个对象

​ Object类中定义有:public boolean equals(Object obj)方法,提供定义“对象内容相等”的逻辑。比如,我们在公安系统中认为id相同的人就是同一个人、学籍系统中认为学号相同的人就是同一个人。

​ Object 的 equals 方法默认就是比较两个对象的hashcode,是同一个对象的引用时返回 true 否则返回 false。但是,我们可以根据我们自己的要求重写equals方法。

JDK提供的一些类,如String、Date、包装类等,重写了Object的equals方法,调用这些类的equals方法, x.equals (y) ,当x和y所引用的对象是同一类对象且属性内容相等时(并不一定是相同对象),返回 true 否则返回 false。

8.super关键字

super是直接父类对象的引用。可以通过super来访问父类中被子类覆盖的方法或属性。

使用super调用普通方法,语句没有位置限制,可以在子类中随便调用。

若是构造方法的第一行代码没有显式的调用super(…)或者this(…);那么Java默认都会调用super(),含义是调用父类的无参数构造方法。这里的super()可以省略。

super和this的异同:

  • super(参数):调用基类中的某一个构造函数(应该为构造函数中的第一条语句)
  • this(参数):调用本类中另一种形成的构造函数(应该为构造函数中的第一条语句)
  • super: 它引用当前对象的直接父类中的成员(用来访问直接父类中被隐藏的父类中成员数据或函数,基类与派生类中有相同成员定义时如:super.变量名 super.成员函数据名(实参)
  • this:它代表当前对象名(在程序中易产生二义性之处,应使用this来指明当前对象;如果函数的形参与类中的成员数据同名,这时需用this来指明成员变量名)
  • 调用super()必须写在子类构造方法的第一行,否则编译不通过。每个子类构造方法的第一条语句,都是隐含地调用super(),如果父类没有这种形式的构造函数,那么在编译的时候就会报错。
  • super()和this()类似,区别是,super()从子类中调用父类的构造方法,this()在同一类内调用其它方法
  • super()和this()均需放在构造方法内第一行。
  • 尽管可以用this调用一个构造器,但却不能调用两个
  • this和super不能同时出现在一个构造函数里面,因为this必然会调用其它的构造函数,其它的构造函数必然也会有super语句的存在,所以在同一个构造函数里面有相同的语句,就失去了语句的意义,编译器也不会通过。
  • this()和super()都指的是对象,所以,均不可以在static环境中使用。包括:static变量,static方法,static语句块。
  • 从本质上讲,this是一个指向本对象的指针, 然而super是一个Java关键字

9.继承树追溯

·属性/方法查找顺序:(比如:查找变量h)

​ 1. 查找当前类中有没有属性h

​ 2. 依次上溯每个父类,查看每个父类中是否有h,直到Object

​ 3. 如果没找到,则出现编译错误。

​ 4. 上面步骤,只要找到h变量,则这个过程终止。

·构造方法调用顺序:

​ 构造方法第一句总是:super(…)来调用父类对应的构造方法。所以,流程就是:先向上追溯到Object,然后再依次向下执行类的初始化块和构造方法,直到当前子类为止。从上到下继承父类的所有方法。

​ 注:静态初始化块调用顺序,与构造方法调用顺序一样,不再重复。

猜你喜欢

转载自blog.csdn.net/weixin_44139445/article/details/87473147