首页 > 编程语言 >【Java Lambda系列】新玩法,用Lambda重构设计模式

【Java Lambda系列】新玩法,用Lambda重构设计模式

时间:2024-08-16 21:27:29浏览次数:10  
标签:20 void Java 设计模式 out public Lambda

前言

前面三章通过理论+案例的方式对Lambda的描述,应该能基本上解决大家日常开发中所遇到的Lambda问题,为了更好的展现Lambda魅力,和加深巩固Lambda知识点,今天咱们讨论Lambda如何重构设计模式!

关于设计模式

众所周知,设计模式是一群大佬程序员将对程序设计的经验归纳总结起来的方案。它不是语法规定,而是一套用来提高代码可复用性、可维护性、可读性、稳健性以及安全性的解决方案。

本文不会对设计模式过度讲解,默认大家对设计模式有一定的了解(后面我会针对所有的设计模式单独进行讲解)

Lambda重构设计模式

接下来列举常用的几个设计模式,通过传统和lambda形式对比写法,让大家感受不一样的、Lambda版本的设计模式!

简单工厂模式

简单工厂模式因为内部方法常被定义为静态,所以又叫做静态工厂方法(Static Factory Method)模式,是由一个工厂对象通过客户端需要的类型创建出某个产品类的实例,简单工厂模式是工厂模式家族中最简单实用的模式,可以理解为是不同工厂模式的一个特殊实现。

案例:水果工厂

包含橘子、香蕉、苹果

  • 传统写法
/**
 * 简单工厂
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20
 */
public class FruitFactory {
    public static Fruit getFruit(String fruitName) {
        Fruit fruit = null;
        switch (fruitName) {
            case "Banana":
                fruit = new Banana();
                break;
            case "Apple":
                fruit = new Apple();
                break;
        }
        return fruit;
    }
}

工厂模式的本质就是调用的时候返回一个对象,仔细回想一下,Java内置的函数式接口中Supplier似乎就是干这个事的,话不多说直接改写吧,

  • lambda写法
/**
 * Lambda形式工厂模式
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20
 */
public class FruitLambdaFactory {
    /**
     * 初始化工厂数据
     */
    private static Map<String, Supplier<Fruit>> FRUIT_FACTORY = new HashMap<String, Supplier<Fruit>>(){{
       put("Banana", Banana::new);
       put("Apple", Apple::new);
    }};

    public static Fruit getLambdaFruit(String fruit) {
        Supplier<Fruit> fruitSupplier = FRUIT_FACTORY.get(fruit);
        return Objects.nonNull(fruitSupplier) ? fruitSupplier.get() : null;
    }
}

题外话: 上述写法其实存在瑕疵,比如:再新增一个水果品类,就必须改动工厂的静态方法,违背了开闭原则,而且通过传入字符串的形式也很容易导致拼错问题,实际业务中我们可以使用反射来解决这个问题。

  • 反射写法
/**
 * 反射形式的工厂模式
 *
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20
 */
public class FruitReflectFactory {

    public static <T extends Fruit> T getLambdaFruit(Class<T> fruitCls) {
        try {
            return fruitCls.newInstance();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
        return null;
    }
}

策略模式

策略模式就是封装一系列算法,让这些算法之间可以在运行时相互替换,可以很好的解决过多使用if...else if...else的问题!

案例:多种发送信息的通道

包含:邮件通道、websocket通道

  • 传统模式
/**
 * 发送通道策略
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20  20:18
 */
public interface SendChannelStrategy {
    /**
     * 发送接口
     * @param msg
     */
    void send(String msg);
}

/**
 * 邮箱策略
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20  20:19
 */
public class EmailChannelStrategy implements SendChannelStrategy{
    @Override
    public void send(String msg) {
        System.out.println("使用email发送:" + msg);
    }
}

/**
 * websocket策略
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20  20:19
 */
public class WebSocketChannelStrategy implements SendChannelStrategy{
    @Override
    public void send(String msg) {
        System.out.println("使用websocket发送:" + msg);
    }
}
/**
 * 消息服务,使用发送策略
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20
 */
@AllArgsConstructor
@Getter@Setter
public class SendService {
    private SendChannelStrategy sendChannelStrategy;
    
    public void send(String msg) {
        sendChannelStrategy.send(msg);
    }
}
// 使用方式如下
public void test(){
    SendService sendService = new SendService(new EmailChannelStrategy());
    sendService.send("卡诺"); // 使用email发送:卡诺
    sendService.setSendChannelStrategy(new WebSocketChannelStrategy());
    sendService.send("卡诺"); // 使用websocket发送:卡诺
}

这里我们可以发现上面的策略模式接口其实是个函数式接口(只有一个抽象方法哦),方法拥有一个入参,无返回值,是不是一下子就想到了Java的内置函数式接口Consumer呢?下面我们就来看看Lambda的处理方式把!

  • lambda写法
public void testLambda(){
    SendService sendService = new SendService(msg -> System.out.println("使用lambda email发送:" + msg));
    sendService.send("卡诺"); // 使用email发送:卡诺
    
    sendService.setSendChannelStrategy(msg -> System.out.println("使用websocket发送:" + msg));
    sendService.send("卡诺"); // 使用websocket发送:卡诺
}

代码如所想,使用Consumer进行替换,我们可以直接省略掉EmailChannelStrategyWebSocketChannelStrategy类的定义!

模版方法模式

模版方法模式指:父类定义好流程,允许子类实现流程中的一个或多个步骤。这样就可以保证整体流程不变的情况下,子类实现自己的个性化操作。

案例:游戏的过程

选择一个游戏的姿势玩(躺着/坐着)、开始游戏、游戏结束

