java8_Lambda

Lambda

简洁的表示可传递的匿名函数的一种方式:它没有名称,但它有参数列表,函数主体,返回类型,可能还有一个可以抛出的异常列表。

箭头操作符将lambda分成了两个部分:参数部分->函数体

1.只有一条语句的,大括号和renturn都可以不用写

2.在lambda中的参数列表,可以不用写参数的类型

3.lambda表达式所用的接口,必须是函数式接口

函数式接口简介

链接:(https://blog.csdn.net/qq_28410283/article/details/80962325)

函数式接口定义:接口中只有一个抽象方法的接口,称为函数式接口;使用@FunctionalInterface注解修饰;

举例:无参无返回值的(Runnable接口),有一个参数,无返回值的(Consumer),有多个参数有返回值的(BiFunction)……等等

根据不用的作用,java8中,内置了4个核心接口

1.Consumer 消费型接口,具体用法实例和介绍如下:

Consumer<Integer> consumer = x -> {
            int a = x + 2;
            System.out.println(a);// 12
            System.out.println(a + "_");// 12_
        };
        consumer.accept(10);

2…Supplier 供给型接口,具体用法实例如下

Supplier<String> supplier = String::new;
        System.out.println(supplier.get());//""
        Supplier<Emp> supplierEmp = Emp::new;
        Emp emp = supplierEmp.get();
        emp.setName("dd");
        System.out.println(emp.getName());//dd

3.Function<T,R> 函数型接口,具体介绍如下:

    Function<Integer, Integer> function1 = x -> x * 2;
    System.out.println(function1.apply(4));// 8
 
    Function<Integer, String> function2 = x -> x * 2 + "dd";
    System.out.println(function2.apply(4));//8dd
    
    Function<String, String> strFunction1 = (str) -> new String(str);
    System.out.println(strFunction1.apply("aa"));//aa
    
    Function<String, String> strFunction2 = String::new;
    System.out.println(strFunction2.apply("bb"));//bb
 
    Function<String, Emp> objFunction1 = (str) -> new Emp(str);
    System.out.println(objFunction1.apply("cc").getName());//cc
    
    Function<String, Emp> objFunction2 = Emp::new;
    System.out.println(objFunction2.apply("dd").getName());//dd

Funtion这个接口的“扩展”的原始类型特化的一些函数接口

IntFunction,IntToDoubleFunction,IntToLongFunction,LongFunction,LongToDoubleFunction,LongToIntFunction,DoubleFunction,ToIntFunction,ToDoubleFunction,ToLongFunction
我们在做基础数据处理的时候(eg: Integer i=0; Integer dd= i+1;),会对基础类型的包装类,进行拆箱的操作,转成基本类型,再做运算处理,拆箱和装箱,其实是非常消耗性能的,尤其是在大量数据运算的时候;这些特殊的Function函数式接口,根据不同的类型,避免了拆箱和装箱的操作,从而提高程序的运行效率;

4.Predicate 断言型接口,或者判断型的接口,具体用法如下

public static void main(String[] args) {
		// 数字类型 判断值是否大于5
		Predicate<Integer> predicate = x -> x > 5;
		System.out.println(predicate.test(10));//true

		// 字符串的非空判断
		Predicate<String> predicateStr = x -> null == x || "".equals(x);
		System.out.println(predicateStr.test(""));//true
	 
	}

另外,我还简单列举了一些其他的函数式接口的用法

《 JAVA8 UnaryOperator接口》继承自Function;

UnaryOperator<Integer> dda = x -> x + 1;
		System.out.println(dda.apply(10));// 11
		UnaryOperator<String> ddb = x -> x + 1;
		System.out.println(ddb.apply("aa"));// aa1

《JAVA8 BiConsumer 接口》

@FunctionalInterface
public interface BiConsumer<T, U> {
void accept(T t, U u);
 
/**本接口中的accept先执行,传入的BiConsumer 接口类型的参数,后执行accept*/
default BiConsumer<T, U> andThen(BiConsumer<? super T, ? super U> after) {
    Objects.requireNonNull(after);
 
    return (l, r) -> {
        accept(l, r);
        after.accept(l, r);
    };
}

}

使用这个函数式接口的终端操作有map的遍历,比如;

        map.forEach((k, v) -> {            System.out.println(k);            System.out.println(v);        });

JDK 1.8之前已有的函数式接口:
java.lang.Runnable
java.util.concurrent.Callable
java.security.PrivilegedAction
java.util.Comparator
java.io.FileFilter
java.nio.file.PathMatcher
java.lang.reflect.InvocationHandler
java.beans.PropertyChangeListener
java.awt.event.ActionListener
javax.swing.event.ChangeListener
最后我们列举下java8中 java.util.function包下,内置所有的接口简介和表达的意思

1 BiConsumer<T,U>

代表了一个接受两个输入参数的操作,并且不返回任何结果

2 BiFunction<T,U,R>

代表了一个接受两个输入参数的方法,并且返回一个结果

3 BinaryOperator

代表了一个作用于于两个同类型操作符的操作,并且返回了操作符同类型的结果

4 BiPredicate<T,U>

代表了一个两个参数的boolean值方法

5 BooleanSupplier

代表了boolean值结果的提供方

6 Consumer

代表了接受一个输入参数并且无返回的操作

7 DoubleBinaryOperator

代表了作用于两个double值操作符的操作,并且返回了一个double值的结果。

8 DoubleConsumer

代表一个接受double值参数的操作,并且不返回结果。

9 DoubleFunction

代表接受一个double值参数的方法,并且返回结果

10 DoublePredicate

代表一个拥有double值参数的boolean值方法

11 DoubleSupplier

代表一个double值结构的提供方

12 DoubleToIntFunction

接受一个double类型输入,返回一个int类型结果。

13 DoubleToLongFunction

接受一个double类型输入,返回一个long类型结果

14 DoubleUnaryOperator

接受一个参数同为类型double,返回值类型也为double 。

15 Function<T,R>

接受一个输入参数,返回一个结果。

16 IntBinaryOperator

接受两个参数同为类型int,返回值类型也为int 。

17 IntConsumer

接受一个int类型的输入参数,无返回值 。

18 IntFunction

接受一个int类型输入参数,返回一个结果 。

19 IntPredicate

:接受一个int输入参数,返回一个布尔值的结果。

20 IntSupplier

无参数,返回一个int类型结果。

21 IntToDoubleFunction

接受一个int类型输入,返回一个double类型结果 。

22 IntToLongFunction

接受一个int类型输入,返回一个long类型结果。

23 IntUnaryOperator

接受一个参数同为类型int,返回值类型也为int 。

24 LongBinaryOperator

接受两个参数同为类型long,返回值类型也为long。

25 LongConsumer

接受一个long类型的输入参数,无返回值。

26 LongFunction

接受一个long类型输入参数,返回一个结果。

27 LongPredicate

R接受一个long输入参数,返回一个布尔值类型结果。

28 LongSupplier

无参数,返回一个结果long类型的值。

29 LongToDoubleFunction

接受一个long类型输入,返回一个double类型结果。

30 LongToIntFunction

接受一个long类型输入,返回一个int类型结果。

31 LongUnaryOperator

接受一个参数同为类型long,返回值类型也为long。

32 ObjDoubleConsumer

接受一个object类型和一个double类型的输入参数,无返回值。

33 ObjIntConsumer

接受一个object类型和一个int类型的输入参数,无返回值。

34 ObjLongConsumer

接受一个object类型和一个long类型的输入参数,无返回值。

35 Predicate

接受一个输入参数,返回一个布尔值结果。

36 Supplier

无参数,返回一个结果。

37 ToDoubleBiFunction<T,U>

接受两个输入参数,返回一个double类型结果

38 ToDoubleFunction

接受一个输入参数,返回一个double类型结果

39 ToIntBiFunction<T,U>

接受两个输入参数,返回一个int类型结果。

40 ToIntFunction

接受一个输入参数,返回一个int类型结果。

41 ToLongBiFunction<T,U>

接受两个输入参数,返回一个long类型结果。

42 ToLongFunction

接受一个输入参数,返回一个long类型结果。

43 UnaryOperator
接受一个参数为类型T,返回值类型也为T。

方法的引用

方法的引用的语法,主要有三类

1.指向静态方法的方法引用,例如Integer的parseInt方法 ,可以写成Integer::parseInt

 类::静态方法名

2.指向任意类型实例方法的方法引用,例如String的length方法,写成String::length;

类::实例方法名

3.指向现有对象的实例方法的方法引用

对象::实例方法名

构造器的引用:对于一个现有构造函数,你可以利用它的名称和关键字new来创建它的一个引用ClassName::new;

在java8中的函数式接口,提供了,无参构造函数,以及有参构造函数创建实例的方式;构造器的参数列表,需要与函数式接口中参数列表保持一致!

@Test
	public void test6() {
		/*************** 方法的引用 ****************/
		// 类::静态方法名
		Comparator<Integer> cam1 = (x, y) -> x.compareTo(y);
		System.out.println(cam1.compare(3, 2));
		Comparator<Integer> cam = Integer::compareTo;
		System.out.println(cam.compare(3, 2));
		// 类::实例方法名
		BiPredicate<String, String> bp = (x, y) -> x.equals(y);
		System.out.println(bp.test("a", "b"));
		BiPredicate<String, String> bp1 = String::equals;
		System.out.println(bp1.test("a", "b"));	
// 对象::实例方法名
	Consumer<String> con1 = x -> System.out.println(x);
	con1.accept("abc");
	Consumer<String> con = System.out::println;
	con.accept("abc");
 
	Emp emp = new Emp("上海", "xiaoMIng", 18);
	Supplier<String> supper1 = () -> emp.getAddress();
	System.out.println(supper1.get());
	Supplier<String> supper = emp::getAddress;
	System.out.println(supper.get());
 
	/*************** 构造器的引用 ****************/
	// 无参构造函数,创建实例
	Supplier<Emp> supper2 = () -> new Emp();
	Supplier<Emp> supper3 = Emp::new;
	Emp emp1 = supper3.get();
	emp1.setAddress("上海");
	// 一个参数
	Function<String, Emp> fun = address -> new Emp(address);
	Function<String, Emp> fun1 = Emp::new;
	System.out.println(fun1.apply("beijing"));
	// 两个参数
	BiFunction<String, Integer, Emp> bFun = (name, age) -> new Emp(name, age);
	BiFunction<String, Integer, Emp> bFun1 = Emp::new;
	System.out.println(bFun1.apply("xiaohong", 18));
 
}

最后,特别要注意:

①方法引用所引用的方法的参数列表与返回值类型,需要与函数式接口中抽象方法的参数列表和返回值类型保持一致!

②若Lambda 的参数列表的第一个参数,是实例方法的调用者,第二个参数(或无参)是实例方法的参数时,格式: ClassName::MethodName

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值