[转载] JAVA面向对象之代码块 继承 方法的重写 super关键字与重写toString()方法介绍

参考链接: 可以重写Java中的私有方法吗

JAVA面向对象之代码块与继承 

代码块分类 

局部代码块 

作用:限制变量生命周期

书写位置:在方法中 

构造代码块 

开发中很少使用

书写位置:类中  方法外

调用时机:如果你有构造代码块 系统会帮你调用 帮你在创建对象时调用 

静态代码块(一定是被static修饰) 

依赖类 随着类的加载而加载

注意:只加载一次(系统只创建一次 不管你调用多少对象)

应用场景:U盘装载驱动程序(第二次插入U盘,不会再加载驱动程序)

         加载驱动(数据库驱动 JDBC) 

同步代码块(多线程) 

这里暂时不做介绍 后续给予说明 

代码示例 

public class Demo24 {

           //第三代码块

    {

         int a = 10;

         System.out.println(a);

         System.out.println("我是Demo24构造代码块");

    }

    public static void main(String[] args){

        Person2 person = new Person2();

        person.name = "张三";

        person.age = 15;

        person.sayHi();

        System.out.println("我是main函数中得普通方法");

        {

            System.out.println("我是局部代码块");

        }

        Person2 person1 = new Person2("李四",22);

    }

}

class Person2{

    String name;

    int age;

    //第一个代码块

    static {

          System.out.println("我是person类的静态代码块");

    }

    //第二个代码块

    {

        //每一个对象 都会调用同一个方法 这时可以使用

           sleep();

           System.out.println("我是构造代码块");

    }

    public Person2(){

           System.out.println("我是无参数的构造方法");

    }

    public Person2(String name, int age){

          this.name = name;

          this.age = age;

          System.out.println("我是有参数的构造方法");

    }

 

    public String getName(){

            return name;

    }

    public void setName(){

            this.name = name;

    }

    public int getAge(){

            return age;

    }

    public void setAge(int age){

             this.age = age;

    }

 

    public void sayHi(){

           System.out.println("姓名:" + name + "年龄:" + age);

    }

 

     public void sleep(){

          System.out.println("睡觉");

     }

}

 

 

运行结果 

我是person类的静态代码块

睡觉

我是构造代码块

我是无参数的构造方法

姓名:张三年龄:15

我是main函数中得普通方法

我是局部代码块

睡觉

我是构造代码块

我是有参数的构造方法

姓名:李四年龄:22 

解释 

当程序运行时 会先加载Demo01.class文件,进入.class文件后,会寻找静态代码块,如果没有,会继续寻找构造代码块,由于没有

Demo24类的实例对象,所有构造代码块与构造方法均不会执行,如果另外新建一个类,Demo25,在Demo25类中建立Demo24类的实例

对象,Demo24类中的构造代码块与构造方法便会执行,之后会首先寻找main函数,找到main函数后,(局部代码块定义在方法中,在该

方法中并没有优先级)会首先遇到Person2类,遇到Person2 person = new Person2();这时会加载Person2类,

加载Person2.class文件,静态代码块与静态方法和静态属性一样,会随着类的加载而加载,存放在方法区的静态区,与对象无关,所

以在加载Person2.class文件时,会将static代码块一同加载,所有会输出'我是Person类的静态代码块',之后需要new一个对象,

继续执行Person2类中的代码,执行时,会先寻找有没有构造代码块(构造代码块存放在类中方法外,优先级高于构造方法);发现有构

造代码块,即先执行构造代码块,构造代码块中有sleep方法,故先执行sleep方法,打印'睡觉',之后执行下一句,打印'我是构造代码

块',构造代码块执行之后,会执行相应的构造方法(有参或者无参),构造代码块优先于构造方法执行,与构造代码块的位置无关,new

对象的时候没有传递参数,所有这里调用无参构造方法,打印'我是无参数的构造方法',之后回到Demo01函数,对对象中的变量进行赋

值,赋完值后,调用对象的sayHi方法,打印'姓名:张三','年龄:15',随后执行到打印语句,打印出"我是main函数中得普通方法",

之后遇到局部代码块,打印"我是局部代码块"(顺序执行),当new第二个Person2对象时,静态代码块不会再执行,即在函数运行过

