JDK1.8 新特性之常用函数式接口

本文详细介绍了Java中常用的四个函数式接口:Supplier提供数据、Consumer消费数据、Function数据转换和Predicate进行判断,通过实例演示了如何在Lambda表达式中灵活运用这些接口。

  

目录

常用4种函数式接口         

Supplier接口

Consumer接口

Function接口

Predicate 接口

        常用的函数式接口主要在java.util.function包下,函数式接口是Lambda表达式使用的前提,而Lambda表达式不关心接口名,抽象方法名,只关心抽象方法的参数列表和返回值类型,因此,为了让Lambda表达式使用更方便,jdk提供了大量函数式接口。(常用的接口:Supplier接口、Consumer接口、Function接口、Predicate接口等)

常用4种函数式接口    
     

        本篇文章只介绍常用的Supplier接口、Consumer接口、Function接口、Predicate接口四种接口。

Supplier接口

@FunctionalInterface
public interface Supplier<T>{ public abstract T get(); }

        对应的Lambda表达式需要“对外提供"一个符合泛型类型的对象数据。供给型接口,通过Supplier接口中的get方法可以得到一个值,无参有返回的接口。

//获取数组中最大的值
public class GetMaxValue {
    public static void main(String[] args) {
        getMax(()->{
            int[] arr = {13,89,45,32,80};
            Arrays.sort(arr);  //排序 
            return  arr[arr.length - 1];  //数组默认升序排列 最后一个就是最大值
        });
    }
    private static void getMax(Supplier<Integer> sup) {
        Integer max = sup.get();
        System.out.println("max : "+max);
    }
}

Consumer接口

@FunctionalInterface
public interface Consumer<T>{
    public abstract void accept<T t);
}

        Consumer<T>接口不是生产一个数据,而是消费一个数据,和Supplier接口相反,其数据类型由泛型参数决定。

//将传入的值转化为大写或小写
public class GetMaxValue {
    public static void main(String[] args) {
       //Consumer将值转化为大写
        changeMax((String s)->{
            System.out.println(s.toUpperCase());//将传入的值转为大写
            System.out.println(s.toLowerCase());//将传入的值转为小写
        });
        
        //Consumer接口中的andThen方法  先打印大写 然后打印小写
        andThenWith(s -> System.out.print(s.toLowerCase() +"  ----  "),s -> System.out.print(s.toUpperCase()));
    }
    private static void changeMax(Consumer<String> con){
        String t = "abcDeH";//传入的值
        con.accept(t);
    }
    
    //Consumer接口中的默认方法andThen
    private  static  void andThenWith(Consumer<String> c1,Consumer<String> c2){
        c2.andThen(c1).accept("HelloWorld");
    }
}

Function接口

@FunctionalInterface
public interface Function<T,R>{
    public abstract R apply(T t);
}

        用来根据一个类型的数据得到另一个类型的数据,前者称为前置条件,后置称为后置条件,有参数有返回值。Function转换型接口,对apply方法传入的T类型数据进行处理,返回R类结果,有参有返回的接口。

//将String类型转换为Integer类型
public class GetMaxValue {
    public static void main(String[] args) {
        //Function接口将值转化为数字
        changeSToI(s -> {
            return Integer.parseInt(s);
        });
        
        //Function接口将字符串转换为数字并乘以10
        funAndThen(s -> {
            //将传入的值转换为int类型
           return  Integer.parseInt(s); 
        },s->{
            //将上一步的结果乘以10返回
            return s * 10;
        });
    }
    //Function接口将一个字符串转化为数字
    private static void changeSToI(Function<String,Integer> fun){
       Integer fun =  fun.apply("58");
       System.out.println(" in "+(fu + 10));
    }
    
    //Function接口中的andThen方法
    private  static  void funAndThen(Function<String,Integer> fun1,Function<Integer,Integer> fun2){
        Integer value = fun1.andThen(fun2).apply("56");
        System.out.println("value : "+value);
    }
}

Predicate 接口

@FunctionalInterface
public interface Predicate<T>{
    public abstract boolean test(T t);
}
//该接口用于做判断,返回boolean类型的值

        有时我们需要对某种数据类型进行判断,从而得到一个boolean值结果,使用Predicate接口即可,该接口用于做判断,返回boolean类型的值。