  • 传统写法
/**
 * 游戏
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20  20:49
 */
public abstract class Game {

    public void play(String posture) {
        selectPlayPosture(posture);
        System.out.println("开始游戏");
        System.out.println("结束游戏");
    }

    /**
     * 选择游戏姿势
     */
    public abstract void selectPlayPosture(String posture);
}
/**
 * 躺着玩
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20  20:53
 */
public class LieGame extends Game{
    @Override
    public void selectPlayPosture(String posture) {
        System.out.println("沙发上:" + posture);
    }
}
/**
 * 坐着玩
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20  20:52
 */
public class SitGame extends Game{
    @Override
    public void selectPlayPosture(String posture) {
        System.out.println("板凳上:" + posture);
    }
}
// 测试
public void test(){
    Game game = new LieGame();
    game.play("躺着");
    game = new SitGame();
    game.play("坐着");
}

通过上述的代码,我们可以发现模版模式中需要被子类实现的抽象接口是一个入参,无返回值的抽象方法,Java内置函数式接口Consumer与之恰好匹配,我们可以将需要子类实现的方法,通过Consumer提取作为游戏流程的参数,即可大幅度减少代码,如下:

  • Lambda写法
/**
 * Lambda形式的模版方法
 * @author : uu
 * @version : v1.0
 * @Date 2022/1/20  20:49
 */
public class LambdaGame {

    public void play(String posture, Consumer<String> consumer) {
        consumer.accept(posture);
        System.out.println("开始游戏");
        System.out.println("结束游戏");
    }
}

// 使用方式
public void testLambda(){
    LambdaGame game = new LambdaGame();
    game.play("躺着", System.out::println);
    game.play("坐着", System.out::println);
}

通过Lambda的改写是不是看起来简单了许多呢?

总结

  • 本章通过使用Lambda重构设计模式的形式展现Lambda给代码编写带来的便捷;
  • 本次仅列出了几个常用的设计模式改写,小伙伴们有兴趣可以一起讨论其他模式的改写哦;
  • 案例中代码比较简单,实际开发时还是得依托业务的需求,并不能一味的为了简洁而简洁。

标签:20,void,Java,设计模式,out,public,Lambda
From: https://blog.csdn.net/qq_16828223/article/details/141270038

相关文章

  • Java 开发者必备:一文解决 AES 加密中的“非法密钥大小”异常
    彻底告别java.security.InvalidKeyException,轻松应对不同JDK版本引言在Java开发过程中,我们经常会遇到各种各样的安全相关的问题。其中一个常见的问题是当使用Java的加密功能时遇到的“Illegalkeysizeordefaultparameters”错误。本文将详细介绍如何解决这一问......
  • 暑假Java自学进度总结06
    一.今日所学:1.for循环for(初始化语句;条件判断语句;条件控制语句){循环体语句;}执行流程:1>执行初始化语句2>执行条件判断语句,若为true则执行循环体语句,若为false,循环结束3>执行条件控制语句4>回到2>继续执行条件判断语句注:初始化语句只执行一次2.while循环初始化语句;......
  • Java基础
    Java基础1注释单行注释格式://注释信息多行注释格式:/*注释信息*/文档注释格式:/**注释信息*/注:新手小白,文档注释暂时用不上。publicclassHelloWorld{publicstaticvoidmain(String[]args){//叫做main方法,程序的主入口S......
  • Java入门基础篇
    一、java简介:java是一种广泛使用的高级编程语言,由SunMicrosystems公司于1995年推出。Java的设计目标之一是实现“一次编写,到处运行”的能力,即编写的代码可以在任何支持Java的平台上运行,而无需进行额外的修改。Java广泛应用于企业级应用、桌面应用、移动应用和Web开发等领域。......
  • 【蓝桥杯】第七届蓝桥杯大赛个人赛省赛(软件类)Java 大学C组 真题
    第七届蓝桥杯大赛个人赛省赛(软件类)Java大学C组真题及部分解析A 有奖猜谜小明很喜欢猜谜语。最近,他被邀请参加了X星球的猜谜活动。每位选手开始的时候都被发给777个电子币。规则是:猜对了,手里的电子币数目翻倍,猜错了,扣除555个电子币,扣完为止。小明一共猜了15条谜语......
  • 10步打造专业级Java FX应用:从入门到放弃的艺术
    ......
  • 文心快码Baidu Comate 帮你解大厂面试题:Java G1 GC中,region是什么意思?有哪些不同的reg
    ......
  • JAVA毕业设计161—基于Java+Springboot+vue+微信小程序的校园论坛二手闲置系统(源代码
    毕设所有选题:https://blog.csdn.net/2303_76227485/article/details/131104075基于Java+Springboot+vue+微信小程序的校园论坛二手闲置系统(源代码+数据库+万字论文)161一、系统介绍本项目前后端分离带小程序,分为用户、管理员两种角色,可自行分配角色菜单1、用户:注册、......
  • 《提升前端性能的 JavaScript 技巧》
    在前端开发中,性能优化是至关重要的一环。JavaScript作为前端开发的核心语言,掌握一些关键的技巧可以显著提升应用的性能。本文将为您介绍一些实用的JavaScript技巧,帮助您优化前端性能。一、避免不必要的计算在代码中,尽量避免在频繁执行的代码块中进行复杂且不必要的计算。......
  • day22 Java基础——方法(干货)
    day22Java基础——方法在Java中,方法是一段组织好的、可重复使用的代码块,用于执行一个特定的操作。方法提供了一种封装代码的方式,使得代码模块化,便于管理和重用。以下是关于Java中方法的一些基本介绍:文章目录day22Java基础——方法1.方法的定义2.方法的调用2.1方法......