Java——抽象类和接口

简介: 抽象类是一种不能被实例化的类,至少包含一个抽象方法(无实现体的方法),常用于定义一组相关类的共同特征,并强制子类实现特定方法。抽象方法不能被 `static` 或 `final` 修饰,且必须被重写。接口则是一个完全抽象的类,用于规范类的行为。接口使用 `interface` 关键字定义,不能实例化,并且类与接口之间是实现关系。内部类是在一个类内定义的类,分为成员内部类、静态内部类、局部内部类和匿名内部类。成员内部类可被修饰符修饰,静态内部类只能访问外部类的静态成员,局部内部类定义在方法内,匿名内部类则隐藏了名字,直接通过 `new` 关键字定义并实现接口或继承类。

抽象类

抽象类是一种不能被实例化的类,其中至少包含一个抽象方法(即没有实现体的方法)。抽象类通常用于定义一组相关的类的共同特征,并强制其子类实现特定的方法。

抽象方法

将共性的方法抽取到父类之后,由于每个子类执行的内容不一样,所以在父类中不能确定具体的方法体,这样的方法就可以定义成为抽象方法

定义格式:
public abstract 返回值类型 方法名(参数列表);

public abstract void work();

抽象方法不能被static , final 修饰,抽象方法就是为了被重写的
匿名对象的方式调用抽象方法
前提已经写好了work的抽象方法和类的继承关系

public class Text {
    public static void working(Person p){
        p.work();
    }
    public static void main(String[] args) {
        
        Student student = new Student();
        working(student);
        working(student);
        //匿名对象的方式调用方法
        working(new Student());
        working(new Student());
    }
}

普通的调用一个对象可以多次调用方法,匿名对象的方法调用每次调用都创建了对象

抽象类

如果一个类中有抽象方法,这个类就必须声明成抽象类

定义格式:
public abstract class 类名{}

public abstract class Person {
    public abstract void work();
}

注意:
1.抽象类不能实例化对象

2.抽象类中不一定有抽象方法,有抽象方法的类一定要设置成抽象类
3.抽象类可以有构造方法
4.抽象方法的子类要不重写父类的抽象方法,要不自身也是抽象类
那可能就有疑问了,既然抽象类不能实例化对象,那么要构造方法有什么用

通过向上转型的方式创建一个子类对象,通过父类的构造方法初始化子类继承的属性

接口

接口是一个完全抽象的类,用来规范类的行为,

比如有以上的继承关系,根据fish,student,teacher的共性来定义swim的行为,此时swim并不存在任何继承的关系,就可以把它定义为一个接口去被fish,student,teacher实现

接口的实现

1.接口使用关键字interface来定义
public interface 接口名{ }
2.接口不能实例化
3.接口和类是实现关系,通过关键字
implements表示
public class 类名 implements 接口名{}
4.接口的子类要么重写接口中所有的抽象方法,要么是抽象类
5.接口和类之间可以单实现,也可以多实现,接口的出现,弥补了Java不能多继承的缺陷

接口中的成员变量默认是public static final修饰的,用static修饰调用方便,接口名.方法名调用,用》>final修饰,不可更改,所以也说明了接口中定义的是一种规范
接口中的方法是public abstract修饰的
所以说接口的实现类在重写接口的方法时只能用public修饰
接口中不能有代码块和构造方法

如果方法是被default和static修饰的可以有具体的实现

以下就是接口的实现例子:

实现多个接口

以下分别定义好了两个接口

public interface IRun {
    void run();
}
public interface ISwim {
    void swim();
}

Dog类继承Animal类之后再实现两个接口

public class Dog extends Animal implements ISwim,IRun{
     public Dog() {
        super();
    }
    public Dog(String name, int age) {
        super(name, age);
    }
    @Override
    public void run(){
        System.out.println(this.name + "在跑");
    }
    @Override
    public void swim() {
        System.out.println(this.name + "在游泳");
    }
}

测试类实现:

