Java——面向对象进阶(三)

在这里插入图片描述

前言:
抽象类,接口,内部类


文章目录

  • 一、抽象类
    • 1.1 抽象方法
    • 1.2 抽象类
    • 1.3 抽象类的使用
  • 二、 接口
    • 2.1 接口的定义和实现
    • 2.2 default 关键字
    • 2.3 实现接口时遇到的问题
  • 三、内部类
    • 3.1 成员内部类
    • 3.2 静态内部类
    • 3.3 成员内部类
    • 3.4 匿名内部类(最常用)

一、抽象类

在Java中,抽象类是一种不能被实例化的类,它通常用于定义一些共用的方法和字段,在该类中不知道具体的实现,但还是想让子类必须去实现这些方法,所以将成员方法定义成抽象方法,该类定义成抽象类。abstract 用于修饰方法方法和类,修饰的方法是抽象方法,修饰的类是抽象类。

1.1 抽象方法

使用abstract 关键字修饰方法,该方法就成了抽象方法,抽象方法只包含一个方法名,而没有方法体。

定义格式:
修饰符 abstract 返回值类型 方法名 (参数列表);示例:
public abstract void run();

1.2 抽象类

使用abstract 关键字修饰类,

形式:
abstract class 类名字 { }示例:
public abstract class Animal {public abstract void run()}

!注意:抽象类不一定有抽象方法,但有抽象方法的类一定要定义成抽象类

1.3 抽象类的使用

  1. 继承抽象类:

    • 如果一个类继承了一个抽象类,并且这个类不是抽象类,那么它必须实现所有从抽象类继承的抽象方法。
    • 如果一个类继承了一个抽象类,但没有实现所有的抽象方法,那么这个类必须声明为抽象类。
  2. 抽象类的继承链:

    • 一个抽象类可以继承另一个抽象类,并且可以选择性地实现部分或全部继承的抽象方法。
    • 最终,必须有一个具体(非抽象)的子类实现所有抽象方法。

示例代码:

  1. 子类实现所有抽象方法

    abstract class Animal {abstract void makeSound();abstract void eat();
    }class Dog extends Animal {@Overridevoid makeSound() {System.out.println("Woof!");}@Overridevoid eat() {System.out.println("Dog is eating.");}
    }public class Main {public static void main(String[] args) {Animal myDog = new Dog();myDog.makeSound();myDog.eat();}
    }
    

    Dog类继承了Animal类,并且实现了所有的抽象方法。

  2. 子类没有实现所有抽象方法,必须声明为抽象类

    abstract class Animal {abstract void makeSound();abstract void eat();
    }abstract class Canine extends Animal {@Overridevoid makeSound() {System.out.println("Howl!");}// `eat` 方法没有实现,所以 Canine 仍然是抽象类
    }class Dog extends Canine {@Overridevoid eat() {System.out.println("Dog is eating.");}
    }public class Main {public static void main(String[] args) {Animal myDog = new Dog();myDog.makeSound();myDog.eat();}
    }
    

    Canine类实现了makeSound方法,但没有实现eat方法,因此Canine必须声明为抽象类。Dog类继承了Canine并实现了eat方法,使得Dog成为具体类,可以被实例化。


二、 接口

接口(Interface)在Java中是一个重要的概念,用于定义一组方法的规范,而不提供任何实现。接口主要用于定义类之间的契约和规范,确保实现这些接口的类遵循相同的方法签名。

2.1 接口的定义和实现

接口的定义:
接口使用interface关键字定义,接口中的所有方法默认是publicabstract,所有成员变量默认是public static final。以下是一个简单的接口示例:

public interface Animal {void makeSound();void eat();
}

实现接口
类通过implements关键字实现接口,一个类可以实现多个接口,这与抽象类的单继承限制不同。以下是实现Animal接口的两个类:

public class Dog implements Animal {@Overridepublic void makeSound() {System.out.println("Woof!");}@Overridepublic void eat() {System.out.println("Dog is eating.");}
}public class Cat implements Animal {@Overridepublic void makeSound() {System.out.println("Meow!");}@Overridepublic void eat() {System.out.println("Cat is eating.");}
}

类实现接口的要求:

  1. 必须重写实现的全部接口中所有抽象方法。
  2. 如果一个类实现了接口,但是没有重写完全部接口的全部抽象方法,这个类也必须定义成抽象类。

