java方法重写的好处_Java基础:继承

3. 继承

3.1 Java中的继承

math?formula=%5Ccolor%7Bred%7D%7B%E4%BB%80%E4%B9%88%E6%98%AF%E7%BB%A7%E6%89%BF%EF%BC%9F%7D

继承是类与类的一种关系,是一种“is a”的关系。

注意:Java中的继承是单继承,也就是说,一个类只有一个父类。

继承的好处有哪些?

子类拥有父类的所有属性和方法(关键字private修饰的属性和方法除外)

实现代码复用

继承的语法规则是什么?

class 子类 extends 父类

例如:

class Dog extends Animal{

......

}

下面来实践一下,实现一个简单的继承。

第一步:创建一个类 Animal

package com.example;

public class Animal {

public int age;

public String name;

public void eat() {

System.out.println("动物具有吃东西的能力!");

}

}

第二步:创建另一个类 Dog,让其继承自类Animal

package com.example;

public class Dog extends Animal {

}

math?formula=%5Ccolor%7Bred%7D%7B%E6%B3%A8%E6%84%8F%EF%BC%9A%7D这一步,在我们的子类中没有定义任何的变量和方法

第三步:main方法中实例化Dog类

package com.example;

public class Main {

public static void main(String[] args) {

// write your code here

Dog dog = new Dog(); // 实例化一个Dog对象

dog.age = 10;

dog.name = "huahua";

dog.eat();

}

}

运行一下看看,结果如下:

动物具有吃东西的能力!

我们看到,在这里我们实例化了一个子类对象,然而它也拥有了属性 age 和 name,以及方法eat(),这些属性和方法完全来自于其父类。

math?formula=%5Ccolor%7Bred%7D%7B%E8%BF%99%E9%87%8C%E9%9C%80%E8%A6%81%E6%B3%A8%E6%84%8F%E7%9A%84%E6%98%AF%EF%BC%8C%E5%A6%82%E6%9E%9C%E7%88%B6%E7%B1%BB%E7%9A%84%E5%B1%9E%E6%80%A7%E6%88%96%E6%96%B9%E6%B3%95%E6%98%AF%E7%94%B1private%E5%85%B3%E9%94%AE%E5%AD%97%E4%BF%AE%E9%A5%B0%E7%9A%84%EF%BC%8C%E9%82%A3%E4%B9%88%EF%BC%8C%E5%AD%90%E7%B1%BB%E5%B0%B1%E4%B8%8D%E8%83%BD%E7%BB%A7%E6%89%BF%E8%AF%A5%E5%B1%9E%E6%80%A7%E6%88%96%E6%96%B9%E6%B3%95%E3%80%82%7D比如,我们将父类中的属性age改成private修饰后,测试类中的子类对象再次调用age属性时,编译器提示错误如下:

43cf61fe5ae1

3.2 Java中的方法重写

math?formula=%5Ccolor%7Bred%7D%7B%E4%BB%80%E4%B9%88%E6%98%AF%E6%96%B9%E6%B3%95%E7%9A%84%E9%87%8D%E5%86%99%EF%BC%9F%7D

如果子类对继承父类的方法不满意,是可以重写父类继承的方法的,当调用方法时会优先调用子类的方法。

方法重写的语法规则是什么?

返回值类型

方法名

参数类型及个数

以上这几点都要与继承父类的方法相同。同样,我们也来代码实践一下。

// 父类:Animal

package com.example;

public class Animal {

public int age;

public String name;

public void eat() {

System.out.println("动物具有吃东西的能力!");

}

}

// 子类:Dog

package com.example;

public class Dog extends Animal {

// 重写父类的eat()方法 [注意:返回类型/方法名/参数个数 都要与父类方法相同]

public void eat() {

System.out.println("狗具有吃东西的能力!");

}

}

// main方法

package com.example;

public class Main {

public static void main(String[] args) {

// write your code here

Dog dog = new Dog(); // 实例化一个子类Dog对象

dog.eat();

}

}

我们来看看程序运行结果,如下:

狗具有吃东西的能力!

