java8新特性lambda等

本文详细介绍了Java 8中的Lambda表达式,包括其作用、转换、语法及应用。此外,还阐述了函数式接口的概念及其与Lambda的关系。同时,文章讨论了方法引用和构造器引用的使用。最后,重点讲解了Stream API的基本概念、操作步骤以及在数据处理中的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.lambda表达式

为什么使用Lambda表达式?

lambda表达式是一个匿名函数,我们可以把lambda表达式理解为是一段可以传递的代码,可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使java的语言能力得到了提升。

lambda表达式从匿名类到lambda的转换

//匿名内部类
Runnable r1 = new Runnable(){
    @Override
    public void run(){
        System.out.println("Thanks");
    }
};

//Lambda表达式
Runnable r1 = () -> System.out.println("xxx");
//原来的使用匿名内部类作为参数传递
TreeSet<String> ts = new TreeSet<>(new Comparator<String>(){
    @Overide
    public int compare(String o1, String o2){
        return Integer.compare(o1.length(), o2.length());
    }
});

//lambda表达式作为参数传递
TreeSet<String> ts2 = new TreeSet<>(
    (o1, o2) -> Integer.compare(o1.length(), o2.length())
);

lambda表达式语法

lambda表达式在java语言中引入了一个新的语法元素和操作符。这个操作符为“->”,该操作符被称为lambda操作符或箭头操作符。它将lambda分为两个部分:

左侧:指定了lambda表达式需要的所有参数

右侧:指定了lambda体,即lambda表达式要执行的功能

语法格式:

//语法格式一:无参,无返回值,lambda体只需一条语句
Runnable r1 = () -> System.out.println("hello");

//语法格式二:lambda需要一个参数
Consumer<String> fun = (args) -> System.out.println(args);

//语法格式三:lambda只需要一个参数时,参数的小括号可以忽略
Consumer<String> fun = args -> System.out.println(args)

//语法格式四:lambda需要两个参数,并且有返回值
BinaryOperator<Long> bo = (x, y) -> {
    System.out.println("实现函数接口的方法");
    return x + y;
}

//语法格式五:当lambda体只有一条语句时,return与大括号可以省略
BinaryOperator<Long> bo = (x, y) -> x + y;

//语法格式六:数据类型可以省略,因为可由编译器推断得出,称为类型推断

2.函数式接口

只包含一个抽象方法的接口,称为函数式接口。

可以在函数式接口上使用@FunctionalInterface注解,这样做可以检查它是否为函数式接口。

@FunctionalInterface
public interface MyNumber{
    public double getValue();
}

//函数式接口中使用泛型:
@FunctionalInterface
public interface MyFunc<T>{
    public T getValue(T t);
}

//作为参数传递lambda表达式
public String toUpperString(MyFunc<String> mf, String str){
    return mf.getValue(str);
}

String newStr = toUpperString(
    (str) -> str.toUpperCase(), "abcdef"
)

3.方法引用与构造器引用

方法引用:

当要传递给Lambda体的操作,可以使用方法引用。

使用操作符: " :: " 将方法名和对象或类的名字分隔开来

如下主要三种情况:

对象::实例方法

类::静态方法

类::实例方法
例如:

(x) -> System.out.println(x);

等同于

System.out::println


BinaryOperator<Double> bo = (x,y) -> Math.pow(x, y);
等同于:
BinaryOperator<Double> bo = Math::pow;


compare((x,y) -> x.equals(y), "abcdef", "abcdef");
等同于:
compare(String::equals, "abc","abc");

构造器引用:

格式:   ClassName::new 

与函数式接口相结合,自动与函数式接口中方法兼容。可以把构造器引用赋值给定义的方法,与构造器参数列表要与接口中抽象方法的参数列表一致。

例如:
Function<Integer, MyClass> fun = (n) -> new MyClass(n);
等同于
Function<Integer, MyClass> fun = MyClass::new 

数组引用:

格式:type[] :: new 

例如:

Function<Integer, Integer[]> fun = (n) -> new Integer[n];

等同于:

Function<Integer, Integer[]> fun = Integer[] :: new;

4.Stream API

流(Stream)是数据渠道,用于操作数据源(集合,数组等)所生成的元素序列。集合讲的是数据,流讲的是计算。

注意:
stream自己不会存储元素
stream不会改变源对象。相反,他们会返回一个持有结果的新stream
stream操作是延迟执行的。这意味着它们会等到需要结果时才执行

stream的操作三个步骤

①创建Stream

一个数据源(如:集合、数组),获取一个流。

java8扩展了Collection接口,提供了两个获取流的方法
default Stream<E> stream(): 返回一个顺序流
default Stream<E> parallelStream(): 返回一个并行流


由数组创建流
static <T> Stream<T> stream(T[] array): 返回一个流
重载:
public static IntStream stream(int[] array)
public static LongStream stream(long[] array)
public static DoubleStream stream(double[] array)


由值创建流
可以用静态方法Stream.of(), 通过显示值创建一个流。它可以接收任意数量的参数
public static<T> Stream<T> of(T... values): 返回一个流


创建无限流,由函数创建流
可以使用静态方法Stream.iterate()和Stream.generate(),创建无限流
迭代
public static<T> Stream<T> iterate(final T seed, finalUnaryOperator<T> f)
生成:
public static<T> Stream<T> generate(Supplier<T> s)

②中间操作

一个中间操作链,对数据源的数据进行处理

多个中间操作可以连接起来形成一个流水线,除非流水线上触发终止操作,否则中间操作不会进行任何的处理。而在终止操作时一次性全部处理,称为惰性求值。

③终止操作(终端操作)

一个终止操作,执行中间操作链,并产生结果

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

伊人秋采唐

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值