public class Text {
    public static void main(String[] args) {
        running(new Dog("大黄", 3));
        swimming(new Dog("大黄", 3));
    }
    public static void running(IRun iRun){
        iRun.run();
    }
    public static void swimming(ISwim iSwim){
        iSwim.swim();
    }
}

接口与接口的关系

接口和接口之间是继承的关系,可以单继承也可以多继承

interface Ia{
    void showA();
}
interface Ib{
    void showB();
}
interface Ic extends Ia,Ib{
    void showC();
}
class D implements Ic{
    @Override
    public void showA() {
        
    }
    @Override
    public void showB() {
    }
    @Override
    public void showC() {
    }
}
public class Demo {
}

Ic继承了Ia,Ib,同时类D实现了Ic接口,此时Ia,Ib,Ic的方法都需要重写,所以,接口间的继承相当于把多个接口结合在一起

内部类

内部类其实就是一个类里边再定义一个类,表示是外部类的一部分
访问特点:
内部类可以直接访问外部类的成员,包括私有的
外部类要访问内部类的成员,必须创建对象

public class Car {
    String carName;
    String carColor;
    //Car this可写可不写
    public void show(Car this){
        System.out.println(this.carName);
        //创建内部类的对象才能访问内部类成员变量
        Engine e = new Engine();
        System.out.println(e.engineName);
    }
    class Engine{
        String engineName;
        int engineAge;
        public void show(){
            //可以直接调用外部类成员变量
            System.out.println(engineName);
            System.out.println(engineAge);
        }
    }
}

成员内部类

成员内部类也就是写在成员位置的,属于外部类的成员,上面的代码就是一个成员内部类的例子,成员内部类可以被一些修饰符所修饰,例如private,默认,protexted,public,static等
在JDK16版本之前,成员内部类里面不能定义静态变量,JDK16开始才可以定义静态变量
获取内部类成员对象
1.在外部类中编写方法,对外提供内部类对象
2.直接创建格式:
外部类名.内部类名 对象名 = 外部对象.内部对象;

遇到重名时的访问:

public class Outer {
    String name;
    private int a = 10;
    class Inner{
        private int a = 20;
        public void show(){
            int a = 30;
            System.out.println(Outer.this.a);//10
            System.out.println(this.a);//20
            System.out.println(a);//30
        }
    }
    //外部类编写获取内部类的方法
    public Inner getInnerInstance(){
        return new Inner();
    }
}

静态内部类

静态内部类其实就是在成员内部类的基础上加上static修饰之后,就变为了静态内部类,静态内部类只能访问外部类中的静态变量和静态方法,如果要想访问非静态的需要创建对象
1.创建静态内部类对象的格式:外部类名.内部类名 对象名 = new 外部类名.内部类名();
2.调用非静态方法的格式:先创建对象,用对象调用;
3.调用静态方法的格式:外部类名.内部类名.方法名

修改为以下代码

public class Souter {
    int a;
    static int b;
    static class Inner{
        public void print(){
            Souter s = new Souter();
            System.out.println(s.a);
            System.out.println(b);
        }
    }
}

方法的调用:

public class Souter {
    int a;
    static int b;
    static class Inner{
        public void print(){
            Souter s = new Souter();
            System.out.println(s.a);
            System.out.println(b);
        }
        public static void show(){
            System.out.println("静态内部类");
        }
    }
    public static void main(String[] args) {
        //创建对象,对象名调用非静态方法
        Souter.Inner inner = new Souter.Inner();
        inner.print();
        //类名调用静态方法
        Souter.Inner.show();
    }
}

局部内部类

将内部类定义在方法里面就是局部内部类,类似于方法里面的局部变量
1.外界无法直接使用,需要在方法内部创建对象并使用
2.该类可以直接访问外部类的成员,也可以访问方法里的局部变量

public class Oouter {
    public void show(){
        int a;
        class Inner{
            String name;
            public void method1(){
                System.out.println("局部内部类方法");
            }
            public static void method2(){
                System.out.println("局部内部类静态方法");
            }
        }
        //创建对象访问局部内部类属性及方法
        Inner inner = new Inner();
        System.out.println(inner.name);
        inner.method1();
        //类名访问静态方法
        Inner.method2();
    }
}

