8.面向对象(代码块,继承,方法重写与方法重载的区别,final关键字)

1.代码块的概述和分类

1.代码块概述
    在Java中,使用{}括起来的代码被称为代码块。
2.代码块分类
    根据其位置和声明的不同,可以分为局部代码块,构造代码块,静态代码块,同步代码块(多线程讲解)。
3.常见代码块的应用
    a:局部代码块 
        在方法中出现;限定变量生命周期,及早释放,提高内存利用率
    b:构造代码块 
        在类中方法外出现;多个构造方法方法中相同的代码存放到一起,每次调用构造都执行,并且在构造方法前执行
    c:静态代码块 
        在类中方法外出现,加了static修饰
        在类中方法外出现,并加上static修饰;用于给类进行初始化,在加载的时候就执行,并且只执行一次。

局部代码块:

public class MyTest {
    public static void main(String[] args) {
        //代码块:就是被一对大括号,所扩起起来的代码。
        //代码块,根据定义的位置不同,分为:局部代码块,构造代码块,静态代码块。同步代码块(后面讲)
        //局部代码块,定义在方法中的代码块。
        //构造代码块:定义在类中方法外,跟构造方法平级
        //静态代码代码块:定义在类中方法外,用static所修饰的代码块、
        int a=100;

        {
            //在方法中出现;限定变量生命周期,及早释放,提高内存利用率
            int num=100;
            System.out.println("这是局部代码块"+num);
            System.out.println(a);
        }

      // System.out.println(num);

    }
}


 Student类:

public class Student {
   static int num=100;
    {
        System.out.println("这是构造代码块");
    }
    static{
        System.out.println("这是静态代码块");
        System.out.println("静态代码块,比较常用,经常在静态代码块里面做一些初始化的准备工作。");
        System.out.println(num);
    }
    public Student() {
        System.out.println("这是空参构造");
    }
}

测试类:

public class MyTest2 {
    public static void main(String[] args) {
        //创建对象时,会调用构造代码块,而且还是优先于构造方法执行。每次创建对象,都会调用构造代码块。
        //静态代码块,随着类的加载而加载,优先于构造代码块执行。且只会执行一次。在静态代码块里面,只能访问静态成员。
        Student student = new Student();

        Student student1 = new Student();
    }
}


2.代码块的面试题

看程序写结果
    class Student {
        static 
            System.out.println("Student 静态代码块");  //3
        }
        
        {
            System.out.println("Student 构造代码块");//4 6
        }
        
        public Student() {
            System.out.println("Student 构造方法");//5 7
        }
    }

    class StudentDemo {
        static {
            System.out.println("StudentDemo的静态代码块"); //1
        }
        
        public static void main(String[] args) {
            System.out.println("我是main方法"); //2
        
            Student s1 = new Student();
            Student s2 = new Student();
        }
    }  


3.继承的引入和概述

1.继承概述
    多个类中存在相同属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属性和行为,只要继承那个类即可。
2.继承格式
    通过extends关键字可以实现类与类的继承
    class 子类名 extends 父类名 {}  
    单独的这个类称为父类,基类或者超类;这多个类可以称为子类或者派生类。


4.继承案例演示以及继承的好处和弊端

1.需求:
    Dog类和Cat类和Animal类。
    定义两个功能(吃饭,睡觉)
2.案例演示
    使用继承前
3.案例演示
    使用继承后
4.继承的好处
    a:提高了代码的复用性
    b:提高了代码的维护性
    c:让类与类之间产生了关系,是多态的前提
5.继承的弊端
    类的耦合性增强了。
    开发的原则:高内聚,低耦合。
    耦合:类与类的关系
    内聚:就是自己完成某件事情的能力

Animal类:

public class Animal {
    String name;
    int age;

    public void eat() {
        System.out.println("吃饭饭55555555555555555555555");
    }

    public void sleep() {
        System.out.println("睡觉");
    }

Dog类:

public class Dog extends Animal{


    public void lookDoor(){
        System.out.println("看门");
    }

}

Cat类:

public class Cat extends Animal{