如果一个类实现了多个接口,而这些接口有相同的抽象方法,那么这个类只需要实现这个抽象方法一次即可,因为Java不允许同一个类中出现重复的方法签名。这种情况下,类只需要提供一次方法的实现,而不需要重复实现多次。


2.2 default 关键字

default 关键字主要用于接口中的默认方法(default methods),这是从Java 8开始引入的特性。默认方法允许在接口中为方法提供默认的实现,这样可以在不破坏现有实现的情况下,向接口添加新的方法。

默认方法(Default Methods)定义方式:

public interface InterfaceName {// 抽象方法void regularMethod();// 默认方法default void defaultMethod() {// 默认实现System.out.println("This is a default method.");}
}

在上面的例子中,defaultMethod 就是一个默认方法。默认方法使用 default 关键字修饰,它提供了一个默认的实现。所有实现了这个接口的类都会继承这个默认方法,如果需要的话,可以在实现类中重写默认方法。

示例:

public interface Vehicle {void start();default void honk() {System.out.println("Vehicle is honking.");}
}public class Car implements Vehicle {@Overridepublic void start() {System.out.println("Car is starting.");}// Car可以选择不重写honk方法,使用默认实现
}public class Bike implements Vehicle {@Overridepublic void start() {System.out.println("Bike is starting.");}@Overridepublic void honk() {System.out.println("Bike is honking."); // Bike提供了自己的honk方法实现}
}

当需要扩展一个接口,但又不能破坏所有实现这个接口的类时,可以使用默认方法。

注意事项:
如果一个类实现了多个接口,而这些接口有相同的默认方法,实现类必须重写这个默认方法来解决冲突。

interface A {default void hello() {System.out.println("Hello from A");}
}interface B {default void hello() {System.out.println("Hello from B");}
}class C implements A, B {@Overridepublic void hello() {A.super.hello(); // 明确指定调用接口A的默认方法//B.super.hello(); 或明确指定调用接口B的默认方法}
}

2.3 实现接口时遇到的问题

  1. 如果一个类继承了一个类并实现了一个接口,且父类和接口中有都有相同签名的方法。

    • 处理办法一:如果父类中的方法体,能满足当前业务的需求,在子类中可以不用重写。
    • 处理办法二:如果父类中的方法体,不能满足当前业务的需求,需要在子类中重写。
  2. 如果一个接口中,有多个抽象方法,但在实现类中,只需要用其中一个或部分

    • 方法一:实现类声明为抽象类
      你可以将实现类声明为抽象类,并只实现部分抽象方法,这样子类就可以根据需要选择性地实现接口的方法。

      // 定义一个接口
      interface MyInterface {void method1(); // 抽象方法1void method2(); // 抽象方法2void method3(); // 抽象方法3
      }// 实现类声明为抽象类,并只实现部分方法
      abstract class MyAbstractClass implements MyInterface {@Overridepublic void method1() {System.out.println("Implemented method1");}// 不需要实现 method2 和 method3
      }// 子类继承抽象类,选择性实现部分方法
      class MyClass extends MyAbstractClass {@Overridepublic void method2() {System.out.println("Implemented method2");}
      }
      
    • 方法二:使用适配器模式(Adapter Pattern)
      如果不想使用抽象类,可以使用适配器模式,它允许提供接口的默认实现,然后子类可以选择性地覆盖它们。

      // 定义一个接口
      interface MyInterface {void method1(); // 抽象方法1void method2(); // 抽象方法2void method3(); // 抽象方法3
      }// 创建适配器类,提供接口的默认实现
      abstract class MyInterfaceAdapter implements MyInterface {@Overridepublic void method1() {// 默认实现}@Overridepublic void method2() {// 默认实现}@Overridepublic void method3() {// 默认实现}
      }// 子类选择性地覆盖需要的方法
      class MyClass extends MyInterfaceAdapter {@Overridepublic void method2() {System.out.println("Implemented method2");}
      }
      

三、内部类

内部类(Inner Class)是定义在另一个类内部的类。Java 提供了多种内部类,包括成员内部类、局部内部类、匿名内部类和静态内部类。

按定义的位置来分:

  1. 成员内部内,类定义在了成员位置 (类中方法外称为成员位置,无static修饰的内部类)
  2. 静态内部类,类定义在了成员位置 (类中方法外称为成员位置,有static修饰的内部类)
  3. 局部内部类,类定义在方法内
  4. 匿名内部类,没有名字的内部类,可以在方法中,也可以在类中方法外。

3.1 成员内部类

定义格式:

class OuterClass {class InnerClass {// Inner class members}
}

内部类的使用格式

 外部类.内部类。 // 访问内部类的类型都是用 外部类.内部类

创建内部类实例:

  • 方式一:外部直接创建成员内部类的对象

    OuterClass outer = new OuterClass();
    OuterClass.InnerClass inner = outer.new InnerClass();
    //或
    OuterClass.InnerClass inner = new OuterClass().new InnerClass();
    
  • 方法二:在外部类中定义一个方法提供内部类的对象

    public class Outer {private class Inner{}public Inner getInstance(){return new Inner();}
    }public class Test {public static void main(String[] args) {Outer outer = new Outer();Inner inner = outer.getInstance();}
    }
    

成员内部类的细节

  1. 成员内部类可以被一些修饰符所修饰,比如: private,默认,protected,public,static等

  2. 成员内部类可以访问外部类的所有成员,包括私有成员。这是因为成员内部类持有一个对外部类实例的引用Outer.this

    public class Outer {private String outerField = "Outer field";class Inner {void display() {System.out.println("Inner class accessing: " + outerField);System.out.println("Inner class accessing through Outer.this: " + Outer.this.outerField);}}public static void main(String[] args) {Outer outer = new Outer(); // 创建外部类对象Outer.Inner inner = outer.new Inner(); // 创建内部类对象inner.display(); // 调用内部类的方法}
    }//输出:
    Inner class accessing: Outer field
    Inner class accessing through Outer.this: Outer field
    

    内存图示例:内存图示例

  3. 在 JDK 8 之前,局部变量必须显式声明为 final 才能被成员内部类。在 JDK 8 及之后,只要局部变量在初始化后没有被修改,它们就被隐式地视为 final(即有效 final),不再需要显式声明为 final。


3.2 静态内部类

定义格式:

class OuterClass {static class StaticInnerClass {// 静态内部类的成员}
}

创建实例:

OuterClass.StaticInnerClass innerObject = new OuterClass.StaticInnerClass();

静态内部类的特点

  1. 静态内部类可以直接访问外部类的静态成员,但不能访问外部类的非静态成员,如果要访问需要创建外部类的对象,用对象调用。

    public class Outer {private static String staticOuterField = "Static Outer Field";private String nonStaticOuterField = "Non-Static Outer Field";static class StaticInner {void display() {// 静态内部类可以访问外部类的静态成员System.out.println("Static Inner Class accessing static outer field: " + staticOuterField);// 但不能直接访问外部类的非静态成员// System.out.println("Static Inner Class accessing non-static outer field: " + nonStaticOuterField); // 编译错误}}public static void main(String[] args) {// 创建静态内部类的实例StaticInner inner = new StaticInner();inner.display(); // Output: Static Inner Class accessing static outer field: Static Outer Field// 如果要访问外部类的非静态成员,需要创建外部类的对象Outer outer = new Outer();System.out.println("Access non-static outer field through outer object: " + outer.nonStaticOuterField);}
    }
    
  2. 无需外部类实例: 可以直接创建静态内部类的实例,而不需要外部类的实例。

  3. :静态内部类中没有隐含的 Outer.this


3.3 成员内部类

局部内部类是定义在方法或作用域内部的内部类。

特点

  • 作用域限制: 局部内部类的作用域仅限于定义它的方法或代码块中,不能在外部访问。
  • 访问权限: 局部内部类可以访问外部类的所有成员,包括私有成员。
  • 访问局部变量: 局部内部类可以访问方法内的 final 局部变量(JDK 8+版本可以访问非 final 的局部变量,但其值不能在局部类中被修改)。
  • 生命周期: 局部内部类的生命周期仅限于方法调用,方法结束后局部内部类实例也会被销毁。

示例

public class Outer {private int outerField = 10;public void methodWithLocalInnerClass() {final int localVar = 20; // JDK 8+ 可以不用final修饰,但不可变class LocalInner {void display() {System.out.println("Outer field: " + outerField);System.out.println("Local variable: " + localVar);}}// 创建局部内部类的实例并调用方法LocalInner inner = new LocalInner();inner.display();}public static void main(String[] args) {Outer outer = new Outer();outer.methodWithLocalInnerClass();}
}

3.4 匿名内部类(最常用)

匿名内部类是一种没有显式声明类名的内部类,它在创建对象的同时定义类的实现。匿名内部类通常用于创建只需使用一次且没有额外的逻辑的类实例。

格式:

new 类名或者接口名() {重写方法;
};
这里同时包含了继承或实现,方法重写,创建对象
从new关键字到";"整个整体是匿名内部类的实例,"{}"内的是匿名内部类,没有类名

使用:
以接口为例,匿名内部类的使用,代码如下:

interface Swim {public abstract void swimming();
}public class Test {public static void main(String[] args) {// 接口 变量 = new 实现类(); // 多态,走子类的重写方法Swim s2 = new Swim() {@Overridepublic void swimming() {System.out.println("蛙泳...");}};s2.swimming();}
}


如果你喜欢这篇文章,点赞👍+评论+关注⭐️哦!
欢迎大家提出疑问,以及不同的见解。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/352999.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

6月17(信息差)

1.马斯克最新预测:未来不再需要手机 将被脑机芯片替代 当地时间6月17日,马斯克高仿号“Not Elon Musk”发帖称:“你会在你的大脑上安装一个Neuralink接口,让你通过思考来控制你的新X手机吗?”对此,马斯克本…

C++ 62 之 冒泡排序

#include <iostream> // #include <string> #include <cstring>using namespace std;// 冒泡排序:函数模板 template<typename T> void mySort(T arr[], int len){ // size参数是数组的个数&#xff0c;一定是int型的for (size_t i 0; i < len -1;…

88. 合并两个有序数组(简单)

88. 合并两个有序数组 1. 题目描述2.详细题解3.代码实现3.1 Python3.2 Java 1. 题目描述 题目中转&#xff1a;88. 合并两个有序数组 2.详细题解 两个数组均有序&#xff08;非递减&#xff09;&#xff0c;要求合并两个数组&#xff0c;直观的思路&#xff0c;借助第三个数…

利用AI云防护实现高效负载均衡

在当今高度数字化的世界里&#xff0c;保证网站和应用的高可用性和响应速度对企业的业务连续性和用户体验至关重要。传统的负载均衡技术虽然能够分发流量&#xff0c;但在面对突发流量、DDoS攻击或资源动态调整时往往力不从心。本文将探讨如何借助AI云防护服务&#xff0c;不仅…

超图论文细品——2019年AAAI《Hypergraph Neural Networks》

我是“导航” 1 摘要1.1 简介1.2 问题描述 2 超图2.1 图和超图对比 参考 1 摘要 1.1 简介 文章提出了一种名为超图神经网络的框架&#xff0c;用于高维数据的表示学习。 该方法英文称呼为 Hypergraph Neural Networks&#xff0c;简写为 HGNN。 1.2 问题描述 传统的 GNN 是…

大模型精调:实现高效迁移学习的艺术

在人工智能领域&#xff0c;大型预训练模型&#xff08;以下简称“大模型”&#xff09;已经取得了令人瞩目的成果。这些模型通过在海量数据上进行预训练&#xff0c;能够捕捉到丰富的特征信息&#xff0c;为各种下游任务提供强大的支持。然而&#xff0c;如何将这些大模型应用…

graalvm编译springboot3 native应用

云原生时代容器先行&#xff0c;为了更好的拥抱云原生&#xff0c;spring boot3之后&#xff0c;推出了graalvm编译boot项目&#xff0c;利用jvm的AOT&#xff08; Ahead Of Time &#xff09;运行前编译技术&#xff0c;可以将java源码直接构建成机器码二进制的文件&#xff0…

(资料收藏)王阳明传《知行合一》共74讲,王阳明知行合一音频讲解资料

今天给大家带来的不是软件&#xff0c;而是一份精神食粮——《知行合一》的教程福利。这可不是一般的教程&#xff0c;它关乎心灵&#xff0c;关乎智慧&#xff0c;关乎我们如何在纷繁复杂的世界中找到自己的位置。 咱们得聊聊王阳明&#xff0c;这位明代的大儒&#xff0c;他…

JAVAEE之网络原理(2)_传输控制协议(TCP)、概念、格式、确认应答及超时重传机制

前言 在上一节中&#xff0c;我们介绍了 UDP (用户数据报) 的相关知识&#xff0c;在这一节中我们将继续介绍传输层中另一种更为重要的协议。 一、什么是TCP协议&#xff1f; 1.1 TCP 基本概念 TCP协议全称&#xff1a;传输控制协议&#xff08;TCP&#xff0c;Transmission C…

Prometheus配置文件与核心功能

Prometheus配置文件与核心功能 环境部署完成以后&#xff0c;我们就需要对产品进行稳定性监控。在知道怎么监控收集数据之前&#xff0c;我们需要知道prometheus的配置文件和核心功能 全局配置文件 首先我们从针对prometheus的全局文件进行说起&#xff0c;首先进入promethe…

springboot宠物医院信息管理系统-计算机毕业设计源码04164

摘 要 现如今在中国&#xff0c;随着人民生活质量的逐渐提高&#xff0c;以及人民群众消费能力的日渐增长&#xff0c;各种各样的家养小动物&#xff0c;已经逐渐成为人类越来越亲密的生活伴侣。并且&#xff0c;现如今社会竞争及其激烈&#xff0c;人们的生活节奏越发急促、紧…

【免费API推荐】:满足您的开发需求,加速项目上线

免费API助力项目开发&#xff0c;为开发者们提供了强大的支持和工具。这些API提供了各种功能和服务&#xff0c;能够帮助项目开发者们快速构建出高质量的应用。无论是地理位置服务、支付接口、社交媒体集成还是图像识别&#xff0c;这些免费API为项目开发者们提供了丰富的功能和…

图解ZGC

ZGC&#xff08;Z Garbage Collector&#xff09; 是一款性能比 G1 更加优秀的垃圾收集器。ZGC 第一次出现是在 JDK 11 中以实验性的特性引入&#xff0c;这也是 JDK 11 中最大的亮点。在 JDK 15 中 ZGC 不再是实验功能&#xff0c;可以正式投入生产使用了&#xff0c;使用 –X…

Axure8.0实例|数量编辑器

Axure8.0实例&#xff5c;数量编辑器 一、元件准备 1、添加三个矩形框&#xff0c;分别取名为“减少数量”、“数量背景”、“增加数量”。“减少数量”矩形框中输入“-”号&#xff0c;“增加数量”矩形框中输入“”号&#xff0c;待用&#xff1b; 2、添加一个文本框&#…

高等数学笔记(一):映射与函数

一、映射 1.1 映射的概念 存在一个法则 f &#xff0c;使得对 X 中每个元素 x &#xff0c;在 Y 中有唯一确定的元素 y 与之对应&#xff08;X、Y 非空集&#xff09; 称 f 为从 X 到 Y 的映射&#xff0c;如图所示 其中 y 称为元素 x&#xff08;在映射 f 下&#xff09;的…

通过LotusScript中的NotesDateTime类来进行时间计算

大家好&#xff0c;才是真的好。 今天我们介绍的是时间日期处理&#xff0c;其实以前也讲过&#xff0c;主要是通过LotusScript中的NotesDateTime类来进行时间计算。 但是这里也存在一个问题&#xff1a;就是时间日期类的比较只能计算出秒&#xff0c;不能计算出毫秒。 毕竟…

手写一个JSON可视化工具

前言 JSON 平时大家都会用到&#xff0c;都不陌生&#xff0c;今天就一起来实现一个 JSON 的可视化工具。 大概长成下面的样子&#xff1a; 树展示 相比于现有的一些 JSON 格式化工具&#xff0c;我们今天制作的这个小工具会把 JSON 转为树去表示。其中&#xff1a; 橙色标…

Cloneable接口和对象的克隆——浅拷贝和深拷贝

用Clonable接口实现对象的克隆——浅拷贝和深拷贝 1. 浅拷贝2. 深拷贝 在Object类中提供了clone方法&#xff0c;用来是实现对象的克隆&#xff01; 1. 浅拷贝 我们首先来尝试用clone方法去克隆一个Person对象 public class Person {public String name;public int age;publi…

数据驱动决策:工单统计工具如何赋能企业精准运营

在当今这个数字化飞速发展的时代&#xff0c;企业对于内部运营效率的追求已经达到了前所未有的高度。你是否曾为了繁杂的工单统计管理而头疼不已&#xff1f;是否曾因为无法准确进行工单统计数据而错失商机&#xff1f;今天&#xff0c;我将向你展示一款革命性的工单统计工具&a…

企业为何需要搭建线上虚拟品牌展厅?

在数字化时代&#xff0c;线上虚拟品牌展厅已成为企业不可或缺的一部分。以下是构建线上虚拟品牌展厅的4大关键理由&#xff1a; 1、迎合在线购物趋势 随着移动互联网的飞速发展和普及&#xff0c;消费者越来越倾向于在线购物。一个线上虚拟品牌展厅能够完美地满足这一需求&am…