程中,只会加载一次,非静态构造代码块将会继续加载,第二次new对象时候,先执行构造代码块,构造代码块中有sleep方法,所有会

先执行sleep方法,打印"睡觉",随后打印"我是构造代码块",new对象时候,传进来了参数,所有会调用Person2类中的有参数构造

方法,打印"我是有参数的构造方法",赋值后,调用sayHi方法,打印"姓名:李四年龄22",函数运行结束. 

继承 

继承特点 

1.减少你的代码量

2.让类与类之间产生关联(产生 父子的) 

继承弊端 

当父类中添加新德属性时候,比如白血病,子类即使不想继承这个属性,却还是由于

继承的关系,自动得到了白血病这个属性 

注意 

1.继承时  可把多个类中 相同的功能或方法 抽取出来 重新构造一个类出来  

把这些类建立 继承关系

2.建立继承关系的同时 一定要符合逻辑(切记不要为了继承而继承)

3.继承使用关键字:extends 

举例 

继承:

手机类  <打电话 发短信 玩游戏>

苹果手机类 继承 手机类 <打电话 发短信 玩游戏 爆炸>

小米手机类 继承 手机类 <打电话 发短信 玩游戏 暖手功能> 

项目经理:姓名 工资 工号 分红

程序员:姓名 工资 工号

 

项目经理 继承 程序员 就继承了姓名 工资 和 工号 ,特有功能 分红 这样不行 

不符合逻辑

应该是相同的功能抽取出来

员工类 姓名 工资 工号

项目经理 和 程序员 继承员工类  这样才符合逻辑 

注意 

如果是继承关系 一定符合什么是什么

项目经理是员工 子类是父类的 

动物 猫 狗 马

猫 是 动物 √

动物 是 猫 ×

 

水果(父类) 香蕉 苹果 橘子

水果 是 香蕉 ×

香蕉 是 水果 √ 

继承的写法 

class 子类 extends 父类{

 

代码示例 

/*

 * 猫类

 * 姓名 颜色  种类 会睡觉 会抓老鼠

 * 狗类 

 * 姓名 颜色  种类 会睡觉 会啃骨头

 */

 

// 抽取出 相同部分 组成 动物类

public class Demo02{

    public static void main(String[] args){

        Cat cat = new Cat();

        cat.name = "汤姆";

        cat.color = "灰色";

        cat.kind = "灰猫";

        cat.sleep();

        cat.sayHi();

    }

}

 

class Animal{

    String name;

    String color;

    String kind;

 

    public void sleep(){

        System.out.println("睡觉");

    }

    public void sayHi(){

        System.out.println("姓名:" + name +"颜色:" + color + "种类:" + kind);

    }

}

 

class Cat extends Animal{

    public void hitMouse(){

        System.out.println("抓老鼠");

    }

}

 

class Dog extends Animal{

    public void eatBone(){

        System.out.println("啃骨头");

    }

运行结果: 

抓老鼠

睡觉

姓名:汤姆颜色:灰色种类:灰猫 

JAVA中的继承 

注意 

1.java 只允许 单继承(多继承 可以 使用 接口 来间接实现)

2.java 中 允许 多层继承(爷爷 父亲 儿子 孙子 重孙子....)

 

java中 最顶层的父类(最基础类) Object类

如果我这个类没有写 继承哪个父亲 默认就是继承 Object类 

1.如果我要使用 这些 类中 共有的方法(属性) 使用 哪个类?

    创建当前继承中最 顶端的 类 去使用

2.如果我要使用 这些 类中 特有的方法(属性) 使用 哪个类?

    创建当前继承中 最末端类 去使用 

代码示例 

public class Demo03 extends Object{

 

}

//A类 爷爷类 B类 是父亲 C类 是 孙子类

//A类中又name C类中 会叫爷爷

 

class A extends Object{

    String name;

}

 

class B extends A{

 

}

class C extends B{

    public void speak(){

        System.out.println("会叫爷爷");

    }

构造方法能不能被继承? 

爷爷 父亲 儿子

爹 和 儿子 都用同一中 方法生出来 行吗? 乱

奶奶不愿意  妈妈不愿意

 

构造方法是不能被继承的

 

为了保证继承的完整性 在你创建对象的时候

如果你不调用 父类的构造方法 

系统会帮你去调用 父类的无参构造方法 

代码举例 

public class Demo04{