    public void catchMouse(){
        System.out.println("抓老鼠");
    }
}

测试类:

public class MyTest {
    public static void main(String[] args) {
        //面向对象的三大特征:封装(class) 继承 多态
        //生活中的继承:儿子可以继承老子的财产。
        //Java中的继承 :子类可以继承父类的成员。
        //可以使用 extends 关键字,可以让类和类之间形成父子关系。
        //继承的好处:可以提高代码的复用性和维护性。
        //继承的思想:把多个子类的共性内容,向上抽取到父类当中,以实现代码的复用性和维护性。
        //继承的弊端:让类和类之间产生了关系,增加了耦合性。
        //软件设计原则中:高内聚(某个事物独立完成功能),低耦合
        Cat cat = new Cat();
        cat.name="汤姆";
        cat.age=15;
        System.out.println(cat.name);
        System.out.println(cat.age);
        cat.eat();
        cat.sleep();
        cat.catchMouse();

        System.out.println("====================================");
        Dog dog = new Dog();
        dog.name="旺财";
        dog.age=5;
        System.out.println(dog.name);
        System.out.println(dog.age);
        dog.sleep();
        dog.eat();
        dog.lookDoor();
    }
}


5.Java中类的继承特点

1.Java中类的继承特点
    a:Java只支持单继承,不支持多继承。
        有些语言是支持多继承,格式:extends 类1,类2,...
    b:Java支持多层继承(继承体系)
2.案例演示
    Java中类的继承特点


6.继承的注意事项和什么时候使用继承

1.继承的注意事项
    a:子类只能继承父类所有非私有的成员(成员方法和成员变量)
    b:子类不能继承父类的构造方法,但是可以通过super关键字去访问父类构造方法。
    c:不要为了部分功能而去继承
2.什么时候使用继承
    继承其实体现的是一种关系:"is a" .        
    采用假设法。
        如果有两个类A,B。只有他们符合A是B的一种,或者B是A的一种,就可以考虑使用继承。

public class MyTest {
    public static void main(String[] args) {
        //继承的注意事项;
        //1.在Java中,只支持单继承,一个子类只能有一个父类,不能有多个父类,但是支持多层继承。
        //Object 是所有类的顶层父类,所有类都是直接或间接继承自他。
        //2. 构造方法不参与继承。当我们创建子类对象时,会先调用父类的构造方法,来完成父类数据的初始化。
        //3. 父类私有的成员,子类不能继承。

        Zi zi = new Zi();
        System.out.println(zi.a);
        System.out.println(zi.b);
        //zi.fu();
    }
}

class A extends Object{
    int a=100;
    private int bb=20;
}

class Fu extends A{
    int b=20;
    private double cc=3.14;

    private void fu(){

    }
}

class Zi extends Fu{

}

 


7.继承中成员变量的关系

1.子类中的成员变量和父类中的成员变量名称不一样
2.子类中的成员变量和父类中的成员变量名称一样
    在子类中访问一个变量的查找顺序("就近原则")
        a: 在子类的方法的局部范围找,有就使用
        b: 在子类的成员范围找,有就使用
        c: 在父类的成员范围找,有就使用
        d:如果还找不到,就报错


8.this和super的区别和应用

1.通过问题引出super
    子类局部范围访问父类成员变量
2.说说this和super的区别
    this    代表的是本类对象的引用
    super  代表的是父类存储空间的标识(可以理解成父类的引用,可以操作父类的成员)
3.this和super的使用
    a:调用成员变量
        this.成员变量  调用本类的成员变量
        super.成员变量     调用父类的成员变量
    b:调用构造方法
        this(...)            调用本类的构造方法
        super(...)            调用父类的构造方法
    c:调用成员方法
        this.成员方法         调用本类的成员方法
        super.成员方法     调用父类的成员方法

练习:

public class MyTest {
    public static void main(String[] args) {
        Zi zi = new Zi();
        zi.show(5);
    }
}

class Fu{
    int num=100;
}

class Zi extends Fu{
    int num=20;

