原来Java接口多实现还可以这样玩

前言

多继承的目的是为了提高子类的功能,扩展性得到了提高。

为了扩充子类的功能,java改良成多实现。这里带来两个问题,多实现就没有不确定性问题了么?接口中都是抽象方法,实现多实现结构还要重写,意义大么?

第二个问题,就是在于要重写方法,而多实现解决的是让实现的子类具备一些功能,至于重写函数的麻烦不在考虑的范围之内。那么第一个不确定性问题呢?

多继承不支持,是因为方法的不确定性,声明相同,方法体不同,子类在继承执行相同的方法时,不知道执行哪个方法体。毛病出现在方法体上。禁止使用多继承,就是为了预防出现这样的问题。但是接口中定义的全是抽象方法,在多个继承的时候(说法不正确,要说成实现,但是本质还是继承覆盖),即使碰到相同的函数,也没关系,因为里面没有方法体,要执行的方法体是子类的自己重写,压根不会产生要执行多个不同的方法体的选择问题。而且,我认为多接口中,根本不会出现相同的方法,因为,编写的时候是从上往下的书写的。只是在最初在理解的时候,是从下往上的,可能要遇到这些问题。多继承不执行,也有父类中有一般函数,不是抽象函数的问题。不是说子类多继承几个父类不行,有一些情况也是可以的,但是为了预防出现的问题,所以禁止它了。

接口是给子类提高了一些功能,告诉子类你可以具备哪些功能,至于怎么实现这些功能,需要子类自己书写。

Java中的接口类通常是为了提取共同点,规范实现,便于阅读,处理好接口类多实现并提供优雅的命中具体实现,能够帮助我们简化代码,提高可读性;下面介绍几种用起来很舒服的多实现方式及调用方式供大家参考。

示例

枚举实现

接口定义

public interface Breakfast {
    void eat();
}

实现

public enum BreakfastEnum implements Breakfast {


    Beijing("北京") {
        @Override
        public void eat() {
            System.out.println("北京人早餐吃豆汁和焦圈");
        }
    },

    Wuhan("武汉") {
        @Override
        public void eat() {
            System.out.println("武汉人早餐吃热干面、豆皮...");
        }
    },

    Unknown("未知") {
        @Override
        public void eat() {
            System.out.println("不吃早餐!");
        }
    };

    private String city;

    BreakfastEnum(String city) {
        this.city = city;
    }

    private String getCity() {
        return this.city;
    }

    /**
     * 提供统一入口 找到对应子类并执行
     *
     * @param city
     */
    public static void eat(String city) {
        BreakfastEnum[] values = BreakfastEnum.values();
        Arrays.stream(values).filter(e -> city.equals(e.city)).findFirst().orElse(Unknown).eat();
    }
}

测试一下

原来Java接口多实现还可以这样玩_第1张图片

原来Java接口多实现还可以这样玩_第2张图片

通过枚举类实现接口,每一个枚举相当于一个实现,在代码块实现方法即可,最后在枚举类提供一个静态方法作为统一入口,调用方便,代码简洁,提供通用实现处理无特定实现的场景,适合用于替换ifelse较多的业务代码,优化复杂的工具类等等,对于方法很多,业务复杂的业务慎用。

常规多实现(调用示例)

业务场景

我们有一个消息服务用于监听消息并发送到客户端,消息中有一个发布方式字段

原来Java接口多实现还可以这样玩_第3张图片

1.根据消息的发布方式字段发送到指定途径

2.将消息发到所有途径

接口定义

public interface MessageHandle {

    /**
     * 发布消息
     *
     * @param msg
     */
    void publish(JSONObject msg);
}

实现

/**
 * 发送短信
 */
@Service("sms")
public class SmsMessageHandle implements MessageHandle {
    @Override
    public void publish(JSONObject msg) {
        // 发送短信
        // 省略实现...
    }

}

/**
 * 推送
 */
@Service("push")
public class PushMessageHandle implements MessageHandle {
    @Override
    public void publish(JSONObject msg) {
        // 推送到app
        // 省略实现...
    }

}

应用

// 1.指定途径发送
@Component
public class MessageListener {

    @Autowired
    private Map messageHandleMap;
    
    @KafkaListener(groupId = "message-server", topics = "message")
    public void listener(String message, Acknowledgment ack) {
        JSONObject messageJson = JSON.parseObject(message);
        // 获取发布方式 sms push...对应实现@Service注解中的名称
        String publishType =  messageJson.getString("publishType");
        // 获取实现
        MessageHandle handle = messageHandleMap.get(publishType);
        if(handle != null) {
            handle.publish(messageJson);
        }
        // 提交偏移量
        ack.acknowledge();
    }
    
 }
    
// 2.每种途径都发送
@Component
public class MessageListener {

    @Autowired
    private List messageHandleList;
    
    @KafkaListener(groupId = "message-server", topics = "message")
    public void listener(String message, Acknowledgment ack) {
        JSONObject messageJson = JSON.parseObject(message);
        // 每种途径都发送
        for(MessageHandle handle : messageHandleList){
            handle.publish(messageJson);
        }
        // 提交偏移量
        ack.acknowledge();
    }
    
 }
    

总结

到此这篇关于Java接口多实现的文章就介绍到这了,更多相关Java接口多实现内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

你可能感兴趣的:(原来Java接口多实现还可以这样玩)