    public static void main(String[] args){

        Son son = new Son();

        son.name = "张三";

        son.sayHi();

 

        //有参构造对象

        Son son2 = new Son("小明");

        son2.sayHi();

    }

}

 

class Father{

    String name;

    //有参 无参 构造方法

    public Father(){

        System.out.println("我是无参构造方法");

    }

    public Father(String name){

        this.name = name;

        System.out.println("我是有参构造方法");

    }

    public void sayHi(){

        System.out.println("姓名:" + name);

    }

}

 

class Son extends Father{

    //无参构造

    public Son(){

        //如果你没有在子类的构造方法中 调用父类的构造方法

        //系统会默认给你的子类构造方法中 添加一行代码

        super();//调用父类的 无参构造方法

        System.out.println("我是儿子类的无参构造方法");

    }

    //有参的

    public Son(String name){

        //如果你没有在子类的构造方法中 调用父类的构造方法

        //系统 会默认 给你的子类 构造方法中 添加一行代码

        super(); //系统帮你调用父类的构造方法

        this.name = name;

        System.out.println("我是儿子类的有参构造方法");

    }

输出结果 

我是爸爸类无参的构造方法

我是儿子类无参构造方法

张三

 

我是爸爸类无参的构造方法

我是儿子类有参的构造方法

小明 

结果解释 

当new一个Son对象时,由于没有传进去参数,所以会先调用儿子类的无参构造方法,

由于Son类是Father的子类,所有在子类的构造方法中会自动调用父类的无参构造方

法,从而实现Son类的实例对象同时具有Son类与Father类的属性与方法,因此先打印

父类无参构造方法中的"我是爸爸类的无参构造方法",之后返回子类构造方法,打印

子类无参构造器中得"我是儿子类无参构造方法",之后对Son的nane属性赋值,调用s

ayHi方法将姓名打出

 

第二次new一个Son类对象的时候,传进去姓名这个参数,所有会调用Son类中得有参

构造方法,该有参构造方法中第一句会先执行super(),即会先执行父类无参构造方

法,向上调用父类构造方法时,没有传进去参数,所有不会调用父类有参构造方法,打

印"我是爸爸类无参构造方法"后,返回子类,即Son类,打印"我是儿子类有参构造方

法",然后将姓名赋值,随后返回main函数,调用sayHi方法将name打印出来. 

super关键字 

super 用于指向子类对象中的父类对象(构造方法)  相当于父类的对象

super 调用对象 super.对象

super 调用方法 super.方法()

 

 

super(); 调用父类的构造方法

this();  调用的是本类的构造方法 

代码示例 

public class Demo05{

    public static void main(String[] args){

        TestB b = new TestB();

        b.fun();

    }

}

 

class TestA{

    int num1 = 10;

    int num2 = 20;

 

    public void sayHi(){

        System.out.println("我是父类的sayHi方法");

    }

}

 

class TestB extends TestA{

    int num1 = 30;

 

    public void fun(){

        //使用this时 会先在本类中寻找该属性

        //没找到 就去父类中找  就近原则

        System.out.println(this.num1);  //30

        System.out.println(this.num2);  //20

        System.out.println(this.num3);  //10

    }

思考:如果父类中没有无参构造方法 咋整? 

建议:不管是父类 还是 子类 构造方法一定要写全,避免出现问题 

代码举例 

public class Demo04{

 

}

class Phone{

    String name;

 

    public Phone(){

 

    }

    public Phone(String name){

        this.name = name;

    }

}

class MI extends Phone{

    public MI(){

        //子类的构造方法 如果你什么都不写 会默认调父类无参构造

        //如果父类中 没有无参构造 就证明父类中一定有有参的构造方法

        //父类构造方法无论有参 还是 无参, 子类的构造方法都必须要调用一个

 

 

        //必须手动指定一个有参构造方法去调用

        super("Note2");

    }

