接口和多态详解,还不快点学?

2024-09-01 04:44
文章标签 接口 详解 多态 快点

本文主要是介绍接口和多态详解,还不快点学?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、接口和多态基础知识

1. 抽象类

1.1 子类调用父类

现在有IDEA集成开发环境,可以给大家实时提醒哪个地方编译错误,但假如要大家用.txt文件编写程序呢。胡广问:现在这段代码错在了哪?

class Base {public Base(String s) {System.out.print("B");}
}public class Derived extends Base {public Derived (String s) {System.out.print("D");}public static void main(String[] args) {new Derived("C");}
}

假如父类和子类同时拥有有参构造方法,子类的构造方法必须显性地调用父类的构造方法,否则会编译错误。所以正常的写法应该是这样。

    public Derived (String s) {super(s);  System.out.print("D");}

另外大家还需要注意一点,调用父类的构造方法必须在子类构造方法的第一行,调用父类的构造方法也只能出现在子类的构造方法上,否则也会是编译报错。

1.2 子类访问父类

如下代码,一共有两处编译错误。提示:错误在Child类里,能快速找出来吗?

class Parent {public static String staticVar = "Static Variable from Parent";private static String privateStaticVar = "Private Static Variable from Parent";public static void staticMethod() {System.out.println(staticVar);}private static void privateStaticMethod() {System.out.println(privateStaticVar);}
}class Child extends Parent {public void staticMethod() {System.out.println("Static method in Child");}public void display() {System.out.println(staticVar);System.out.println(privateStaticVar);privateStaticMethod();staticMethod();
}

(1)父类的私有变量、私有方法,子类是有继承的,但是不能访问。所以Child.display()里的以下调用是编译错误的。

System.out.println(privateStaticVar);
privateStaticMethod();

(2)子类可以继承,同时也可以访问父类的static变量、方法。但父类的static方法大家需要注意,子类是不能直接覆盖的,所以以下代码会编译错误。

    public void staticMethod() {System.out.println("Static method in Child");}

正确的做法是为该方法添加一个static修饰符,代表这是子类的一个新方法。这种写法叫做方法隐藏,子类和父类中都有一个相同名称和参数的静态方法时,子类的方法将隐藏父类的方法。

    public static void staticMethod() {System.out.println("Static method in Child");}

另外如果父类的方法使用final修饰,子类也是不能覆盖的。

1.3 父类不可访问的方法

紧跟着上文代码的例子,父类的方法同样使用static修饰,子类的privateStaticMethod方法算不算覆盖父类的方法呢?有没有编译报错?

class Parent {public static String staticVar = "Static Variable from Parent";private static String privateStaticVar = "Private Static Variable from Parent";private static void privateStaticMethod() {System.out.println(privateStaticVar);}
}class Child extends Parent {public void privateStaticMethod() {System.out.println(staticVar);}
}

答案是编译正常。

父类中不可访问的方法,子类编写相同名称和参数的方法并不算覆盖。父类的方法都不能访问了,也就没有覆盖这一说法了。。。

2. 接口

2.1 访问修饰符的区别

接口和抽象类有三个方面的区别,分布是类的修饰、方法的修饰、变量的修饰。我们往下看看。

(1)类

接口使用interface修饰,而抽象类使用abstract修饰。当它们作为外部类时,只能使用public、default修饰,不能使用private修饰。

(2)方法

普通接口方法只能由public abstractdefaultstatic修饰。

抽象接口方法可以由所有修饰符修饰,除了final。

总结下,它们两者也有共同点,就是都不能使用final修饰。

(3)变量

普通接口变量只能由public static final修饰。

抽象接口变量可以由所有修饰符修饰。

2.2 静态分派

这算是一个很偏的知识点了,如下代码有三个名为getType的重载方法,它们的返回类型相同、方法名也相同,只有入参类型不同。

胡广问:程序执行结果是什么?

public class Test {public static void main(String[] args) {for(Collection<?> collection: collections) {System.out.println(getType(collection));}}public static final Collection<?>[] collections = {new HashSet<String>(), new ArrayList<String>()};public static String getType(Collection<?> collection) {return "Super:collection";}public static String getType(List<?> list) {return "Super:list";}public String getType(ArrayList<?> list) {return "Super:arrayList";}
}

胡广给大家这么一行代码:Collection<?> collection = new ArrayList<Integer>()左边Collection<?>其实是静态类型,右边的new ArrayList<Integer>()其实是动态类型。

编译器在处理重载方法时,是根据参数的静态类型作为判断依据,而不是根据动态类型。collections数组里面的所有实例的静态类型都是Collection<?>getType方法也都是执行上文的第一个重载方法。

# 程序执行结果
Super:collection
Super:collection

你学会(fei)了吗?学fei之后就开始看看面试题把,看看自己是否能过关呢?

二、接口和多态常见面试题

1. 什么是 Java 接口?接口的主要用途是什么?

回答: Java 接口是一种特殊的引用数据类型,用于定义类必须实现的一组方法。接口只能包含方法的声明,而不能包含方法的实现。接口的主要用途是提供一种机制,使得不同的类可以以一致的方式进行交互。接口支持多继承,可以让类实现多个接口,提供了灵活的设计方式。

2. 接口和抽象类的区别是什么?

回答: 接口和抽象类都是用于定义规范的工具,但有以下主要区别:

  • 接口: 只能包含方法的声明(从 Java 8 起,可以有默认方法和静态方法),不能有构造函数、实例变量。一个类可以实现多个接口。
  • 抽象类: 可以包含方法的实现、构造函数和实例变量。一个类只能继承一个抽象类(Java 语言只支持单继承)。

3. Java 8 中接口有什么新特性?

回答: Java 8 引入了几个接口的新特性:

  • 默认方法: 可以在接口中定义具有默认实现的方法,使用 default 关键字。
  • 静态方法: 可以在接口中定义静态方法。
  • 函数式接口: 使用 @FunctionalInterface 注解来标记一个接口为函数式接口,确保接口只有一个抽象方法。

4. 如何在 Java 中实现多态?

回答: 多态是在 Java 中实现灵活、可扩展的对象行为的一种机制。主要有两种实现方式:

  • 方法重载: 同一个类中方法名相同但参数不同。
  • 方法重写: 子类重写父类的非静态方法。多态通过方法重写和引用类型的向上转型实现,即使用父类引用指向子类对象,可以调用子类重写的方法。

5. 什么是方法重载?方法重载和方法重写的区别是什么?

回答:

  • 方法重载: 在同一个类中,方法名相同但参数列表不同(参数类型、个数或顺序不同),且方法的返回类型可以不同。
  • 方法重写: 子类重新实现父类的已存在方法,方法名、参数列表和返回类型必须完全相同。

6. 什么是抽象方法?如何定义抽象方法?

回答: 抽象方法是没有实现的方法,只包含方法的声明。定义抽象方法时,使用 abstract 关键字,且方法体为空。抽象方法只能在抽象类或接口中定义,子类必须实现抽象方法,除非子类也是抽象类。

7. 如何使用 Java 中的接口进行回调?

回答: 接口可以用于实现回调机制。例如:

public interface Callback {void onComplete(String result);
}public class Task {private Callback callback;public Task(Callback callback) {this.callback = callback;}public void doWork() {// 执行一些工作callback.onComplete("任务完成");}
}public class Main {public static void main(String[] args) {Task task = new Task(result -> System.out.println(result));task.doWork();}
}

在这个例子中,Callback 接口用于回调机制,Task 类接受一个 Callback 实例,并在完成工作后调用 onComplete 方法。

8. 什么是接口的默认方法?你能给一个示例吗?

回答: 接口的默认方法是在接口中定义的具有默认实现的方法,使用 default 关键字。例如:

public interface MyInterface {default void defaultMethod() {System.out.println("这是一个默认方法");}
}public class MyClass implements MyInterface {// 可以选择重写默认方法,也可以使用默认实现
}public class Main {public static void main(String[] args) {MyClass myClass = new MyClass();myClass.defaultMethod(); // 输出:这是一个默认方法}
}

9. 如果一个类实现了多个接口,其中包含相同的方法名但有不同的默认实现,如何解决冲突?

回答: 如果一个类实现了多个接口,并且这些接口有相同的方法名但不同的默认实现,编译器会报错。解决这个问题的方法是,在实现类中显式地重写这个方法,并提供一个新的实现。例如:

public interface InterfaceA {default void method() {System.out.println("InterfaceA 的实现");}
}public interface InterfaceB {default void method() {System.out.println("InterfaceB 的实现");}
}public class MyClass implements InterfaceA, InterfaceB {@Overridepublic void method() {System.out.println("MyClass 的实现");}
}

10. 如何判断一个类是否实现了某个接口?

回答: 可以使用 instanceof 操作符来判断一个对象是否实现了某个接口。例如:

if (obj instanceof MyInterface) {// obj 实现了 MyInterface 接口
}

11. 你能解释一下“鸭子类型”在多态中的作用吗?

回答: “鸭子类型”是一种编程范式,基于对象的行为而非对象的实际类型。在 Java 中,这种类型的实现方式就是接口。如果一个对象实现了某个接口的方法,我们可以说这个对象是该接口的实现类型。通过这种方式,我们可以在不关心具体实现的情况下,利用对象的行为特性来进行编程。

12. 如何在接口中定义常量?

回答: 在接口中定义的常量使用 `public static final` 修饰符。例如: java public interface MyInterface { int CONSTANT_VALUE = 42; } 接口中的常量默认是 `public static final`,并且必须初始化。

13. Java 接口的继承和实现有什么区别?

回答: 接口的继承使用 `extends` 关键字,可以继承多个接口,并且可以继承其他接口的方法声明。而实现接口的类使用 `implements` 关键字,必须实现接口中定义的所有抽象方法。接口之间的继承是为了扩展接口的功能,而类的实现则是提供具体的实现。

14. 接口的多个继承会导致冲突吗?如何解决?

回答: 接口的多重继承不会导致问题,因为接口只定义方法的签名,不包含实现。如果多个接口中有相同的方法名但不同的默认实现,冲突会在实现类中解决。实现类需要重写这个方法,提供一个新的实现。

15. 什么是函数式接口?如何创建一个函数式接口?

回答: 函数式接口是只包含一个抽象方法的接口,可以用来作为 lambda 表达式或方法引用的目标。使用 `@FunctionalInterface` 注解来标记一个接口为函数式接口。例如: java @FunctionalInterface public interface MyFunctionalInterface { void doSomething(); } 

16. 什么是接口的多继承?这与类的多继承有何不同?

回答: 接口的多继承指的是一个接口可以继承多个接口,这种继承方式是合法的且支持的。接口之间的多继承是允许的,因为接口只定义方法签名,不包含实现。与此不同的是,Java 不支持类的多继承,以避免复杂的继承关系和潜在的冲突。

17. 在 Java 中,如何实现接口的动态代理?

回答: 可以使用 `java.lang.reflect.Proxy` 类来创建接口的动态代理。例如:

 


public interface MyInterface {void doSomething();
}public class MyInvocationHandler implements InvocationHandler {@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("调用方法:" + method.getName());return null;}
}public class Main {public static void main(String[] args) {MyInterface proxyInstance = (MyInterface) Proxy.newProxyInstance(MyInterface.class.getClassLoader(),new Class[]{MyInterface.class},new MyInvocationHandler());proxyInstance.doSomething();}
}

18. Java 中如何使用接口来实现策略模式?

回答: 策略模式是一种行为设计模式,用于定义一系列算法,并使它们可以互换。通过接口来定义这些算法,然后在上下文中使用。示例:

public interface Strategy {int execute(int a, int b);
}public class AdditionStrategy implements Strategy {@Overridepublic int execute(int a, int b) {return a + b;}
}public class Context {private Strategy strategy;public Context(Strategy strategy) {this.strategy = strategy;}public int executeStrategy(int a, int b) {return strategy.execute(a, b);}
}public class Main {public static void main(String[] args) {Strategy strategy = new AdditionStrategy();Context context = new Context(strategy);System.out.println(context.executeStrategy(5, 3)); // 输出:8}
}

19. Java 接口是否支持静态方法?如何定义和使用?

回答: 是的,Java 接口支持静态方法,从 Java 8 开始,可以在接口中定义静态方法。静态方法不能被实现类重写,只能通过接口名调用。例如: 

public interface MyInterface {static void staticMethod() {System.out.println("接口的静态方法");}
}public class Main {public static void main(String[] args) {MyInterface.staticMethod(); // 输出:接口的静态方法}
}

20. 如何在接口中定义默认方法,并让实现类选择是否重写?

回答: 默认方法在接口中使用 `default` 关键字定义,可以为接口中的方法提供默认实现。实现类可以选择是否重写默认方法。如果实现类不重写默认方法,类将使用接口提供的默认实现。例如: 

public interface MyInterface {default void defaultMethod() {System.out.println("默认实现");}
}public class MyClass implements MyInterface {@Overridepublic void defaultMethod() {System.out.println("重写后的实现");}
}public class Main {public static void main(String[] args) {MyClass myClass = new MyClass();myClass.defaultMethod(); // 输出:重写后的实现}
}

让我们一起学习,一起进步!期待在评论区与你们见面。

祝学习愉快!

这篇关于接口和多态详解,还不快点学?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1125980

相关文章

Springboot配置文件相关语法及读取方式详解

《Springboot配置文件相关语法及读取方式详解》本文主要介绍了SpringBoot中的两种配置文件形式,即.properties文件和.yml/.yaml文件,详细讲解了这两种文件的语法和读取方... 目录配置文件的形式语法1、key-value形式2、数组形式读取方式1、通过@value注解2、通过

自定义注解SpringBoot防重复提交AOP方法详解

《自定义注解SpringBoot防重复提交AOP方法详解》该文章描述了一个防止重复提交的流程,通过HttpServletRequest对象获取请求信息,生成唯一标识,使用Redis分布式锁判断请求是否... 目录防重复提交流程引入依赖properties配置自定义注解切面Redis工具类controller

Python容器转换与共有函数举例详解

《Python容器转换与共有函数举例详解》Python容器是Python编程语言中非常基础且重要的概念,它们提供了数据的存储和组织方式,下面:本文主要介绍Python容器转换与共有函数的相关资料,... 目录python容器转换与共有函数详解一、容器类型概览二、容器类型转换1. 基本容器转换2. 高级转换示

HTML5的input标签的`type`属性值详解和代码示例

《HTML5的input标签的`type`属性值详解和代码示例》HTML5的`input`标签提供了多种`type`属性值,用于创建不同类型的输入控件,满足用户输入的多样化需求,从文本输入、密码输入、... 目录一、引言二、文本类输入类型2.1 text2.2 password2.3 textarea(严格

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

MySQL中between and的基本用法、范围查询示例详解

《MySQL中betweenand的基本用法、范围查询示例详解》BETWEENAND操作符在MySQL中用于选择在两个值之间的数据,包括边界值,它支持数值和日期类型,示例展示了如何使用BETWEEN... 目录一、between and语法二、使用示例2.1、betwphpeen and数值查询2.2、be

python中的flask_sqlalchemy的使用及示例详解

《python中的flask_sqlalchemy的使用及示例详解》文章主要介绍了在使用SQLAlchemy创建模型实例时,通过元类动态创建实例的方式,并说明了如何在实例化时执行__init__方法,... 目录@orm.reconstructorSQLAlchemy的回滚关联其他模型数据库基本操作将数据添

Java中ArrayList与顺序表示例详解

《Java中ArrayList与顺序表示例详解》顺序表是在计算机内存中以数组的形式保存的线性表,是指用一组地址连续的存储单元依次存储数据元素的线性结构,:本文主要介绍Java中ArrayList与... 目录前言一、Java集合框架核心接口与分类ArrayList二、顺序表数据结构中的顺序表三、常用代码手动

JAVA线程的周期及调度机制详解

《JAVA线程的周期及调度机制详解》Java线程的生命周期包括NEW、RUNNABLE、BLOCKED、WAITING、TIMED_WAITING和TERMINATED,线程调度依赖操作系统,采用抢占... 目录Java线程的生命周期线程状态转换示例代码JAVA线程调度机制优先级设置示例注意事项JAVA线程

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra