Java函数式断言接口Predicate的一个小小实践

开发 后端
最近在搞Excel导入,数据校验是少不了的,但是不同的数据字段有不同的校验策略,五花八门的,甚至不确定,没有办法使用JSR303。所以就搞一个校验策略工具,把校验策略抽象出来。

[[392982]]

最近在搞Excel导入,数据校验是少不了的,但是不同的数据字段有不同的校验策略,五花八门的,甚至不确定,没有办法使用JSR303。所以就搞一个校验策略工具,把校验策略抽象出来。这里尝试了Java 8 提供的一个断言函数接口java.util.function.Predicate,非常给力图片!把校验策略地封装了起来。

Predicate接口

Predicate的应用

先来看看效果:

boolean validated = new Validator<String>() 
            .with(s -> s.length() > 5) 
            .with(s -> s.startsWith("felord")) 
            .with(s -> s.endsWith("cn")) 
            .with(s -> s.contains(".")) 
            .validate("felord.cn"); 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.

我拿校验字符串为例子,通过一系列的Predicate断言来对字符串felord.cn进行校验。这里不局限于String提供的方法,只要你满足 String -> boolean,也就是一个字符串入参能得到一个布尔返回值,都可以作为条件。例如我们在数据库中检索某个字符串,我们可以写一个非常常见的UserService查询用户名是否存在或者有效:

public class UserServiceImpl implements UserService { 
    @Override 
    public boolean checkUserByName(String name) { 
        return false
    } 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.

对应的校验可以改为:

UserServiceImpl userService = new UserServiceImpl(); 
   boolean validated = new Validator<String>() 
           .with(s -> s.length() > 5) 
           .with(s -> s.startsWith("felord")) 
           .with(userService::checkUserByName) 
           .validate("felord.cn"); 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.

好奇的同学该想知道是怎么实现的,Validator源码是这样的:

import java.util.function.Predicate; 
  
public class Validator<T> { 
    /** 
     * 初始化为 true  true &&其它布尔值时由其它布尔值决定真假 
     */ 
    private Predicate<T> predicate = t -> true
 
    /** 
     * 添加一个校验策略,可以无限续杯😀 
     * 
     * @param predicate the predicate 
     * @return the validator 
     */ 
    public Validator<T> with(Predicate<T> predicate) { 
        this.predicate = this.predicate.and(predicate); 
        return this; 
    } 
 
    /** 
     * 执行校验 
     * 
     * @param t the t 
     * @return the boolean 
     */ 
    public boolean validate(T t) { 
        return predicate.test(t); 
    } 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.

逻辑不是很复杂,却可以胜任各种复杂的断言策略组合。接下来我们来对Predicate 一探究竟。

Predicate

@FunctionalInterface 
public interface Predicate<T> { 
 
    /** 
     *  函数接口方法 
     */ 
    boolean test(T t); 
 
    /** 
     * and 默认方法 相当于逻辑运算符 && 
     */ 
    default Predicate<T> and(Predicate<? super T> other) { 
        Objects.requireNonNull(other); 
        return (t) -> test(t) && other.test(t); 
    } 
 
    /** 
     * negate 默认方法 相当于逻辑运算符 !  
     */ 
    default Predicate<T> negate() { 
        return (t) -> !test(t); 
    } 
 
    /** 
     * or 默认方法  相当于逻辑运算符 || 
     */ 
    default Predicate<T> or(Predicate<? super T> other) { 
        Objects.requireNonNull(other); 
        return (t) -> test(t) || other.test(t); 
    } 
 
    /** 
     *  这个方法是提供给{@link Objects#equals(Object, Object)}用的,不开放给开发者 
     */ 
    static <T> Predicate<T> isEqual(Object targetRef) { 
        return (null == targetRef) 
                ? Objects::isNull 
                : object -> targetRef.equals(object); 
    } 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.

断言函数接口提供了test方法供我们开发实现,同时提供了and、negate、or分别对应Java中的逻辑运算符&&、!、||。完全满足了布尔型变量运算,在需要多个条件策略组合时非常有用。

总结

今天通过演示了Predicate的使用,回答了曾经一个同学到底lambda表达式和函数式编程到底如何使用的问题。函数式编程在Java的诞生,“消灭"了很多设计模式,尤其是策略模式。如果你想用好函数式编程就要加强抽象能力,多看看一些框架源码,一定不要强行使用函数式。玩出类似下面这行代码的笑话:

if (Objects.equals(bool,true)){ 
         //TODO 
 } 
  • 1.
  • 2.
  • 3.

本文转载自微信公众号「码农小胖哥」,可以通过以下二维码关注。转载本文请联系码农小胖哥公众号。

 

责任编辑:武晓燕 来源: 码农小胖哥
相关推荐

2021-08-03 07:51:43

Java 8 函数接口

2012-08-23 14:23:33

函数式编程

2023-10-23 14:16:01

Java函数式编程

2015-05-25 15:06:28

JavaScript函数式编程

2024-02-28 08:37:28

Lambda表达式Java函数式接口

2012-05-08 13:14:05

JavaComparable

2024-07-18 08:00:00

2021-02-20 11:34:43

Linux内核指针

2017-04-19 11:22:11

demoPool2Java

2015-10-16 10:19:01

2010-06-22 13:32:26

函数式编程JavaScript

2012-10-22 14:17:42

函数式程序员

2024-12-02 10:56:29

2020-05-25 16:25:17

Java8Stream函数式接口

2021-06-16 17:46:55

函数指针结构

2021-05-21 07:26:15

DataSource接口数据库

2024-02-21 16:13:36

CNCF开源监控工具Prometheus

2020-09-22 07:50:23

API接口业务

2024-02-28 08:12:25

SSE接口代理

2022-01-06 14:59:53

Java框架magic-api
点赞
收藏

51CTO技术栈公众号