    public void show(int num){
        System.out.println(num); //5
        System.out.println(this.num); //20
        System.out.println(super.num); //100
    }
}

Father类:

public class Father {
    int num = 3000;
    double aa =6.999;
    boolean flag=true;

    public void show() {
        System.out.println("父类的show方法");
    }

    public void hehe() {
        System.out.println("父类的hehe方法");
    }
}

Son类:

public class Son extends Father {
    int num=100;
    double aa=3.14;

    char ch='a';

    public void ziShow(int num){
        /*
        * 变量的访问原则:遵循就近原则,先在局部范围找(方法发内,以及形参) 如果找不到,就去本类的成员范围找(成员变量)
        * 如果本类成员范围没找到,再去父类成员范围找,找到就使用,再找不到就报错。
        *
        * 当局部变量,本类成员变量,父类成员变量重名了,你可以使用 this super 来区分。
        *
        *
        * */
        System.out.println("子类的show方法");
        System.out.println(num);
        System.out.println(this.num);
        System.out.println(super.num);
        System.out.println(aa);
        System.out.println(super.aa);
        System.out.println(this.aa);
        System.out.println(flag);
        System.out.println(super.flag);
        System.out.println(this.flag);

        System.out.println(ch);
    }


    public void haha(){
        this.show(); //父类的这个show()方法被你继承了,相当于是你的了,所以前面加this也可以。
        super.show(); //调用父类的方法
        this.hh(); //调用本类方法
    }

    public void hh(){
        System.out.println("hhhhhhhh");
    }
}

测试类:

public class MyTest {
    public static void main(String[] args) {
        //this 和 super
        //super:代表父类空间的一个标识,你可以认为是父类的一个引用,使用它可以访问父类的数据。
        //this:代表本类的一个引用,你使用this,那访问的就是本类的数据。
        //访问成员变量
        //this.成员变量  this.num
        //super.成员变量  super.num
        //访问成员方法:
        // this.成员方法名() this.show()
        // super.成员方法名() super.show()
        // 访问构造方法
        // this()
        // super()

        Son son = new Son();
        son.ziShow(20);
        son.haha();
    }
}


9.继承中构造方法的关系

1.案例演示
    子类中所有的构造方法默认都会访问父类中空参数的构造方法
2.为什么呢?
    因为子类会继承父类中的数据,可能还会使用父类的数据。
    所以,子类初始化之前,一定要先完成父类数据的初始化。
    
    其实:
        每一个构造方法的第一条语句默认都是:super()
        Object类是父类。

public class MyTest {
    public static void main(String[] args) {
        //Father father = new Father();
        //当我创建子类对象时,为什么会去先调用父类的空参构造方法,为什么?
        /*
        *  因为子类继承了父类,那也就是说,子类继承了父类的数据,甚至还有去使用父类的数据,
        * 所以当我们创建子类对象时,应该先完成父类数据的初始化,那么子类才能继承父类的数据和使用父类的数据。
        * 既然你要先完成父类数据的初始化,那肯定先要调用父类的构造方法,才能完成父类数据的初始化。
        *
        * */
        Son son = new Son();
    }
}

class Father{
    int num=100;
    public Father() {
        System.out.println("Father父类的空参构造执行了");
    }
}

class Son extends Father{
    int s=20;
    public Son() {
        super();
        System.out.println("son的空参构造执行了"+num+"==="+s);
    }
}

 


10.继承中构造方法的注意事项

1.案例演示
    父类没有无参构造方法,子类怎么办?
    a: 在父类中添加一个无参的构造方法
    b:子类通过super去显示调用父类其他的带参的构造方法
    c:子类通过this去调用本类的其他构造方法
        本类其他构造也必须首先访问了父类构造
2.注意事项
    super(…)或者this(….)必须出现在第一条语句上

public class MyTest {
    public static void main(String[] args) {
        //当我创建子类对象时,为什么会去先调用父类的空参构造方法,为什么?
        /*
         *  因为子类继承了父类,那也就是说,子类继承了父类的数据,甚至还有去使用父类的数据,
         * 所以当我们创建子类对象时,应该先完成父类数据的初始化,那么子类才能继承父类的数据和使用父类的数据。
         * 既然你要先完成父类数据的初始化,那肯定先要调用父类的构造方法,才能完成父类数据的初始化。
         * 怎么去调用的父类的构造方法呢?其实在每个类的构造方法前面第一行,有一条默认语句  super() 他就去调用父类的空参构造
         *
         * */

        Zi zi = new Zi();
        /*
        *
        *  如果说父类没有提供空参构造呢?
        * //1. 你自己手动提供一个空参构造。
        * //2. 可以调用父类有参构造方法。
        * */
    }
}

class Fu extends Object{
    int num=200;
    /*
    public Fu() {
        super();
        System.out.println("父类的构造方法调用了");
    }

     */
    public Fu(int num){
        System.out.println("父类的构造方法调用了");
        System.out.println(num);
    }
}
class Zi extends Fu{
    int aa=20;
    public Zi() {
        //super();
        //super(10); //调用父类有参构造
        this(20); //调用本类的有参构造
        System.out.println("子类的构造方法调用了");
    }

