函数式编程在Java8中使用Lambda表达式进行开发

事情起因

如果不喜欢看故事的同学,请前往下一章节。

事情起因是我打算集成Redis缓存,配置了Redis以后,发现@Cacheable注解始终不生效,折腾了大半天以后,我的心态已经崩了,决定不使用@Cacheable注解了,自己来设置缓存和读取缓存,当然这篇文章的主题是Lambda表达式,所以缓存的问题一笔带过。在自己获取缓存和设置缓存的时候,发现大量重复的代码,都是一样的if判断语句,所以决定写一个通用的方法把重复的代码干掉,就想到了可以使用函数式编程,Lambda表达式可以精简代码,就分享给大家一下作为参考。


什么是Lambda表达式

Lambda表达式推动Java8发布的最重要新特性。Lambda表达式允许把函数作为一个方法的参数(函数作为参数传递进方法中)。所以你必须升级到 JDK8 以上才可以使用Lambda表达式,如果是 JDK7 的同学就无缘Lambda表达式了。


Lambda表达式的格式

(parameters) -> expression 或 (parameters) ->{ statements; }

圆括号中的参数可以不写类型,编译器可以统一识别参数值;大括号呢在只有一句的情况下可以省略掉大括号


开始使用Lambda表达式

第一步,需要先定义一个函数式接口,例如

@FunctionalInterface
public interface IFunctionObject {
    Object function();
}

所谓的函数式接口,首先是接口,然后就是在这个接口里面只能有一个抽象方法。也称为SAM(Single Abstract Method interfaces)接口。注解@FunctionalInterface用于编译检查,如果不符合函数式接口在编译时就会报错,是一个错误检查的功能。

第二步,在具体业务逻辑中调用函数式接口,例如

/**
 * 从缓存中获取对象,如果没有则运行对象获取接口放入缓存
 *
 * @param key            H
 * @param hashKey        HK
 * @param functionObject 对象获取接口
 * @return 对象
 */
protected Object cacheGet(String key, String hashKey, IFunctionObject functionObject) {
    Object object = null;
    try {
        //从缓存中获取对象
        object = redisTemplate.opsForHash().get(key, hashKey);
        if (object == null) {
            //没有获取到对象,运行对象获取接口放入缓存
            object = functionObject.function();
            if (object != null && object instanceof Serializable)
                cacheSetHash(key, hashKey, object);
        }
    } catch (Exception e) {
        log.error(e.getMessage(), e);
    }
    return object;
}

第三步,使用Lambda表达式调用我们的业务代码,例如:

/**
 * 根据ID获取文章
 *
 * @param id ID编号
 * @return 文章
 */
public Article getArticleById(Long id) {
    //先从缓存中获取,如果没有就从数据库中获取
    Article article = (Article) cacheGet("article", id.toString(), () -> articleMapper.selectByPrimaryKey(id));
    //拿到以后,浏览量加一
    if (article != null) {
        setViewAdd(id);
    }
    return article;
}

其他函数式接口使用举例:下面的方法都可以使用Lambda表达式调用

java.lang.Runnable,

java.awt.event.ActionListener, 

java.util.Comparator,

java.util.concurrent.Callable

java.util.function包下的接口,如Consumer、Predicate、Supplier等

总结

说一下我的理解,Lambda表达式其实就是一个匿名函数,而这个函数可以当做一个参数在方法间传递,可以把函数式接口理解为一个Class类型,Lambda表达式就是这个类型的变量,只不过这个变量比较特殊,他可以运行一些逻辑以后返回结果