从结果来看,方法重写后,优先调用的是子类本身重写后的方法。

3.3 Java中的继承初始化顺序

继承的初始化顺序是什么?

初始化父类,再初始化子类

先执行初始化对象中的属性,再执行构造方法中的初始化

说白了,初始化过程其实就是相当于执行类的构造方法。同样,我们也来代码实践一下。

首先在父类Animal中的构造方法中打印一句话,如下:

// 父类:Animal

package com.example;

public class Animal {

public int age;

public String name;

public void eat() {

System.out.println("动物具有吃东西的能力!");

}

// Animal类的无参构造方法

public Animal(){

System.out.println("Animal类执行了!");

}

}

然后,在子类Dog中的构造方法中同样也打印一句话,如下:

// 子类:Dog

package com.example;

public class Dog extends Animal {

// 重写父类的eat()方法 [注意:返回类型/方法名/参数个数 都要与父类方法相同]

public void eat() {

System.out.println("狗具有吃东西的能力!");

}

public Dog(){

System.out.println("Dog类执行了!");

}

}

然后,在main方法中只创建一个子类对象,如下:

// main方法

package com.example;

public class Main {

public static void main(String[] args) {

// write your code here

Dog dog = new Dog(); // 实例化一个子类Dog对象

}

}

接着,我们来执行一下程序,看看结果如下:

Animal类执行了!

Dog类执行了!

从上面的结果我们可以看到,math?formula=%5Ccolor%7Bred%7D%7B%E5%9C%A8%E5%88%9B%E5%BB%BA%E5%AD%90%E7%B1%BB%E5%AF%B9%E8%B1%A1%E7%9A%84%E6%97%B6%E5%80%99%EF%BC%8C%E5%85%88%E6%89%A7%E8%A1%8C%E7%9A%84%E6%98%AF%E7%88%B6%E7%B1%BB%E7%9A%84%E6%9E%84%E9%80%A0%E6%96%B9%E6%B3%95%EF%BC%8C%E5%90%8E%E6%89%A7%E8%A1%8C%E7%9A%84%E6%98%AF%E5%AD%90%E7%B1%BB%E7%9A%84%E6%9E%84%E9%80%A0%E6%96%B9%E6%B3%95%7D

3.4 Java中final的使用

final可以修饰 类/属性/方法/变量

final 修饰类的时候,该类不允许被继承

final 修饰方法的时候,该方法不允许被覆盖(重写)

final 修饰属性的时候,该类的属性不会进行隐式的初始化(类的初始化属性必须要有值)或者在构造方法中赋值

final 修饰变量的时候,该变量的值只能赋一次值,即变为常量

3.5 Java中super的使用

math?formula=%5Ccolor%7Bred%7D%7Bsuper%E5%85%B3%E9%94%AE%E5%AD%97%E6%98%AF%E5%95%A5%E6%84%8F%E6%80%9D%EF%BC%9F%7D

在对象内部使用,可以代表父类对象

既然super可以代表父类对象,就可以访问父类的属性,以及调用父类的方法。同样,我们也来代码实践一下。

// 父类:Animal

package com.example;

public class Animal {

public int age = 10;

public String name;

public void eat() {

System.out.println("动物具有吃东西的能力!");

}

// 父类的构造函数

public Animal(){

System.out.println("Animal类执行了!");

}

}

// 子类:Dog

package com.example;

public class Dog extends Animal {

public int age = 20;

//重写父类eat()方法

public void eat() {

System.out.println("狗具有吃东西的能力!");

}

// 子类的构造函数

public Dog() {

System.out.println("Dog类执行了!");

}

public void show() {

System.out.println("super关键字获取父类属性age:"+ super.age); //使用super关键字获得父类的属性age

System.out.println("子类本身的属性age:"+ age); // 获取子类本身的属性age

super.eat(); // 使用super关键字获取父类的方法

eat(); // 获取子类本身的方法

}

}

// main方法

package com.example;

public class Main {

public static void main(String[] args) {

// write your code here

Dog dog = new Dog(); // 实例化一个子类Dog对象

dog.show();

}

}

