当前位置: 首页 > article >正文

面向对象进阶(下)(JAVA笔记第二十五期)

p.s.这是萌新自己自学总结的笔记,如果想学习得更透彻的话还是请去看大佬的讲解

目录

  • 抽象方法和抽象类
    • 抽象方法定义格式
    • 抽象类定义格式
    • 抽象方法和抽象类注意事项
  • 接口
    • 接口的定义
    • 接口中成员变量的特点
    • 接口中没有构造方法
    • 接口中成员方法的特点
    • 在接口中定义具有方法体的方法
      • 在接口中新增默认方法(解决接口升级的问题)
      • 在接口中新增静态方法(不能被重写)
      • 在接口中新增私有方法
    • 接口与类之间的关系
      • 类与类之间的关系
      • 类与接口之间的关系
      • 接口与接口之间的关系
    • 接口的多态
  • 设计模式
    • 适配器设计模式
  • 内部类
    • 成员内部类
      • 获取成员内部类的对象
      • 成员内部类如何获取外部类的成员变量
      • 特殊的成员内部类:静态内部类
        • 创建静态内部类的对象
        • 调用非静态方法的格式:先创建对象,用对象调用
        • 调用静态方法的格式:外部类名.内部类名.方法名();
    • 局部内部类
    • 匿名内部类
      • 使用场景
  • 重写toString()方法

抽象方法和抽象类

抽象方法:如果一个方法抽取到父类中,不确定方法体,那就不要写方法体,再用abstract关键字修饰,那么这个方法就叫做抽象方法。
作用:父类有抽象方法,子类必须强制重写抽象方法,否则子类代码直接报错
而抽象方法所在的类就叫抽象类

抽象方法:将共性(同名)的行为(方法)抽取到父类之后,由于每一个子类执行的内容不一样,所以在父类当中无法确定具体的方法体,该方法即可定义为抽象方法
抽象类:如果一个类里面存在抽象方法,,那么该类则必须声明为抽象类

抽象方法定义格式

public abstract 返回值类型 方法名(参数列表);

抽象类定义格式

public abstract class 类名{}

抽象方法和抽象类注意事项

抽象类不能实例化(即不能创建对象)
抽象类中不一定有抽象方法,有抽象方法的类一定是抽象类
抽象类可以有构造方法
抽象类的子类要么重写抽象类中的所有抽象方法,要么是抽象类
举例

public abstract class Person {

    private String name;
    private int age;

    public abstract void work();


    public Person() {}
    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
    //构造方法的作用:当创建子类对象时,给属性进行赋值

    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;
    }
}
//=============================================
class  Student extends Person{

    @Override
    public void work() {
        System.out.println("学生的工作就是学习");
    }
    //重写抽象类的所有抽象方法

    public Student() {}

    public Student(String name, int age) {
        super(name, age);
    }

}
//=============================================
class Test{
    public static void main(String[] args) {
        Student student = new Student("张三",18);
        System.out.println(student.getName()+", "+student.getAge());
    }
}

接口

接口是一个抽象类型,是抽象方法的集合,一个类通过继承接口的方式,从而来继承接口的抽象方法。(可以看作是一种规则)
.
接口的子类也叫实现类,其要么实现重写接口当中所有的抽象方法,要么其为一个抽象类
.
接口并不是类,编写接口的方式和类很相似,但是它们属于不同的概念。类描述对象的属性和方法。接口则包含类要实现的方法。
.
除非实现接口的类是抽象类,否则该类要定义接口中的所有方法。(具体原因参考抽象方法与抽象类)
.
接口无法被实例化,即不能创建接口的对象,但是可以被实现。一个实现接口的类,必须实现接口内所描述的所有方法,否则就必须声明为抽象类。
.

接口的定义

接口用interface来定义
格式

public interface 接口名{}

接口与类之间是实现关系,通过implements关键字表示
格式

public class 类名  implements 接口名{}

注意:一个类可以实现单个接口(单实现),也可以实现多个接口(多实现)
举例

public class 类名  implements 接口名1,接口名2{}

注意:实现类可以在继承一个类的同时实现多个接口
举例

public class 类名 extends 父类 implements 接口名1,接口名2{}

接口中成员变量的特点

只能是常量
默认修饰符:public static final (即就算你不加,系统也会自动给你加上)
调用方式:接口名.常量名

接口中没有构造方法

接口中成员方法的特点

JDK8以后接口中可以定义抽象方法、有方法体的方法(默认方法、静态方法)、私有方法。
其中抽象方法也有默认修饰符:pubilc abstract



在接口中定义具有方法体的方法

在接口中新增默认方法(解决接口升级的问题)

由于只要添加接口中的抽象方法时,其对应的实现类也要添加对应的抽象方法,否则实现类就会报错

所以JDK8以后允许在接口中定义默认方法,需要用关键字default修饰
定义格式为

public default 返回值类型 方法名(参数列表){   }

