【小家java】java8中接口方法引发的思考(默认方法、static静态方法)

简介: 【小家java】java8中接口方法引发的思考(默认方法、static静态方法)

前言



各位小伙伴大家好,我是A哥。


1、概述


以前,接口里的方法要求全部是抽象方法,java8以后允许在接口里定义默认方法和类方法。不同的是:


默认方法可以通过实现接口的类实例化的对象来调用,而类方法就相对于工具方法了。需要注意的是,此处的静态方法只能被public修饰(或者省略不写),不能是private或者protected


java8可以说是jdk版本的一次重大升级,给我们也带来了非常多的新特性,而本文就针对于java8中很重要的新特性之一:接口方法。来讨论一下平时使用中可能遇到的问题


2、栗子


大家都知道,在jdk8之后,接口里面咱们都可以写具体的方法了,但这方法比较特殊,只能是静态方法或者默认方法。这又让我们有更好的设计,可以设计出更加高内聚的代码,更加方便的管理封装。

public interface MyInter {
    default void df(){    //声明一个接口的默认方法
        System.out.println("i'am default f");
        sf();        //调用本接口的类方法
    }
    static void sf(){    //声明一个接口的类方法
        System.out.println("i'am static f");
    }
}

如上,本接口的默认方法还可以直接调用本类的静态方法。


默认方法的主要优势是提供一种拓展接口的方法,而不破坏现有代码。加入我们有一个已经投入使用接口需要拓展一个新的方法,在JDK8以前,如果为一个使用的接口增加一个新方法,则我们必须在所有实现类中添加该方法的实现,否则编译会出现异常。如果实现类数量少并且我们有权限修改,可能会工作量相对较少。如果实现类比较多或者我们没有权限修改实现类源代码,这样可能就比较麻烦。而默认方法则解决了这个问题,它提供了一个实现,当没有显示提供其他实现时就采用这个实现。这样新添加的方法将不会破坏现有代码。


默认方法的另一个优势是该方法是可选的,子类可以根据不同的需求Override默认实现


咱们先定义两个接口,下面要使用的

interface DemoInterface {
    default void doSomething() {
        System.out.println("默认方法-->demo");
    }
}
interface Demo1Interface {
    default void doSomething() {
        System.out.println("默认方法-->demo1");
    }
}


然后定义一个实现类


class AA implements DemoInterface {
    public static void main(String[] args) {
        AA aa = new AA();
        aa.doSomething(); //默认方法-->demo
    }
}


输出默认方法的输出,但如果AA实现了两个接口,并且这两个接口里有有同名默认方法呢?


image.png


我们会发现,编译报错了,强制要求我们必须实现这个方法。


class AA implements DemoInterface, Demo1Interface {
    @Override
    public void doSomething() {
        Demo1Interface.super.doSomething();
    }
    public static void main(String[] args) {
        AA aa = new AA();
        aa.doSomething(); //默认方法-->demo1
    }
}


再一个情况,接口之间存在继承关系,然后存在同名情况


interface DemoInterface {
    default void doSomething() {
        System.out.println("默认方法-->demo");
    }
}
interface Demo1Interface extends DemoInterface {
    default void doSomething() {
        System.out.println("默认方法-->demo1");
    }
}
class AA implements Demo1Interface {
    public static void main(String[] args) {
        AA aa = new AA();
        aa.doSomething(); //默认方法-->demo1
    }
}


我们发现,有点类似于Maven的原则:采取就近原则


综上,咱们经过实验得出如下结论:


     1.当继承的父类和实现的接口中有相同签名的方法时,优先使用父类的方法。


     2.当接口的父接口中也有同样的默认方法时,就近原则调用子接口的方法。


     3.当实现的多个接口中有相同签名的方法时,必须在实现类中通过重写方法解决冲突问题,否者无法通过编译,在重写的方法中可以通过 接口名.super.方法名(); 的方式显示调用需要的方法。


如果一个类既extend了父类,又实现了接口。如果出现同名方法,那就遵循类优先原则。

3、使用场景


接口是设计模式中一种开闭原则的体验,而java8赋予了接口新的特性,使得接口使用起来更加的得心应手了,这也有助于我们更加内聚自己的代码结构了。比如collection即可类和接口,排序接口,把很多工具方法都放到接口里了

4、最后


多多使用新的特性,就能多多提高生产效率(编码效率)。另外,我可以引出一个提问:为什么java的接口里的属性必须是static的?并且要求必须是final的呢?这个留给大家自己做思考。。。算了,顺便奉上吧:


1.接口中的数据对所有实现类只有一份,所以是static


2.要使实现类为了向上转型成功,所以必须是final的(接口不能被实例化,所以接口里面如果是变量的话不会被赋初始值这样就会出问题,所以必须是final的。其实还是为了安全考虑的) 这样接口也能起到一定的模版的作用


相关文章
|
9天前
|
JSON Java Apache
非常实用的Http应用框架,杜绝Java Http 接口对接繁琐编程
UniHttp 是一个声明式的 HTTP 接口对接框架,帮助开发者快速对接第三方 HTTP 接口。通过 @HttpApi 注解定义接口,使用 @GetHttpInterface 和 @PostHttpInterface 等注解配置请求方法和参数。支持自定义代理逻辑、全局请求参数、错误处理和连接池配置,提高代码的内聚性和可读性。
|
10天前
|
Java
java线程接口
Thread的构造方法创建对象的时候传入了Runnable接口的对象 ,Runnable接口对象重写run方法相当于指定线程任务,创建线程的时候绑定了该线程对象要干的任务。 Runnable的对象称之为:线程任务对象 不是线程对象 必须要交给Thread线程对象。 通过Thread的构造方法, 就可以把任务对象Runnable,绑定到Thread对象中, 将来执行start方法,就会自动执行Runable实现类对象中的run里面的内容。
25 1
|
16天前
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
42 4
|
20天前
|
Java
Java基础(13)抽象类、接口
本文介绍了Java面向对象编程中的抽象类和接口两个核心概念。抽象类不能被实例化,通常用于定义子类的通用方法和属性;接口则是完全抽象的类,允许声明一组方法但不实现它们。文章通过代码示例详细解析了抽象类和接口的定义及实现,并讨论了它们的区别和使用场景。
|
Java
Java接口和抽象类
Java接口和抽象类
89 0
|
3月前
|
设计模式 Java
【惊天揭秘】Java编程绝技大曝光:接口、抽象类、静态类与非静态类的神秘面纱终被揭开!
【8月更文挑战第22天】Java支持面向对象编程,通过接口、抽象类、静态类(如枚举与工具类)及普通类实现设计原则。接口定义行为规范,允许多重继承;抽象类含未实现的抽象方法,需子类完成;静态类常为工具类,提供静态方法;普通类则实例化对象。恰当运用这些结构能提升程序质量。
40 2
|
6月前
|
设计模式 搜索推荐 Java
java接口和抽象类的区别,以及使用选择
java接口和抽象类的区别,以及使用选择
80 0
|
3月前
|
Java 开发者
Java中的接口和抽象类
Java中的接口和抽象类
29 3
|
4月前
|
网络协议 Java
Java面试题:什么是Java中的接口?与抽象类有什么区别?
Java面试题:什么是Java中的接口?与抽象类有什么区别?
40 0
|
6月前
|
Java
Java的接口与抽象类的区别
Java的接口与抽象类的区别