接着,我们来执行一下程序,看看结果如下:

Animal类执行了!

Dog类执行了!

super关键字获取父类属性age:10

子类本身的属性age:20

动物具有吃东西的能力!

狗具有吃东西的能力!

3.6 Java中object类

math?formula=%5Ccolor%7Bred%7D%7BObject%E7%B1%BB%E6%98%AF%E6%89%80%E6%9C%89%E7%B1%BB%E7%9A%84%E7%88%B6%E7%B1%BB%EF%BC%8C%E5%A6%82%E6%9E%9C%E4%B8%80%E4%B8%AA%E7%B1%BB%E6%B2%A1%E6%9C%89extends%E5%85%B3%E9%94%AE%E5%AD%97%E6%98%8E%E7%A1%AE%E6%A0%87%E8%AF%86%E7%BB%A7%E6%89%BF%E5%8F%A6%E4%B8%80%E4%B8%AA%E7%B1%BB%EF%BC%8C%E9%82%A3%E4%B9%88%E8%BF%99%E4%B8%AA%E7%B1%BB%E5%B0%B1%E9%BB%98%E8%AE%A4%E7%BB%A7%E6%89%BF%E8%87%AAobject%E7%B1%BB%7D。Object类中的方法,适合所有子类。

3.6.1 toString()方法

在Object类里面定义toString()方法的时候返回的是 对象的哈希code码(对象地址字符串)。

可以通过重写 toString()方法 表示出对象的属性。

比如:在上面的代码中,如果我们在main方法中直接输出Dog对象,如下:

package com.example;

public class Main {

public static void main(String[] args) {

// write your code here

Dog dog = new Dog(); // 实例化一个子类Dog对象

// dog.show();

System.out.println(dog);

}

}

我们看到结果如下:

Animal类执行了!

Dog类执行了!

com.example.Dog@1b6d3586

显然,上面的输出不是我们满意的输出(因为不知道是啥意思,呜呜~)。如果我们想知道对象的属性值,这个时候就需要重写从父类继承过来的 **toString() **方法,如下:

// 子类:Dog

package com.example;

public class Dog extends Animal {

public int age = 20;

//重写父类eat()方法

public void eat() {

System.out.println("狗具有吃东西的能力!");

}

// 子类的构造函数

public Dog() {

System.out.println("Dog类执行了!");

}

public void show() {

System.out.println("super关键字获取父类属性age:"+ super.age); //使用super关键字获得父类的属性age

System.out.println("子类本身的属性age:"+ age); // 获取子类本身的属性age

super.eat(); // 使用super关键字获取父类的方法

eat(); // 获取子类本身的方法

}

@Override

public String toString() {

return "Dog [age=" + age + "]";

}

}

然后直接运行main()方法,结果如下:

Animal类执行了!

Dog类执行了!

Dog [age=20]

3.6.2 equals()方法

equals()方法:比较的是对象的引用是否指向同一块内存地址。

一般情况下,比较两个对象时,是比较的两个对象的值是否相同,所以要进行重写对象方法。

同样,我们还是来代码实践一下。父类和子类还是之前的Animal和Dog,我们只改变一下main方法,如下:

// main方法

package com.example;

public class Main {

public static void main(String[] args) {

// write your code here

Dog dog1 = new Dog(); // 实例化一个Dog对象

dog1.age = 15;

Dog dog2 = new Dog();

dog2.age = 15;

if(dog1.equals(dog2)) {

System.out.println("两个对象是相同的!");

}else {

System.out.println("两个对象是不相同的!");

}

}

}

我们来看看运行结果,如下:

Animal类执行了!

Dog类执行了!

Animal类执行了!

Dog类执行了!

两个对象是不相同的!

从上面的运行结果,我们看到,虽然两个对象的属性age值时一样的,但通过equals方法比较的结果是 “两个对象是不相同的!” 这是因为实例化两个对象的过程,实际上是在内存中开辟两个内存地址,分别存放两个实例化对象,而equals()方法比较的是对象的引用,也就是对象的地址,所以两个对象是不同的!

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值