public class GetMaxValue {
    public static void main(String[] args) {       
        //Predicate接口用于判断 返回Boolean值
        tesetPre(s -> s.length() >= 3,"太极张三丰");
 
        //Predicate接口判断是否包含 and 其他两个值
        testContain( s->
            s.contains("H")
        ,   s->
            s.contains("W")
        ,"HelloWorld");
 
        //Predicate接口包含H或W
        testContain2(s->s.contains("H"),s->s.contains("W"),"HelloBlus");
        //不包含H
        testContain3(s->s.contains("H"),"youyouku");
    }
    //Predicate接口判断返回一个Boolean值
    private static  void tesetPre(Predicate<String> pre,String name){
        boolean bool = pre.test(name);
        System.out.println("如果名字超过三个则名字太长 : "+bool);
    }
 
    //使用Predicate接口and方法判断一个字符串既包含H也包含W
    private  static  void testContain(Predicate<String> pre1,Predicate<String> pre2,String str){
        boolean test = pre1.and(pre2).test(str);
        System.out.println("该值是否既包含其他两值 : "+test);
    }
 
    //使用Predicate接口or方法判断一个字符串包含H或包含W
    private static  void testContain2(Predicate<String> pre1,Predicate<String> pre2,String str){
        boolean test = pre1.or(pre2).test(str);
        System.out.println("包含H或W :" +test);
    }
 
    //使用Predicate中negate默认方法判断字符串不包含
    private  static  void testContain3(Predicate<String> pre1,String str){
        boolean test = pre1.negate().test(str);
        System.out.println("不包含 : "+test);
    }
}


 

JDK 1.8有多个常用新特性,以下是一些介绍: - **Lambda表达式**:这是JDK 1.8的重要特性之一,结合函数式编程,可简化代码,减少样板代码的编写。例如在使用线程时,传统写法是创建一个`Runnable`的匿名内部类,而使用Lambda表达式则可以让代码更加简洁,示例如下: ```java // 传统写法 new Thread(new Runnable() { @Override public void run() { System.out.println("Hello"); } }).start(); // Lambda表达式写法 new Thread(() -> System.out.println("Hello")).start(); ``` - **函数式接口**:函数式接口是只包含一个抽象方法的接口,它是Lambda表达式的基础。Java 8在`java.util.function`包中提供了许多常用函数式接口,如`Predicate`、`Consumer`、`Function`等,方便进行函数式编程。 - **方法引用**:方法引用提供了一种更简洁的方式来调用已存在的方法。例如可以通过`System.out::println`来引用`System.out`对象的`println`方法,示例如下: ```java import java.util.Arrays; import java.util.List; public class MethodReferenceExample { public static void main(String[] args) { List<String> names = Arrays.asList("Alice", "Bob", "Charlie"); names.forEach(System.out::println); } } ``` - **Stream API**:提供了一种声明式的数据处理方式,允许以更简洁、高效的方式操作集合。它具有声明式编程、惰性求值、链式调用、并行处理等特点。例如对集合进行过滤和输出: ```java import java.util.Arrays; import java.util.List; public class StreamExample { public static void main(String[] args) { List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5); numbers.stream() .filter(n -> n % 2 == 0) .forEach(System.out::println); } } ``` - **日期时间组件**:引入了新的日期时间API,如`LocalDate`、`LocalTime`、`LocalDateTime`、`Instant`、`DateTimeFormatter`等,这些类提供了更简洁、更安全的日期和时间处理方式,解决了旧的`Date`和`Calendar`类存在的线程安全等问题。示例如下: ```java import java.time.LocalDate; import java.time.LocalDateTime; import java.time.format.DateTimeFormatter; public class DateTimeExample { public static void main(String[] args) { LocalDate currentDate = LocalDate.now(); System.out.println("Current Date: " + currentDate); LocalDateTime currentDateTime = LocalDateTime.now(); DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); String formattedDateTime = currentDateTime.format(formatter); System.out.println("Formatted DateTime: " + formattedDateTime); } } ``` - **接口的默认方法和静态方法**:传统上,接口主要包含方法签名,不涉及具体实现细节。从Java 8开始,接口引入了默认方法和静态方法,增强了接口的灵活性和功能性。默认方法使用`default`关键字定义,允许接口有具体的方法实现;静态方法使用`static`关键字定义,可直接通过接口名调用。示例如下: ```java interface MyInterface { default void defaultMethod() { System.out.println("This is a default method."); } static void staticMethod() { System.out.println("This is a static method."); } } class MyClass implements MyInterface { // 可以不实现默认方法,因为接口已经提供了实现 } public class InterfaceExample { public static void main(String[] args) { MyClass myClass = new MyClass(); myClass.defaultMethod(); MyInterface.staticMethod(); } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值