简单使用lambda表达式
Posted 冷血~多好
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了简单使用lambda表达式相关的知识,希望对你有一定的参考价值。
目录
语法格式四: :Lambda 若只需要一个参数时, 参数的小括号可以省略
语法格式五: :Lambda 需要两个或以上的参数,多条执行语句,并且可以有返回值
语法格式六 :当 当 Lambda 体只有 一条, 语句时,return 号 与大括号 若有,都可以省略
什么是lambda表达式?
Lambda 是一个 匿名函数,我们可以把 Lambda 表达式理解为是 一段可以传递的代码(将代码像数据一样进行传递)。
使用它可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升
Lambda 表达式:在Java 8 语言中引入的一种新的语法元素和操作符。这个操作符为 “->” , 该操作符被称为 Lambda 操作符或 箭 头操作符。
它将 Lambda 分为两个部分:左侧:指定了 Lambda 表达式需要的参 参 数 列表: 右侧:指定了 Lambda 体 ,是抽象方法的实现逻辑,也即Lambda 表达式要执行的功能
下面先来个简单案例:
package com.chen.lambda;
/*
* 推导lambda表达式
* */
public class TestLambda
//3.静态内部类
static class Like2 implements ILike
@Override
public void lambda()
System.out.println("I like lambda2");
public static void main(String[] args)
ILike like=new Like();
like.lambda();
like=new Like2();
like.lambda();
//4.局部内部类
class Like3 implements ILike
@Override
public void lambda()
System.out.println("I like lambda3");
like=new Like3();
like.lambda();
//5.匿名内部类,没有类的名称,必须帮助接口或者父类
like =new ILike()
@Override
public void lambda()
System.out.println("I like lambda4");
;
like.lambda();
//6.用lambda简化
like=()->
System.out.println("I like lambda5");
;
like.lambda();
//1.定义一个函数式接口
interface ILike
void lambda();
//2.实现类
class Like implements ILike
@Override
public void lambda()
System.out.println("I like lambda");
简化lambda表达式:
package com.chen.lambda;
public class TestLambda2
static class Love implements ILove
@Override
public void love(int a)
System.out.println("I Love you -->"+a);
public static void main(String[] args)
ILove love=new Love();
love.love(2);
//匿名内部类
ILove love2=new ILove()
@Override
public void love(int a)
System.out.println("I Love you -->"+a);
;
love2.love(5);
//lambda表达式简化
ILove love3=(int a)->
System.out.println("I Love you -->"+a);
;
love3.love(520);
//简化1.参数类型
ILove love4=(a)->
System.out.println("I Love you -->"+a);
;
love4.love(520);
//简化2.简化括号
ILove love5=a ->
System.out.println("I Love you -->" + a);
;
love5.love(520);
//简化3.去掉花括号
ILove love6=a ->
System.out.println("I Love you -->" + a);
love6.love(520);
//总结:
//lambda表达式只能有一行代码的情况下才能称为一行,如果有多行,那么就用代码块包裹
//前提是接口为函数式接口
//多个参数也可以去掉参数类型,要去掉就都去掉,必须加上括号
ILove2 love7=(a,b) ->
a*b;
System.out.println(love7.love(7,8));
interface ILove
void love(int a);
interface ILove2
int love(int a,int b);
class Love implements ILove
@Override
public void love(int a)
System.out.println("I Love You");
总结:
语法格式一 :无参,无返回值
Runnable runnable1=()->System.out.println("ddd");
语法格式二: :Lambda 需要一个参数,但是没有返回值
Consumer<String> con=(String str) -> System.out.println(str);;
//输出
con.accept("aaa");
语法格式三 : 数据类型可以省略
Consumer<String> con=(str) -> System.out.println(str);;
语法格式四: :Lambda 若只需要一个参数时, 参数的小括号可以省略
Consumer<String> con=str -> System.out.println(str);;
语法格式五: :Lambda 需要两个或以上的参数,多条执行语句,并且可以有返回值
Comparator<Integer> com=(x, y) ->
System.out.println("实现函数式接口方法");
return Integer.compare(x,y);
;
//输出结果
System.out.println(com.compare(11,10));
语法格式六 :当 当 Lambda 体只有 一条, 语句时,return 号 与大括号 若有,都可以省略
Comparator<Integer> com=(x, y) -> Integer.compare(x,y);
上述 Lambda 表达式中的参数类型都是由编译器推断得出的。Lambda
表达式中无需指定类型,程序依然可以编译,这是因为 javac 根据程序
的上下文,在后台推断出了参数的类型。Lambda 表达式的类型依赖于
上下文环境,是由编译器推断出来的。这就是所谓的“类型推断”
然后下面说下函数式接口:
只包含一个抽象方法的接口,称为 函数式接口。
1.你可以通过 Lambda 表达式来创建该接口的对象。(若 Lambda 表达式抛出一个受检异常(即:非运行时异常),那么该异常需要在目标接口的抽象方法上进行声明)。
2.我们可以在一个接口上使用 @FunctionalInterface 注解,这样做可以检查它是否是一个函数式接口。同时 javadoc 也会包含一条声明,说明这个接口是一个函数式接口。
在Java8中,有所不同。在
Java8中,Lambda表达式是对象,而不是函数,它们必须依附于一类特别的对象类型——函数式接口。
简单的说,在Java8中,Lambda表达式就是一个函数式接口的实例。这就是Lambda表达式和函数式接口的关系。也就是说,只要一个对象是函数式接口的实例,那么该对象就可以用Lambda表达式来表示
所以以前用匿名实现类表示的现在都可以用Lambda表达式来写
简单使用lambda表达式使用四大函数式接口:
代码案例:
//Consumer<T> 消费型接口:
@Test
public void test1()
happy(10000,(m) -> System.out.println("今天消费了:"+(m+m)));
public void happy(double money, Consumer<Double> con)
con.accept(money);
//Supplier 供给型接口:
@Test
public void test2()
List<Integer> numList= getNumList(10,() ->(int)(Math.random()*100));
for(Integer num:numList)
System.out.println(num);
//需求:产生指定个整数,并放入到集合中
public List<Integer> getNumList(int num, Supplier<Integer> sup)
List<Integer> list=new ArrayList<>();
for(int i=0;i<num;i++)
Integer n=sup.get();
list.add(n);
return list;
//Function<T ,R> 函数型接口
@Test
public void test3()
String newStr = strHandler("\\t\\t\\t 我爱爱学习",(str) -> str.trim());
System.out.println(newStr);
String subStr = strHandler("\\t\\t\\t 我爱爱学习",(str)-> str.substring(2,5));
System.out.println(subStr);
//需求:用于处理字符串
public String strHandler(String str, Function<String,String> fun)
return fun.apply(str);
//Predicate<T> 断言型接口
@Test
public void test4()
List<java.lang.String> list= Arrays.asList("hello","hello2","hello3");
List<String> strList=filterStr(list,(s) -> s.length()>5);
for(String str:strList)
System.out.println(str);
//需求:将满足条件的字符串,放入集合中
public List<String> filterStr(List<String> list, Predicate<String> pre)
List<String> strList=new ArrayList<>();
for(String str:list)
if(pre.test(str))
strList.add(str);
return strList;
方法引用与构造器引用
方法引用
当要传递给Lambda体的操作,已经有实现的方法了,可以使用方法引用!
1.方法引用可以看做是Lambda表达式深层次的表达。换句话说,方法引用就是Lambda表达式,也就是函数式接口的一个实例,通过方法的名字来指向
一个方法,可以认为是Lambda表达式的一个语法糖。
2.要求:实现接口的抽象方法的参数列表和返回值类型,必须与方法引用的方法的参数列表和返回值类型保持一致!3.格式:使用操作符 “::” 将类(或对象) 与 方法名分隔开来。
如下三种主要使用情况:
对象:: 实例方法名
类 :: 静态方法名
类 :: 实例方法
例如:
Consumer<String> con=(x) -> System.out.println(x);
//相当于
Consumer<String> con=System.out::println;
Comparator<Integer> c=(x,y) -> Integer.compare(x,y);
//相当于
Comparator<Integer> c2=Integer::compare;
BiPredicate<String,String> bp=(x,y) -> x.equals(y);
//相当于
BiPredicate<String,String> bp1=String::equals;
//输出结果
System.out.println(bp1.test("s","t"));
构造器 引用
格式: ClassName::new
与函数式接口相结合,自动与函数式接口中方法兼容。
可以把构造器引用赋值给定义的方法,要求构造器参数列表要与接口中抽象
方法的参数列表一致!且方法的返回值即为构造器对应类的对象
Function<Integer,Myclass> fun=(n) -> new Myclass(n);
//相当于
Function<Integer,Myclass> fun=Myclass::new;
数组引用
Function<Integer,Integer[]> fun3=(n) -> new Integer[n];
//相当于
Function<Integer,Integer[]> fun4=Integer[]::new;
System.out.println(fun4.apply(1555).length);
以上是关于简单使用lambda表达式的主要内容,如果未能解决你的问题,请参考以下文章