    public MI(String name){

        super(name);

    }

方法的重写 

思考: 如果父类 和 子类的 方法 重名了,咋整?能不能重名?

 

答案是可以  方法的重写 

注意 

1.方法的声明完全一致的 叫方法的重写

2.方法的重写建立在类与类之间有继承关系(子类重写父类的方法) 

Override(重写)和Overload(重载) 的区别 

1.重写前提:需要继承关系

重载:在同一个类里面实现

 

2.重写:需要方法的声明完全一致

重载:相同的功能,不同的实现方式 只跟参数有关 

代码示例 

public class Demo07{

    public static void main(String[] args){

        IOS8 ios8 = new IOS8();

        ios8.siri();

 

        //如果直接打印对象 相当于 系统帮你打印时 调用 toString()方法

        System.out.println(ios8);   //输出haha

        System.out.println(ios8.toString());   //输出haha

    }

}

 

class IOS7{

    public void call(){

        System.out.println("打电话");

    }

 

    public void siri(){

        System.out.println("说英文");

    }

}

 

class IOS8 extends IOS7{

    //方法的重写:对父类的方法 进行一个功能上的升级

    //调不调父类的方法 要根据实际情况

    public void siri(){

        //中 英文都会说

        super.siri();   //调用父类的方法

        System.out.println("会说中文");

    }

    //重写toString()方法

    //利用toString方法 来写介绍自己的方法

    public String toString(){

        return "haha";

    }

重写toString()方法介绍 

toString方法是Object类中的一个方法,故所有继承Object类的类,这些类中都有toString方法;

假设有一个类为Animal,他的一个对象为animal,则System.out.println(animal),

打印出来的是一个全限定类名com.lanou3g.IOS7@33909752,而打印System.out.println(animal.toString),

.lanou3g.IOS7@33909752,即包名+类名+@33909752,说明打印animal与打印animal.toString()的结果是一样的,toString()方法写完整则是

public String toString(){},即说明toString方法是有返还值的,即打印animal.t

oString()的结果,打印的是toString()函数中return语句返还的值,而打印animal

的结果又与打印animal.toString()方法的值相同,说明打印对象名所得到的结果就

是toString中return语句返还的结果,所有重写Object类中得toString方法,改变re

turn语句的返还值,然后直接打印对象名,便可以得到自己想要的结果.具体结果如上面代码所示 

实例练习继承关系 

需求 

 

老师类 学生类  

* 无参 有参构造 set/get方法 成员变量私有化 介绍自己的方法

  属性:姓名,年龄

* 行为:吃饭

* 老师有特有的方法:讲课

* 学生有特有的方法:学习 

代码实现 

package com.lanou3g;

 

 

public class Demo08 {

    public static void main(String[] args) {

        //创建一个学生

        Student student = new Student("王龙",15,1);

        System.out.println(student);

 

        Teacher teacher = new Teacher("刘",13);

        System.out.println(teacher);

    }

}

class Person1{

    //属性

    //继承中private修饰的变量 是不能直接访问的 但可以间接访问的

    private String name;

    private int age;

    //构造方法

    public Person1() {

 

    }

    public Person1(String name, int age) {

        this.name = name;

        this.age = age;

    }

    //set/get方法

    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 void eat() {

        System.out.println("吃饭");

    }

    //介绍自己的方法

    @Override

    public String toString() {

        // TODO Auto-generated method stub

        return "姓名:" + name + "年龄" + age;

    }

 

}

//学生类

class Student extends Person1{

    //学号

    private int num;

 

    public int getNum() {

        return num;

    }

    public void setNum(int num) {

        this.num = num;

    }

    //构造方法

    public Student(){

    }

    //有参

    public Student(String name, int age, int num) {

        //直接调用父类的构造方法 完成赋值初始化

        //为了保证继承的完整性 在子类构造方法中

        //第一行 必须调用父类的构造方法(无参 有参都行)

        super(name,age);

        //特有属性 直接赋值就行

        this.num = num;

    }

    public void study() {

        System.out.println("学生在学习");

    }

    //介绍自己的方法

    @Override

    public String toString() {

        //可以在父类的基础上 添加自己特有的属性 去打印

        return super.toString() + "学号:" + num;

    }

}

//教师类

class Teacher extends Person1 {

    public Teacher(){

 

    }

    public Teacher(String name, int num) {

        super(name,num);

    }

    public void teach() {       

        System.out.println("老师会讲课");

    }

    @Override

    public String toString() {

        // TODO Auto-generated method stub

        return super.toString();

    }

}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值