匿名内部类

匿名内部类就是隐藏了名字的内部类
格式:
new 类名或接口名{
重写方法
};
大括号和前面的内容要么是实现关系,要么是继承关系,整体是一个类的子类对象或接口的实现类对象,大括号里面的是类
可以写在成员位置,也可以写在局部位置

如果不适用匿名内部类的方法,还需要再创建一个类去继承Animal,再去创建对象调用方法,效率不如匿名内部类高,这也就是匿名内部类的一个重要应用场景

下面是实现接口的例子:

相关文章
|
8天前
|
JSON Java Apache
非常实用的Http应用框架,杜绝Java Http 接口对接繁琐编程
UniHttp 是一个声明式的 HTTP 接口对接框架,帮助开发者快速对接第三方 HTTP 接口。通过 @HttpApi 注解定义接口,使用 @GetHttpInterface 和 @PostHttpInterface 等注解配置请求方法和参数。支持自定义代理逻辑、全局请求参数、错误处理和连接池配置,提高代码的内聚性和可读性。
|
28天前
|
算法 Java 数据处理
从HashSet到TreeSet,Java集合框架中的Set接口及其实现类以其“不重复性”要求,彻底改变了处理唯一性数据的方式。
从HashSet到TreeSet,Java集合框架中的Set接口及其实现类以其“不重复性”要求,彻底改变了处理唯一性数据的方式。HashSet基于哈希表实现,提供高效的元素操作;TreeSet则通过红黑树实现元素的自然排序,适合需要有序访问的场景。本文通过示例代码详细介绍了两者的特性和应用场景。
37 6
|
28天前
|
存储 Java 数据处理
Java Set接口凭借其独特的“不重复”特性,在集合框架中占据重要地位
【10月更文挑战第16天】Java Set接口凭借其独特的“不重复”特性,在集合框架中占据重要地位。本文通过快速去重和高效查找两个案例,展示了Set如何简化数据处理流程,提升代码效率。使用HashSet可轻松实现数据去重,而contains方法则提供了快速查找的功能,彰显了Set在处理大量数据时的优势。
32 2
|
9天前
|
Java
java线程接口
Thread的构造方法创建对象的时候传入了Runnable接口的对象 ,Runnable接口对象重写run方法相当于指定线程任务,创建线程的时候绑定了该线程对象要干的任务。 Runnable的对象称之为:线程任务对象 不是线程对象 必须要交给Thread线程对象。 通过Thread的构造方法, 就可以把任务对象Runnable,绑定到Thread对象中, 将来执行start方法,就会自动执行Runable实现类对象中的run里面的内容。
23 1
|
14天前
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
41 4
|
20天前
|
安全 Java
在 Java 中使用实现 Runnable 接口的方式创建线程
【10月更文挑战第22天】通过以上内容的介绍,相信你已经对在 Java 中如何使用实现 Runnable 接口的方式创建线程有了更深入的了解。在实际应用中,需要根据具体的需求和场景,合理选择线程创建方式,并注意线程安全、同步、通信等相关问题,以确保程序的正确性和稳定性。
|
19天前
|
Java
Java基础(13)抽象类、接口
本文介绍了Java面向对象编程中的抽象类和接口两个核心概念。抽象类不能被实例化,通常用于定义子类的通用方法和属性;接口则是完全抽象的类,允许声明一组方法但不实现它们。文章通过代码示例详细解析了抽象类和接口的定义及实现,并讨论了它们的区别和使用场景。
|
19天前
|
Java 测试技术 API
Java零基础-接口详解
【10月更文挑战第19天】Java零基础教学篇,手把手实践教学!
18 1
|
21天前
|
Java 测试技术 开发者
Java零基础-抽象类详解
【10月更文挑战第17天】Java零基础教学篇,手把手实践教学!
17 2
|
23天前
|
Java 测试技术 开发者
Java零基础-抽象类详解
【10月更文挑战第15天】Java零基础教学篇,手把手实践教学!
19 2