注意:默认方法不是抽象方法,所以不强制重写。但是如果被重写,重写的时候要去掉default关键字
public可省略,但default关键字不可以省略
如果实现了多个接口,多个接口中存在同名的默认方法,则子类就必须对该方法进行重写(因为系统不知道你以后调用的是哪个接口中的默认方法,干脆让你重写得了)

在接口中新增静态方法(不能被重写)

JDK8以后允许在接口中定义静态方法,需要用关键字static修饰
定义格式为

public static 返回值类型 方法名(参数列表){   }

注意:静态方法只能通过接口名调用,不能通过实现类名或对象名调用
public可省略,但static关键字不可以省略
接口和实现类中有两个重名的静态方法,这不叫重写,只是恰好同名而已

在接口中新增私有方法

由于在接口中有一些方法的代码重复了,但我又不想让这些代码被别的接口、类使用,于是我们便可以将这些代码写在一个方法里并将其私有化,使其只能在本接口中使用。
JDK9以后允许在接口中定义私有方法,需要用关键字private修饰,且不能加用default修饰
.
给默认方法调用的叫普通私有方法

private 返回值类型 方法名(参数列表){   }

给静态方法调用的叫静态私有方法

private static 返回值类型 方法名(参数列表){   }

.

接口与类之间的关系

类与类之间的关系

继承关系。只能单继承,不能多继承,但是可以多层继承

类与接口之间的关系

实现关系。可以单实现,也能多实现,还可以再继承一个类的时候同时实现多个接口
如果实现类实现了多个接口。那么就要把所有接口的所有抽象方法进行重写
如果有多个重名的抽象方法,这个抽象方法只需要重写一次就可以了

接口与接口之间的关系

继承关系。能单继承,也能多继承
如果实现类实现的是底层的子接口,那么实现类要把该接口以及其所有父类接口里面的抽象方法重写

接口的多态

当一个方法的参数是接口时,可以传递接口所有实现类的对象,这种方式称为接口多态
定义格式:接口类型 变量 = new 实现类对象();
其遵守编译看左边、运行看右边的原则

设计模式