    public Zi(int num){
        super(num);
        System.out.println("子类的有参构造方法调用了");
    }
}

11.继承中的面试题

1.案例演示
    看程序写结果1
    class Fu{
        public int num = 10;
        public Fu(){
            System.out.println("fu"); //1
        }
    }
    class Zi extends Fu{
        public int num = 20;
        public Zi(){
            System.out.println("zi"); //2
        }
        public void show(){
            int num = 30;
            System.out.println(num); //30
            System.out.println(this.num); //20
            System.out.println(super.num); //10
        }
    }
    class Test {
        public static void main(String[] args) {
            Zi z = new Zi();
            z.show();
        }
    }

2.案例演示
    看程序写结果2
    class Fu {
        static {
            System.out.println("静态代码块Fu"); //  1
        }

        {
            System.out.println("构造代码块Fu"); //3
        }

        public Fu() {
            System.out.println("构造方法Fu");   //4
        }
    }

    class Zi extends Fu {
        static {
            System.out.println("静态代码块Zi");//2
        }

        {
            System.out.println("构造代码块Zi"); //5
        }

        public Zi() {
            super();
            System.out.println("构造方法Zi"); //6
        }
    }
    
    class Test{
        public static void main(String[] args){
            Zi z = new Zi(); //请问执行结果。
        }
    }


12.继承中成员方法关系

A:案例演示
    a:当子类的方法名和父类的方法名不一样的时候
    b:当子类的方法名和父类的方法名一样的时候
        通过子类调用方法:
            a: 先查找子类中有没有该方法,如果有就使用
            b: 在看父类中有没有该方法,有就使用
            c: 如果没有就报错


13.方法重写概述及其应用

1.什么是方法重写
    子类中出现了和父类中一模一样的方法声明(方法名,参数列表,返回值类型),也被称为方法覆盖,方法复写。
2. Override和Overload的区别?Overload能改变返回值类型吗?
3.方法重写的应用:
    当子类需要父类的功能,而功能主体子类有自己特有内容时,可以重写父类中的方法。
    这样,即沿袭了父类的功能,又定义了子类特有的内容。
4.案例演示
    a:定义一个手机类。
    b:通过研究,我发明了一个新手机,这个手机的作用是既可以语音通话,又可以视频通话。

Phone类:

public class Phone {

