前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >函数式编程是如何提升代码的扩展性

函数式编程是如何提升代码的扩展性

作者头像
微观技术
发布2020-09-08 10:58:00
1.1K1
发布2020-09-08 10:58:00
举报
文章被收录于专栏:微观技术微观技术

软件的发展大致经历三个阶段

  • 第一阶段(20世纪40年代中期到50年代中期),主要是科学与工程计算,处理对象为数值数据,以个体方式使用机器(或汇编)语言编制程序
  • 第二阶段(20世纪50年代中期到60年代后期),从高级程序设计语言出现到软件工程提出以前。这个阶段研究对象增加了并发程序,并着重研究高级程序设计语言、编译程序、操作系统以及各种支撑软件和应用软件
  • 第三阶段(20世纪60年代后期软件工程提出以来),由于大型软件的开发耗时耗力,任务重,需要采用合作的方式才能完成,所以引入软件工程的理念来管理项目。

从工程学角度来讲,我们常说的软件工程一般采用面向对象编程,差别在与使用的编程语言不同,有人习惯用java,有人喜欢C#,各有各的特色,除了语法上略有差异,其本质都差不多,所以你可能会经常听到有人说,只要你精通了一门语言,再学习其他语言,会感觉如有神助,基本也是这个道理。

面向对象编程

首先我们来看下面向对象编程的三大特性

  • 封装
  • 继承
  • 多态

面向对象编程是一种具有对象概念的程序编程范型,它可能包含数据、属性、代码与方法。对象则指的是类的实例。它将对象作为程序的基本单元,将程序和数据封装其中,以提高软件的可重用性、灵活性和可扩展性,对象里的程序可以访问及修改对象相关联的数据。在面向对象编程里,计算机程序会被设计成彼此相关的对象。

对象按照执行角色,可以分为数据对象、行为对象。我们常理解的面向对象编程的模式,比如:y=f(x),其中x、y都是数据对象,通过行为对象F的方法运算得到了加工后的对象y。

我们具体看个示例:

代码语言:javascript
复制
/**
 * @Author onlyone
 * <p>
 * 活动模型
 */
public class Activity {

    private Long id; // 活动id
    private String name; // 名称
    private String desc;  // 描述
    private Date time;  // 活动时间
    private String publisher; // 发布人
}

按活动id查找一个活动,代码一般会这么写

代码语言:javascript
复制
public Activity queryById(List<Activity> activityList, String id) {
    for (Activity activity : activityList) {
        if (id.equals(activity.getId())) {
            return activity;
        }
    }
    return null;
}

如果此时业务提出了一个新的需求,按名称来查询活动,得嘞,又得重新造个轮子

代码语言:javascript
复制
public Activity queryByName(List<Activity> activityList, String name) {
    for (Activity activity : activityList) {
        if (name.equals(activity.getId())) {
            return activity;
        }
    }
    return null;
}

过了几天,业务产品又来找你了,想根据时间来查询活动,此时,你是不是有种崩溃的感觉。

重构是我们脑海闪现的第一念想,如何重构,却让我们陷入一脸懵逼的茫然状态。三个需求,处理逻辑各不相同,如何复用抽取?

我们需要改变我们的思维方式,谁规定调用方法传入的实参一定是数值型对象,如果传入一个函数表达式,能不能解决这个问题?

是不是有种豁然开朗的感觉。

函数式编程

函数式编程第一个需要了解的概念就是函数:

  • 函数可以按需创建
  • 函数可以当作实参传给另一个方法
  • 函数可以当作另一个方法的返回值

JDK 8 开始引入函数式编程,并提供了很多预定义接口类,如 Predicates 用于判断,函数 Functions,生产 Suppliers,消费 Consumers,比较 Comparators。

代码示例:https://github.com/aalansehaiyang/java8-tutorial

本文的重构思路就是采用Predicate接口,我们先来看些内部结构

代码语言:javascript
复制
@FunctionalInterface
public interface Predicate<T> {
    boolean test(T t);
}
代码语言:javascript
复制
/**
 * @Author onlyone
 * 面向函数编程
 */
public class FuncitionProgram {

    private List<Activity> activityList;

    /**
     * 基础查询骨架
     */
    Activity queryByPredicate(Predicate<Activity> predicate) {
        for (Activity activity : activityList) {
            if (predicate.test(activity)) {
                return activity;
            }
        }
        return null;
    }
}

按id查询活动,传入对应的Predicate表达式

代码语言:javascript
复制
public Activity queryById(String id) {
    return queryByPredicate(activity -> id.equals(activity.getId()));
}

按名称查询活动,传入对应的Predicate表达式

代码语言:javascript
复制
public Activity queryByName(String name) {
    return queryByPredicate(activity -> name.equals(activity.getName()));
}

是不是有种”牛逼“的感觉。低调,优化是永无止境,有没有更好的优化方式。

作为一名架构师,我们在做系统架构时,为了满足其高并发、扩展性,一般会讲究一个拆分原则,将一个复杂的业务域问题拆分成一个个业务子域,降低系统的复杂度,也能满足其后续的灵活扩展。按照这个思路,我们继续优化,将Predicate 函数独立出来。

代码语言:javascript
复制
// id判断函数
public Predicate<Activity> idPredicate(String id) {
    return activity -> id.equals(activity.getId());
}

// 名称判断函数
public Predicate<Activity> namePredicate(String name) {
    return activity -> name.equals(activity.getName());
}

我们提供了`原子化`函数,具体怎么组装,由上层的调用方根据业务需要自己来拼接。

代码语言:javascript
复制
// 按id查询活动
queryByPredicate(idPredicate(id));
// 按名称查询活动
queryByPredicate(namePredicate(name));
// 按id、名称组合条件查询活动
queryByPredicate(idPredicate(id).and(namePredicate(name)));

画外音:

万事万物,由于都有其个性化特征,如果按常规方式,穷举是很难满足所有业务需求。但如果我们能按其特征抽取,封装组件能力,由流程引擎根据业务诉求,自由组合,则能满足其最大灵活性,更像一个软件高手所为。

本文的代码示例已上传到github https://github.com/aalansehaiyang/project-example

往期推荐

我们热衷于收集&分享高并发、系统架构、微服务、消息中间件、 RPC框架、高性能缓存、搜索、分布式数据框架、分布式协同服务、分布式配置中心、中台架构、领域驱动设计、系统监控、系统稳定性等技术知识

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2020-08-31,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 微观技术 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
云数据库 MySQL
腾讯云数据库 MySQL(TencentDB for MySQL)为用户提供安全可靠,性能卓越、易于维护的企业级云数据库服务。其具备6大企业级特性,包括企业级定制内核、企业级高可用、企业级高可靠、企业级安全、企业级扩展以及企业级智能运维。通过使用腾讯云数据库 MySQL,可实现分钟级别的数据库部署、弹性扩展以及全自动化的运维管理,不仅经济实惠,而且稳定可靠,易于运维。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档