设计模式(Design Pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结(说白了就是一套公式)
使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码的可用性、程序的重用性

适配器设计模式

解决接口和实现类之间的矛盾问题:接口中有n个抽象方法,但实现类只想使用其中一个。如果重写所有抽象方法会提高后期代码的维护难度
适配器设计模式解决方案:定义一个中间类(即适配器)(xxxAdapter)(一般是抽象类,目的是为了让外界无法创建它的对象)实现接口并且对接口中的抽象方法进行空实现(即重写接口中的抽象方法但不写方法体)
这时再让原本的实现类不再实现接口,而是继承中间类。这样需要使用哪个方法,就重写那个方法就行了

内部类

内部类就是在一个类里面再定义一个类
我们可以在外部其他类中创建内部类的对象并调用内部类的方法

意义:假如创建一个汽车的JavaBean类型,属性有发动机的年龄、最大功率、汽车颜色、汽车品牌。
这时我们便可以将发动机的属性定义在一个内部类里面,从而提高了代码的可读性

内部类一般来说遵循以下规则:
内部类中的事物是外部类中的一部分;内部类单独出现没有任何意义(比如汽车的发动机、人的大脑)

内部类的访问特点:
内部类可以直接访问外部类的成员,包括私有;外部类要访问内部类的成员,必须创建对象

内部类分为四种:成员内部类、静态内部类、局部内部类、匿名内部类

成员内部类

写在成员位置(类中方法外)的,属于外部类的成员之一(比如汽车的发动机)

成员内部类可以被一些修饰符所修饰,比如private、默认、protected、public等
用private修饰则外部其他类不能直接创建内部类的对象,只能在外部类里面创建内部类的对象
默认的话则内部类只能在本包当中使用,在其他包里面不起作用
用protected修饰则在其他包的子类中也可以使用
用public修饰则所有的地方都可以创建成员内部类的对象
.
JDK16以后,在成员内部类里面可以定义静态变量

获取成员内部类的对象

方法一:在外部类中编写方法,对外提供内部类的对象(成员内部类被private修饰的时候)
举例:

   public Inner getInstance(){
       return new Inner();
 } 

方法二:直接创建格式:外部类名.内部类名 对象名 = 外部类对象.内部类对象
举例:Outer.Inner inner = new Outer().new Inner();

成员内部类如何获取外部类的成员变量

若没有重名,就直接调用
若重名,则外部类名.this.变量名

举例

public class Outer {
    private int a = 10;
    class Inner{
        private int a = 20;

        public void show(){
            int a = 30;

            System.out.println(Outer.this.a);//a==10
            System.out.println(this.a);//a==20
            System.out.println(a);//a==30
        }
    }
}

内存原理分析

特殊的成员内部类:静态内部类

如果成员内部类前面用static修饰,则这个成员内部类就叫做静态内部类

静态内部类只能访问外部类中的静态变量和静态方法,如果想要访问非静态的需要创建对象,再用外部类的对象进行调用
静态内部类的实例不依赖于外部类的实例。

创建静态内部类的对象

创建格式:外部类名.内部类名 对象名 = 外部类对象.内部类对象
举例:Outer.Inner inner = new Outer.Inner();
只要是静态的东西,都可以用类名.来获取

调用非静态方法的格式:先创建对象,用对象调用
调用静态方法的格式:外部类名.内部类名.方法名();

局部内部类

将内部类定义在方法里面就叫做局部内部类,类似于方法里面的局部变量

外界无法直接使用,需要在方法内创建对象并使用
该类可以直接访问外部类的成员,也可以访问方法内的局部变量

举例

public class Outer1 {
    int a = 10;

    public void show(){
        int b = 20;

        class Inner{//局部内部类
            String name;
            int age;

            public void method(){
                System.out.println("访问外部类的成员"+a);
                System.out.println("访问局部变量"+b);

                System.out.println("局部内部类的方法");
            }

            public static void method1(){
                System.out.println("局部内部类的静态方法");
            }
        }

        //在局部方法中创建内部类的对象
        Inner inner = new Inner();

        System.out.println(inner.age);
        System.out.println(inner.name);

        inner.method();
        Inner.method1();


    }
}
//=========================================================
class Test1{
    public static void main(String[] args) {
        Outer1 outer1 = new Outer1();
        outer1.show();
    }
}

匿名内部类

匿名内部类,即隐藏了名字的内部类。可以写在成员位置,也可以写在局部位置


格式

new 类名或接口名(){
    重写方法;
};

上面的整体格式为创建一个匿名内部类对象的格式
其中包含了三部分:继承/实现、方法重写、创建对象
理解匿名内部类的定义格式

使用场景

当方法的参数是接或者类的时候
以接口为例,可以传递这个接口的实现类对象
如果实现类中使用一次的话,就可以使用匿名内部类简化代码

举例

public class Test2 {
    public static void main(String[] args) {
        

        method( new Animal(){
            @Override
            public void eat() {
                System.out.println("狗吃骨头");
            }
        });
        //创建一个匿名内部类对象并作为参数
        
    }

    public static void method(Animal animal){
        animal.eat();
    }

}

abstract class Animal{
    public abstract void eat();
}

重写toString()方法

当我们想要直接打印一个对象来获取它的信息(即地址值)时,其实是把对象调用toString()方法的返回值给打印出来,由于String自己本身重写了toSting()方法,所以打印对象的时候就不需要重写toString()方法。
比如输出MyClass@15db9742
但是这种地址值可读性不强,并且我们也不清楚该对象的某些信息(如id等)
由于打印对象的时候其实是把对象调用toString()方法的返回值给打印出来,因此我们便可以重写oString()方法来达到我们想要的效果(即不用通过调用get()方法来一 一获得该对象的属性并且自定义打印的格式)

比如

public class Test {
    private int id;
    private String name;

    public Test(int id, String name) {
        this.id = id;
        this.name = name;
    }

    @Override
    public String toString() {
        return "Test{id=" + id + ", name='" + name + "'}";
    }

    public static void main(String[] args) {
        Test obj = new MyClass(1, "Example");
        System.out.println(obj);
    }
}

输出将是:
Test{id=1, name='Example'}
因此在直接打印对象时,我们可以通过重写toString()方法来获取对象的某些信息而不用通过调用get()方法来一 一获得该对象的属性


http://www.kler.cn/a/370188.html

相关文章:

  • AWTK-WEB 快速入门(3) - C 语言 Http 应用程序
  • PHP语言的网络编程
  • Linux——信号量和(环形队列消费者模型)
  • node.js 文件操作
  • 产品经理面试题总结2025【其一】
  • 目标跟踪算法发展简史
  • 【STM32-HAL库】火焰传感器(STM32F407ZGT6)(附带工程下载链接)
  • spring-第十二章 GoF代理模式
  • Android Studio安装完成后,下载gradle-7.4-bin.zip出现连接超时
  • 将 Logstash 管道转换为 OpenTelemetry Collector 管道
  • JavaScript如何判断变量数据类型 - 2024最新版前端秋招面试短期突击面试题【100道】
  • SpringBoot 集成RabbitMQ 实现钉钉日报定时发送功能
  • [LeetCode] 526. 优美的排列
  • Docker | 校园网上docker pull或者docker run失败的一种解决方法
  • 探索C嘎嘎:认识string类
  • 【大数据分析与挖掘模型】matlab实现——非线性回归预测模型
  • 【计算机网络 - 基础问题】每日 3 题(五十七)
  • 《等保测评:安全与发展的双轮驱动》
  • 14 C语言中的关键字
  • Prometheus+Telegraf实现自定义监控项配置
  • RDD的常用转换算子
  • Qt实现播放器顶部、底部悬浮工具栏
  • typescript学习计划(一)--简单介绍typescript
  • VUE组件学习 | 六、v-if, v-else-if, v-else组件
  • OpenAI否认今年将发布“Orion”模型,其语音转写工具Whisper被曝存在重大缺陷|AI日报
  • 【C++】--------- 内存管理