    public void call(){
        System.out.println("语音通话");
    }
}

IPhone类:

public class IPhone extends Phone {
    //方法重写:当子类出现和父类一模一样的方法(返回值类型,方法名,形参一样),就会发生子类方法,覆盖父类的方法的这种现象。
    //为什么要有方法重写的这种机制?因为有可能子类对父类的方法实现,不满意,想要按照自己的逻辑进行实现,那么就可以通过方法重写来覆盖父类的方法。
    //ctrl+O可以快速重写父类方法
    @Override//注解: 可以检测该方法是不是重写父类的
    public void call() {
        super.call(); //继续沿用父类的功能
        System.out.println("视频通话");
    }
}
测试类:
public class MyTest {
    public static void main(String[] args) {
        IPhone iPhone = new IPhone();
        iPhone.call();
    }
}

14.方法重写的注意事项

1.方法重写注意事项
    a:父类中私有方法不能被重写
        因为父类私有方法子类根本就无法继承
    b:子类重写父类方法时,访问权限不能更低,最好就一致
    c:父类静态方法,不参与重写
    子类重写父类方法的时候,最好声明一模一样。
2.案例演示
    方法重写注意事项

public class MyTest {
    public static void main(String[] args) {
        //方法重写的注意事项:
        //1.子类在重写父类的方法时,方法的权限修饰符,不能比父类的低,要比父类的高或一样。
        //2. 父类私有的方法,子类不能重写,因为,私有的方法子类都不能继承,何谈重写。
        //3. 构造方法,不能重写,
        //4. 静态方法也不算重写

        Zi zi = new Zi();
        zi.hehe();
        Zi.hehe();
        Fu.hehe();
    }
}

class Fu {
    void show() {
    }
    public static  void hehe(){
        System.out.println("父类静态方法");
    }
}

class Zi extends Fu {
    @Override
    public void show() {
    }

    //@Override
    public static void hehe() {
        System.out.println("子类静态方法");
    }
}

 

15.使用继承前的学生和老师案例

     使用继承后的学生和老师案例

人类:

public class Person {
    String name;
    int age;

    public void eat() {
        System.out.println("吃饭");
    }

    public void sleep() {
        System.out.println("睡觉");
    }
}

 学生类:

public class Student extends Person{
    @Override
    public void eat() {
        System.out.println("学生爱吃麻辣烫");
    }

    @Override
    public void sleep() {
        System.out.println("学生喜欢上课睡觉");
    }

    public void homeWrok(){
        System.out.println("做作业");
    }
}

老师类:

public class Teacher extends Person{

    @Override
    public void eat() {
        System.out.println("老师爱吃燃面");
    }

    @Override
    public void sleep() {
        System.out.println("老师晚上睡觉");
    }

    public void teach(){
        System.out.println("教书");
    }
}

测试类:

public class MyTest {
    public static void main(String[] args) {
        Student student = new Student();
        student.name="张三";
        student.age=20;
        System.out.println(student.name);
        System.out.println(student.age);
        student.sleep();
        student.eat();
        student.homeWrok();
        System.out.println("===============================================");
        Teacher teacher = new Teacher();
        teacher.name = "老李";
        teacher.age=30;
        System.out.println(teacher.name);
        System.out.println(teacher.age);
        teacher.eat();
        teacher.sleep();
        teacher.teach();


    }
}

 

16.final关键字概述

1.为什么会有final
    由于继承中有一个方法重写的现象,而有时候我们不想让子类去重写父类的方法.这对这种情况java就给我们提供了一个关键字: final
2.final概述
    final关键字是最终的意思,可以修饰类,变量,成员方法。

17.final关键字修饰类,方法以及变量的特点

1.final修饰特点
    修饰类:        被修饰类不能被继承
    修饰方法:    被修饰的方法不能被重写
    修饰变量:    被修饰的变量不能被重新赋值,因为这个量其实是一个常量
2.案例演示
    final修饰特点

18.final关键字修饰局部变量

基本类型,是值不能被改变
引用类型,是地址值不能被改变

public class MyTest {
    public static final int A=100;  //公共的静态常量
    public static void main(String[] args) {
      //  final 最终的,可以修饰变量和方法以及类
        //1. final 修饰变量,此变量为常量 常量名大写。
       final int NUM=20;
        System.out.println(MyTest.A);

        //2.final 修饰方法此方法不能被重写,子类可以继承final 方法,不能重写。
        Zi zi = new Zi();
        zi.show();

        //3. final 修饰类,此类不能被继承


        final int a=20;
        //final 修饰基本数据类型的变量,指的是值不能被改变。

      final Zi zi1 = new Zi();
        //zi1=new Zi();
        //final 修饰引用数据类型的变量,指的是地址值不能被改变。


    }
}

class Fu{
    public final void show(){}
}
class Zi extends Fu{
    //public final void show()
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值