JDK8 一文搞定👍
! https://zhuanlan.zhihu.com/p/442182870
Java8 新特性
学习来源于 B站 尚硅谷yyds
Java学习源码
2021/11/22
- 距离,过年还有 57 天,想家~
- JDK1.8已经发布很久了,在很多企业中都已经在使用
- 虽然,JDK是向下兼容的不会
新特性
也一样可以正常开发,但是作为程序员还是要不断更新新的技术.不要求啥都会,但要求能看懂!
Java 8 是oracle公司于2014年3月发布
- 是自Java 5 以 来最具革命性的版本
- Java 8为Java语言:
编译器、类库、开发 工具与JVM带来了大量新特性.
Lambda表达式
函数式编程
Lambda表达式,最早是Python 的语法,简洁优美,一行代码就是一个方法~
但,说实话可读性 并不是很好
第一次看到这个时候我都懵了. 为了不被同事嘲讽,连夜学习了JDK8Lambda 表达式,也可称为
闭包
Java 8 发布的最重要新特性
闭包:
闭包,就是能够读取其他函数内部变量的函数,例如JS中,只有函数内部的子函数才能读取局部变量
所以:闭包,可以理解成 “定义在一个函数内部的函数“
Lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中
基本语法:
/**Lambda语法:*/
(参数列表) -> { 代码块 }
/**说明:*/
/*Lambda表达式是前提基于:函数式接口来实现的: 只包含一个抽象方法的接口,称为函数式接口 */
左侧:指定了 Lambda 表达式需要的参数列表
右侧:指定了 Lambda 体,是抽象方法的实现逻辑,也即Lambda 表达式要执行的功能
Lambda实例:
LambdaTest.Java
- 以 Runnable接口举例: lambda表达式, 就是
匿名实现类
的,另一种优化声明方式:
@Test
public void test1(){
/** JDK8之前,定义Runnable 接口实现多线程.*/
//1. 类——>实现Runnable接口,创建实例...填入Thread(Runnable r); .start(); 启动线程
//2. 匿名内部类方式,获取Runnable 接口实例: 创建一个接口实例
Runnable runnable = new Runnable() {
@Override
public void run() {
System.out.println("匿名内部类实现Runnable 接口实例");
}
};
Thread thread1 = new Thread(runnable);
thread1.start();
/**JDK8之后,定义Runnable Lambda接口实现多线程.*/
//Lambda
// ->左侧: 指定了 Lambda 表达式需要的参数列表, 这里参数列表是空
// ->右侧: 指定了 Lambda 体,是抽象方法的实现逻辑,也即Lambda 表达式要执行的功能;
Runnable runnable2 = () -> { System.out.println("Lambda实现Runnable 接口实例");};
Thread thread2 = new Thread(runnable2);
thread2.start();
//Lambda 优化:
// lambda形参列表的参数类型可以省略(类型推断), 如果lambda形参列表只有一个参数, 其一对()也可以省略
// lambda体应该使用一对{}包裹, 如果lambda体只有一条执行语句,可能是return语句, 可以省略这一对{}和return关键字.
Runnable runnable3 = () -> System.out.println("Lambda优化实现Runnable 接口实例");
Thread thread3 = new Thread(runnable3);
thread3.start();
System.out.println();
System.out.println("创建的 Runnable接口实例,正常使用!");
}
控制台结果集:
匿名内部类实现Runnable 接口实例
Lambda实现Runnable 接口实例
创建的 Runnable接口实例,正常使用!
Lambda优化实现Runnable 接口实例
练习:LambdaTest.Java
@Test
public void test2(){
/** 练习: Comparator 定制排序
* Comparator接口,也是一个 "函数式接口": 只包含一个抽象方法的接口,称为函数式接口
*/
/** JDK8之前,定义Runnable 接口实现多线程.*/
System.out.println("自然排序/定制排序: 比较基本/引用数据类型,A>B=1 A<B=-1 A==B=0");
Comparator<Integer> com1 = new Comparator<Integer>() {
@Override
public int compare(Integer o1, Integer o2) {
return Integer.compare(o1,o2);
}
};
System.out.println("定制排序1: "+com1.compare(1, 2));
/**JDK8之后,定义Runnable Lambda接口实现多线程.*/
//Lambda
Comparator<Integer> com2 = (Integer o1, Integer o2) -> { return Integer.compare(o1,o2); };
System.out.println("定制排序2: "+com2.compare(3, 2));
//Lambda 优化:
//类型推断: 省略类型
//如果lambda体只有一条执行语句,可能是return语句, 可以省略这一对{}和return关键字.
Comparator<Integer> com3 = (o1,o2) -> Integer.compare(o1,o2);
System.out.println("定制排序3: "+com3.compare(4, 3));
/** 方法引用 */
Comparator<Integer> com4 = Integer :: compare; //后面介绍
System.out.println("定制排序4: "+com4.compare(2, 2));
}
自然排序/定制排序: 比较基本/引用数据类型A>B=1 A<B=-1 A==B=0
定制排序1: -1
定制排序2: 1
定制排序3: 1
定制排序4: 0
总结:
Lambda表达式:依赖于函数式接口, 是对函数式接口的,另一种:实例化形式~
更简洁,难懂
->左侧: 指定了 Lambda 表达式需要的参数列表
- lambda形参列表的参数类型可以省略
(类型推断)
- 如果,lambda形参列表只有一个参数, 其一对
()也可以省略
->右侧: 指定了 Lambda 体,是抽象方法的实现逻辑,也即Lambda 表达式要执行的功能
- lambda体应该使用一对{}包裹
- 如果lambda体只有一条执行语句,可能是return语句, 可以省略这一对{}和return关键字.
省略return时候也要省略 {}
场景:
- 可以在,将
函数式接口的实例作
自定义方法的入参进行传递,完成一些方法内部方便操作... - 直接定义函数式接口,调用内部的方法完成某些操作~
函数式接口:
只包含一个抽象方法的接口,称为函数式接口
JDK8.0 可以通过Lambda表达式,来创建该接口的对象~
所有的 函数式接口 都可以通过Lambda表达式类进行 实例化~
@FunctionalInterface 注解
JDK8.0提供一个注解来标识管理:
该注解, 可以检 查它是否是一个函数式接口,同时 Javadoc 也会包含一条声明,说明这个 接口是一个函数式接口.
JDK8
Java.util.function包下
定义了Java 8 的丰富的函数式接口为了方便不同情况的,lambda表达式的使用场景~
函数式接口实例:
Runnable接口举例:ctrl+单机
进入源码:
- 一个接口,只有一个
abstract抽象方法
,@FunctionalInterface 注解
修饰.
自定义函数式接口:
WsmInterface.Java
/** 注解可以省略,没有影响,注解只是对程序编写的一个提醒标识~
* 提示你: 这是一个函数式接口! */
@FunctionalInterface
public interface WsmInterface {
public abstract void show();
}
Java.util.function 包:
JDK8.0 之后专门为了,Lambda 不同场景提供的不同的函数式接口
Java 内置四大核心函数式接口:
函数式接口 | 参数类型 | 返回类型 | 用途 |
---|---|---|---|
Consumer< T > 消费型接口 |
T | void | 对类型为T的对象应用操作: void accept(T t) |
Supplier< T > 供给型接口 |
无 | T | 返回类型为T的对象: T get() |
Function<T, R> 函数型接口 |
T | R | 对类型为T的对象应用操作,并返回是R类型的对象. R apply(T t) |
Predicate< T > 断定型接口 |
T | Boolean | 确定类型为T的对象是否满足某约束, 返回boolean 值 boolean test(T t) |
其他接口:
四大核心函数式接口:
LambdaTest2.Java
import org.junit.Test;
import java.util.function.Consumer;
/** 四大核心函数式接口Function */
public class LambdaTest2 {
/** 正常情况下,函数式接口 实例,当作方法参数传递在方法中完成事情~ */
/** 消费型接口 Consumer<T> void accept(T t) */
/** ① 声明一个方法传入Consumer<T> 对象实例: */
public void con(Double money, Consumer<Double> con){ //<T> 泛型规范了传入的类型~ Double~
/** ②方法内,使用Consumer<T>类型参数,调用,它对应的方法,还有自己方法内部的一下操作~ */
System.out.println("con方法调用~");
con.accept(money);
System.out.println("");
};
/** ③实现 */
/** J8前 */
@Test
public void ConsumerTest(){
//要传入的参数!
Double dd = 540.0;
System.out.println("调用 con(Double,Consumer<Double>) 方法");
//方式一 创建函数式接口的对象,传入接口的实例: (创建方式,匿名内部类~
Consumer<Double> con1 = new Consumer<Double>() {
@Override
public void accept(Double aDouble) {
System.out.println("接口实例类,参数传递实现: 身上还有"+aDouble+"块钱!");
}
};
//调用方法
this.con(dd,con1);
//方式二: 参数匿名内部类实现
this.con(dd, new Consumer<Double>() {
@Override
public void accept(Double adouble) {
System.out.println("匿名内部类实现: 身上还有"+adouble+"块钱!");
}
});
}
/** J8后 */
@Test
public void ConsumerTest2(){
//要传入的参数!
Double dd = 540.0;
//JDK8 后Lambda表达式,对 参数匿名内部类 的升级
/** 调用 con(Double,Consumer<Double>) 方法 */
this.con(dd, (adouble) -> System.out.println("Lambda表达式实现: 身上还有"+adouble+"块钱!") );
}
}
# ConsumerTest 运行
调用 con(Double,Consumer<Double>) 方法
con方法调用~
接口实例类,参数传递实现: 身上还有540.0块钱!
con方法调用~
匿名内部类实现: 身上还有540.0块钱!
# ConsumerTest2 运行
con方法调用~
Lambda表达式实现: 身上还有540.0块钱!
Java.util.function 包下就是,JDK8为了方便用户操作,二提供的一系列的
函数式接口
Consumer< T >
就是一种
函数式接口
,可以 定义一个方法,使用该类型Consumer<T>
作为参数进行方法实现... 完成一些操作.
方法/构造器/数组 引用:
一种更加 高级
的Lambda表达式 的表现形式
本质上就是一种Lambda表达式的 “语法糖”
当要传递给Lambda体的操作,已经有实现的方法了,可以使用方法引用
要求:
实现接口的抽象方法的
参数列表
和返回值
类型,必须与方法引用的方法的参数列表和返回值类型保持一致!语法格式:
使用操作符
::
将类(或对象) 与 方法名分隔开来三种场景:
对象 :: 实例方法名
类 :: 静态方法名
类 :: 实例方法名
总结:
实现接口的抽象方法的 参数列表
和返回值
类型, **须与方法引用的方法的参数列表和返回值类型保持一致!
也可以理解,方法引用就是更加,简化了Lambda的操作,如果有一个类已经实现了 “函数式接口的方法”`假设有一个函数式接口A 函数式接口A 内部方法 af(); lambda表达式创建 A 的实例 A a = () -> { 方法内部的操作... } 如果 函数式接口 内部的方法,已经有一个B类bf()对其进行了实现,则可以在方法内部直接通过: B对象.bf(); 一行完成方法的调用 A a = () -> { B b = new B(); b.bf(); }方法引用 简化升级: B b = new B(); 创建B类的 对象; A a = b::bf; 放法引用的 对象::实例方法名 引用; 省略方法的参数列表...
注意:
实现接口的抽象方法的
参数列表
和返回值
类型,必须与方法引用的方法的参数列表和返回值类型保持一致!这样:
对象/类 :: 方法名
后面不需要跟着 (参数列表) ,因为函数式接口的方法, 和 实现类方法的 “参数列表一致可以省略...”
对象 :: 实例方法名
com.wsm.met 包下:
A 接口
/** 自定义函数式: */@FunctionalInterfacepublic interface A { //上面实例是一个 无参, 这里定义一个有参的方法(); public void af(int i);}
B类 实现
/** 自定义类,实现函数式接口 */public class B { //方法参数列表,与函数式接口相同~ public void bf(int i){ System.out.println("对象::实例方法引用 参数列表i="+i); }}
MethodTest类 方法引用测试类
import org.junit.Test;/** 方法引用测试类 */public class MethodTest {/** JDK1.8之前创建A接口实例: */ @Test public void Test(){ A a = new A() { @Override public void af(int i) { System.out.println("JDK8之前实现接口~ 参数列表i="+i); } }; a.af(1); }/** Lambda表达式创建A接口实例: */ @Test public void Test2(){ //Lambda表示式实现A接口: A a = i -> System.out.println("Lambda表达式实现接口 参数列表i="+i); int i = 2; a.af(i); }/** 对象 :: 非静态方法 */ /** 升级Lambda表达式: 方法引用 */ @Test public void Test3(){ //① 创建B 类对象; B b = new B(); //② 方法引用: A a = b::bf; //③ 调用方法~ a.af(3); /** 因为 af(i) 和 bf(i) 方法实现`af实现的操作bf已经完成了` 返回值 参数列表相同~ */ /** 则满足方法引用,直接使用! */ }}
cmd
TestJDK8之前实现接口~ 参数列表i=1Test2Lambda表达式实现接口 参数列表i=2Test3对象::实例方法引用 参数列表i=3
类 :: 静态方法名
对象 :: 实例方法名 Demo测试扩展~
B类 扩展
//类的静态方法, bsf(int i);public static void bsf(int i){ System.out.println("类::static静态方法引用 参数列表i="+i);}
MethodTest类 扩展
/** 类 :: 静态方法 */@Testpublic void Test4(){ System.out.println("Test4"); //① 方法引用: A a = B::bsf; //直接通过 类::匹配的静态方法()~ //② 调用方法~ a.af(4);}
cmd
Test4类::static静态方法引用 参数列表i=4
总结
对象 :: 非静态方法
和类 :: 静态方法
- 实现都类似, 一个通过
对象.实例方法~
一个 通过类.静态方法
而,类.实例方法
有点不同~
类 :: 实例方法名
MethodTest.Java
/** 类 :: 实例方法 (有难度) */ /** 以Comparator 方法举例: int comapre(T t1,T t2) 方法 */ /** String中的int t1.compareTo(t2) */ @Test public void Test5(){ System.out.println("Test5"); /** lambda表达式实现 */ Comparator<String> com1 = (s1, s2) -> s1.compareTo(s2); System.out.println("lambda比较两个字符串大小~"+com1.compare("abc","abd")); Comparator<String> com2 = String :: compareTo; System.out.println("方法引用: 类 :: 实例方法 比较两个字符串大小~"+com2.compare("abc","abd")); /** * 如果: * A 函数式接口的的实现来源于~ * af(T1,T2); T1类.方法(T2); 实现,则属于 类::实例方法; 的方法引用~ * */ }
cmd
Test5lambda比较两个字符串大小~-1方法引用: 类 :: 实例方法 比较两个字符串大小~-1
类::实例方法
的方法引用, 需要通过, 函数式接口的方法(T1,T2) 参数列表: T1类型.实例方法(T2参数); 完成“函数式接口的实现!”
构造器引用
B.Java 扩展
public String name = "default";public B(){ System.out.println("B() 无参构造");}public B(String name) { System.out.println("B(name,age) 有参构造"); this.name = name;}//省略get/set
ConstructorRefTest.Java
/** 构造器引用 */ //JDK8提供的函数式接口 Supplier中的T get() 返回一个 T 类型对象 //B 类的空参构造器: B() public void Test(){ /** JDK8之前 */ Supplier<B> supB = new Supplier<B>() { @Override public B get() { return new B(); }; }; System.out.println("JDK8之前"); System.out.println("无参默认创建的对象: supB.get().getName();"+supB.get().getName()); /** Lambda构造器引用 */ Supplier<B> supB1 = () -> new B(); System.out.println("Lambda创建的对象: supB.get().getName();"+supB1.get().getName()); Supplier<B> supB2 = B :: new; System.out.println("构造器引用创建的对象: supB.get().getName();"+supB2.get().getName()); }/** 构造器引用2.0 */ //Function中的R apply(T t); Function<T, R>函数式接口,对类型为T的对象应用操作,并返回是R类型的对象 @Test public void Test2(){ System.out.println("Lambda创建的对象:"); Function<String ,B> func1 = name -> new B(name); System.out.println("Lambda创建的对象: supB.get().getName()= "+func1.apply("wsm1").getName()); System.out.println("*******\n"); System.out.println("构造器引用创建的对象:"); Function<String ,B> func2 = B :: new; //参数自动匹配~ System.out.println("Lambda创建的对象: supB.get().getName()= "+func2.apply("wsm2").getName()); }
CMD
TestJDK8之前B() 无参构造无参默认创建的对象: supB.get().getName();defaultB() 无参构造Lambda创建的对象: supB.get().getName();defaultB() 无参构造构造器引用创建的对象: supB.get().getName();defaultTest2Lambda创建的对象:B(name,age) 有参构造Lambda创建的对象: supB.get().getName()= wsm1*******构造器引用创建的对象:B(name,age) 有参构造Lambda创建的对象: supB.get().getName()= wsm2
总结:
构造器引用,就是
与函数式接口相结合,自动与函数式接口中方法兼容
可以把构造器引用赋值给定义的方法
注意:
- 要求构造器参数列表要与接口中抽象 方法的参数列表一致!
且方法的返回值即为构造器对应类的对象
格式: ClassName::new
数组引用:
与构造器引用类似 不详细介绍了...
@Testpublic void Test3(){ Function<Integer,String[]> func1 = length -> new String[length]; String[] arr1 = func1.apply(5); System.out.println(Arrays.toString(arr1)); System.out.println("*******************"); Function<Integer,String[]> func2 = String[] :: new; String[] arr2 = func2.apply(10); System.out.println(Arrays.toString(arr2));}
Stream API
java.util.stream包下
Java8中有两大最为重要的改变。第一个是 Lambda 表达式;另外一个则 是 Stream API
Stream API 把真正的函数式编程风格引入到Java中
这是目前为止对Java类库最好的补充
Stream API可以极大提供Java程 序员的生产力,让程序员写出高效率、干净、简洁的代码.
Stream 是 Java8 中处理集合的关键抽象概念
它可以指定你希望对集合进 行的操作,可以执行非常复杂的查找、过滤和映射数据等操作
Stream API 对集合数据进行操作,就类似于使用 SQL 执行的数据库查询,Stream API 提供了一种 高效且易于使用的处理数据的方式
为什么要使用Stream API:
- 实际开发中,项目中多数数据源都来自于Mysql,Oracle等
关系性数据库
- 而对于 Redsi MongDB
非关系性数据库
并不能提供,复杂性查询操作:过滤 分组 计算...
而这些NoSQL的数据就需要 Java层面去处理
很麻烦
Stream 和 Collection 集合的区别:
- Collection 是一种静态的内存数据结构
基于内存的存储数据的空间
- 而 Stream 是有关计算的
CPU计算~
Stream 的操作三个步骤
创建 Stream
一个数据源
如:集合、数组
,获取一个流①Stream 自己不会存储元素
②Stream 不会改变源对象。相反,他们会返回一个持有结果的新Stream
③Stream 操作是延迟执行的。这意味着他们会等到需要结果的时候才执行
中间操作
- 一个中间操作链,对数据源的数据进行处理
终止操作(终端操作)
- 一旦执行终止操作,就执行中间操作链,并产生结果。
之后的Stream对象,不会再被使用
创建 Stream
Emp.Java 自定义操作实体类:
/** 自定义一个实体类: */public class Emp { private int id; private String name; private int age; private double salary; //无参构造 public Emp() { } //有参构造 public Emp(int id, String name, int age, double salary) { this.id = id; this.name = name; this.age = age; this.salary = salary; } //自定义操作数据集, `模拟从 Redis 数据库获取到的数据集合!` public static List<Emp> getEmployees(){ List<Emp> list = new ArrayList<>(); list.add(new Emp(1001, "马化腾", 34, 6000.38)); list.add(new Emp(1001, "马化腾", 34, 6000.38)); list.add(new Emp(1002, "马云", 12, 9876.12)); list.add(new Emp(1002, "马云", 12, 9876.12)); list.add(new Emp(1003, "刘强东", 33, 3000.82)); list.add(new Emp(1004, "雷军", 26, 7657.37)); list.add(new Emp(1005, "李彦宏", 65, 5555.32)); list.add(new Emp(1006, "比尔盖茨", 42, 9500.43)); list.add(new Emp(1007, "任正非", 26, 4333.32)); list.add(new Emp(1008, "扎克伯格", 35, 2500.32)); return list; } //重新toString(); @Override public String toString() { return "Emp{" + "id=" + id + ", name='" + name + '\'' + ", age=" + age + ", salary=" + salary + '}'; } //省略get/set...}
StreamTest.Java 测试类:创建Stream
import java.util.Arrays;import java.util.List;import java.util.stream.IntStream;import java.util.stream.Stream;/** 强大的Stream Api */public class StreamTest {/** 创建 Stream */ @Test public void test1(){ /** 方式一: 通过集合: 集合.stream/parallelStream(); 返回一个stream顺序流/并行流~ */ List<Emp> emps = Emp.getEmployees(); //default Stream<E> stream() : 返回一个顺序流 Stream<Emp> stream = emps.stream(); //default Stream<E> parallelStream() : 返回一个并行流 Stream<Emp> parallelStream = emps.parallelStream(); /** Java8 Collection接口添加了新的方法 stream()、parallelStream()、forEach()和removeIf()... */ /** 方式二: 通过数组: Arrays.stream(数组); 返回对应的Stream流对象! */ int[] arr = new int[]{1,2,3,4,5,6}; //调用Arrays类的static <T> Stream<T> stream(T[] array): 返回一个流 IntStream intstream = Arrays.stream(arr); /** 基本数据类型返回,对应的 xxxstream Stream流对象~ */ Emp e1 = new Emp(1001,"Tom",34, 6000.38); Emp e2 = new Emp(1002,"Jerry",34, 6000.38); Emp[] arr1 = new Emp[]{e1,e2}; Stream<Emp> empStream = Arrays.stream(arr1); /** 自定类型数组,返回 Stream<自定义类型> 的Sream类型对象! */ /** 方式三: 通过Stream的of() */ Stream<Integer> stream3 = Stream.of(1, 2, 3, 4, 5, 6); /** 用的少了解即可~ */ } /** 方式四: 创建无限流: 通过Stream类的静态方法 iterate()迭代 generate()生成 */ public void test2(){ /** 两种方式: 迭代流 生成流 */// 迭代// public static<T> Stream<T> iterate(final T seed, final UnaryOperator<T> f) //参数 T , UnaryOperator函数式接口(内部方法: 传入参数T类型,返回T类型结果,正好用于对T的每次迭代的改变~) //Stream.iterate(0, t -> t + 2).forEach(System.out::println); // 注释原因: 迭代流会无限的迭代下去 +2 +2 +2... // .limit(10) 之取前十个数据 // .forEach() 结束操作! 结束时候做的事情~ //遍历前10个偶数 Stream.iterate(0, t -> t + 2).limit(10).forEach(System.out::println);// 生成// public static<T> Stream<T> generate(Supplier<T> s) //参数 Supplier函数式接口对象, 内部方法, 返回一个T类型对象... 可以根据后一个规则无限的来生成一些数据~ Stream.generate(Math::random).limit(10).forEach(System.out::println); /** 生成10个小于 1 随机数! */ }}
总结:
创建Stream 一共有四种方式:
集合.stream/parallelStream(); 返回一个stream顺序流/并行流~
顺序流:
使用主线程,单线程,顺序执行~
并行流:
.parallel()可以将其修改成并行流,内部以多线程并行执行任务的方式执行~
通过数组: Arrays.stream(数组); 返回对应的Stream流对象!
基本数据类型返回,对应的 xxxstream Stream流对象~
自定类型数组,返回 Stream<自定义类型> 的Stream类型对象!
通过Stream的of()
使用的少,类似于数组创建Stream流~
创建无限流: 通过Stream类的静态方法 iterate()迭代 generate()生成
中间操作
创建完Stream流对象之后,就可以通过 流对象S.xx().xx().xx() 各种的中间操作,完成对 流种数据的计算: 筛选 切片 映射 排序...等操作
- 中间操作, 是多个方法, 每个方法可以对流中的数据进行筛选计算~
- 多个方法可以像链条一样 拼接操作~
中间操作,方法描述
方 法( ); | 描 述: |
---|---|
筛选与切片 | |
filter(Predicate p) | 接收 Lambda , 从流中排除某些元素,传入一共 函数式接口 (方法参数,传入一个 T 返回Boolean结果) |
distinct() | 筛选,对流中元素进行 hashCode() 和 equals() 去除重复元素. |
limit(long maxSize) | 截断流,使其元素不超过给定数量 |
skip(long n) | 跳过元素,返回一个扔掉了前 n 个元素的流。若流中元素不足 n 个,则返回一 个空流。与 limit(n) 互补 |
映 射 | |
map(Function f) | 接收一个函数作为参数,该函数会被应用到每个元 素上,并将其映射成一个新的元素 |
flatMap(Function f) | 接收一个函数作为参数,将流中的每个值都换成另 一个流,然后把所有流连接成一个流 |
mapToInt(ToIntFunction f) | 接收一个函数作为参数,该函数会被应用到每个元 素上,产生一个新的 IntStream |
mapToLong(ToLongFunctionf) | 接收一个函数作为参数,该函数会被应用到每个元 素上,产生一个新的 LongStream |
mapToDouble(ToDoubleFunction f) | 接收一个函数作为参数,该函数会被应用到每个元 素上,产生一个新的 DoubleStream |
排 序 | |
sorted() | 产生一个新流,其中按自然顺序排序 |
sorted(Comparator com) | 产生一个新流,其中按比较器顺序排序 |
示例: Demo
StreamTest2.Java
import org.junit.Test;import java.util.ArrayList;import java.util.Arrays;import java.util.List;import java.util.stream.Stream;/** Stream中间操作 * 创建完Stream流对象之后,就可以通过 流对象S.xx().xx().xx() 各种的中间操作,完成对 流种数据的计算: 筛选 切片 映射 排序...等操作 * 中间操作, 是多个方法, 每个方法可以对流中的数据进行筛选计算~ * 多个方法可以像链条一样 拼接操作~ * */public class StreamTest2 { /** 筛选与切片 */ @Test public void test1(){ System.out.println("筛选与切片"); //获取 emp 集合~ List<Emp> employees = Emp.getEmployees(); //JDK8 Collection接口新增,foreach(); 方法: 遍历结果集操作~ employees.forEach(System.out::println); //集合创建Stream 流~ Stream<Emp> stream = employees.stream(); System.out.println("\nfilter: 从流中排除某些元素"); System.out.println("练习: 查询员工表中薪资大于7000的员工信息"); stream.filter(e->e.getSalary()>7000).forEach(System.out::println); //filter(Predicate<T>); lambda表达式,每次传入一个流中对象,返回一共 Boolean结果,过滤掉false数据! /** 注意一共Stream 使用之后就不可以在次使用,流已经关闭了... 但创建流的,集合依然存在~ */// stream.filter(e->e.getSalary()>7000).forEach(System.out::println); //异常:stream has already been operated upon or closed 需要重新创建一共新的流~ System.out.println("\ndistinct:筛选,对流中元素进行 hashCode() 和 equals() 去除重复元素"); employees.stream().distinct().forEach(System.out::println); System.out.println("\nlimit(n)——截断流,使其元素不超过给定数量(获取前几个元素~)"); employees.stream().limit(3).forEach(System.out::println); System.out.println("\nskip(n) —— 跳过元素(跳过前几个元素不要~)"); employees.stream().skip(3).forEach(System.out::println); } /** **映 射** */ @Test public void test2(){ System.out.println("映射"); System.out.println("\nmap(Function f): ——接收一个函数作为参数,将元素转换成其他形式或提取信息,该函数会被应用到每个元素上,并将其映射成一个新的元素"); List<String> strlist = Arrays.asList("aa", "bb", "cc", "dd"); strlist.stream().map(str -> str.toUpperCase()).forEach(System.out::println); //遍历每一个元素,对元素进行二次操作~ System.out.println("\n过年了,给所有员工工资+1000"); //获取 emp 集合~ List<Emp> employees = Emp.getEmployees(); System.out.println("遍历一遍集合:"); employees.stream().forEach(System.out::println); System.out.println("\n加薪: map( 内部函数式接口方法,要求传入什么参数类型,则返回什么类型~ ); 顺便去重 distinct()"); employees.stream().map(emp -> {emp.setSalary(emp.getSalary()+1000.0); return emp; }).distinct().forEach(System.out::println); System.out.println("\nflatMap(f); 是map(f)的高级版~"); System.out.println("1.Map(f) 实现遍历每一个strlist 的字符"); Stream<Stream<Character>> streamStream = strlist.stream().map(StreamTest2::fromStringToStream); streamStream.forEach(s ->{ s.forEach(System.out::println); }); System.out.println("2.flatMap(f) 实现遍历每一个strlist 的字符"); Stream<Character> characterStream = strlist.stream().flatMap(StreamTest2::fromStringToStream); characterStream.forEach(System.out::println); System.out.println("flatMap 会将,内部的每一个元素进行操作,如果是Stream元素也会重新拆开执行~"); System.out.println("\n下面的映射,就不详细介绍了: mapToInt(T) mapToLong(T) mapToDouble(T) 传入泛型T 返回对应的类似数据~"); } /** 将字符串中的多个字符构成的集合转换为对应的Stream的实例 */ public static Stream<Character> fromStringToStream(String str){ ArrayList<Character> list = new ArrayList<>(); for(Character c : str.toCharArray()){ list.add(c); } return list.stream(); } /** flatMap 和 Map :就类似于,数组里面套数组 集合里面套集合 遍历数组和集合的所有元素~*/ @Test public void flat(){ ArrayList list1 = new ArrayList(); list1.add(1); list1.add(2); list1.add(3); ArrayList list2 = new ArrayList(); list2.add(4); list2.add(5); list2.add(6); /** map 就相当于是 add(集合) */// list1.add(list2); /** flatMap 就相当于是 addAll(集合) 将集合拆分,对每个单独的元素进行操作~*/// list1.addAll(list2); System.out.println(list1); } /** 排序 */ @Test public void test3(){ System.out.println("sorted()——自然排序"); System.out.println("注意,如果是自定义类型需要,实现Comparable接口,int 默认实现了Comparable"); List<Integer> list = Arrays.asList(12, 43, 65, 34, 87, 0, -98, 7); list.stream().sorted().forEach(System.out::println); System.out.println("sorted(Comparator com)——定制排序"); list.stream().sorted((e1,e2)->-e1.compareTo(e2)).forEach(System.out::println); /** * 排序 A 比较 B * 返回 0 A相等B * 返回 -1 A小于B * 返回 1 A大于B * */ }}
终止操作(终端操作)
每个Stream流都是有三个步骤:创建
中间操作
终止操作
- 一共Stream一旦调用了
终止操作
就表示改Strema 对象,关闭了, 就不可以在进行操作~ - 同样,一共没有调用
终止操作
的Stream 是不会结束的, 一直占用系统资源~ - 终端操作会从流的流水线生成结果,其结果可以是任何不是流的值,例 如:List、Integer
流进行了终止操作后,不能再次使用
方法 | 描述 |
---|---|
匹配与查找 | |
allMatch(Predicate p) | 检查是否匹配所有元素 |
anyMatch(Predicate p) | 检查是否至少匹配一个元素 |
noneMatch(Predicate p) | 检查是否没有匹配所有元素 |
findFirst() | 返回第一个元素 |
findAny() | 返回当前流中的任意元素 |
count() | 返回流中元素总数 |
max(Comparator c) | 返回流中最大值 |
min(Comparator c) | 返回流中最小值 |
forEach(Consumer c) | 内部迭代(使用 Collection 接口需要用户去做迭代,称为外部迭代) Stream API 使用内部迭 代——它帮你把迭代做了) |
归 约 | |
reduce(T iden, BinaryOperator b) | 可以将流中元素反复结合起来,得到一 个值,返回 T |
reduce(BinaryOperator b) | 可以将流中元素反复结合起来,得到一 个值,返回 Optional< T > |
收 集collect |
|
collect(Collector c) | 将流转换为其他形式。接收一个 Collector 接口的实现,用于给Stream中元素做汇总 的方法 |
collect(Collector c)
Collector 接口中方法的实现决定了如何对流执行收集的操作:
如收集到 List、Set、 Map
Collectors 实用类提供了很多静态方法
可以方便地创建常见收集器实例, 具体方法与实例如下表:
方法 | 返回类型 | 作用 | 示例: |
---|---|---|---|
toList | List< T > | 把流中元素收集到List | List emps= list.stream().collect(Collectors.toList()); |
toSet | Set< T > | 把流中元素收集到Set | Set emps= list.stream().collect(Collectors.toSet()); |
toCollection | Collection< T > | 把流中元素收集到创建的集合 | Collection emps =list.stream().collect(Collectors.toCollection(ArrayList::new)); |
counting | **Long ** | 计算流中元素的个数 | long count = list.stream().collect(Collectors.counting()); |
summingInt | Integer | 对流中元素的整数属性求和 | int total=list.stream().collect(Collectors.summingInt(Employee::getSalary)); |
averagingInt | averagingInt | 计算流中元素Integer属性的平均值 | double avg = list.stream().collect(Collectors.averagingInt(Employee::getSalary)); |
summarizingInt | IntSummaryStatistics | 收集流中Integer属性的统计值 如:平 均值 |
int SummaryStatisticsiss= list.stream().collect(Collectors.summarizingInt(Employee::getSalary)); |
方法 返回类型 作用 示例:
joining String 连接流中每个字符串
String str= list.stream().map(Employee::getName).collect(Collectors.joining());
maxBy Optional<T> 根据比较器选择最大值
Optional<Emp>max= list.stream().collect(Collectors.maxBy(comparingInt(Employee::getSalary)));
minBy Optional<T> 根据比较器选择最小值
Optional<Emp> min = list.stream().collect(Collectors.minBy(comparingInt(Employee::getSalary)));
reducing 归约产生的类型 从一个作为累加器的初始值开始,利用BinaryOperator与流中元素逐个结合,从而归约成单个值
int total=list.stream().collect(Collectors.reducing(0, Employee::getSalar, Integer::sum));
collectingAndThen 转换函数返回的类型 包裹另一个收集器,对其结果转换函数
int how= list.stream().collect(Collectors.collectingAndThen(Collectors.toList(), List::size));
groupingBy Map<K, List<T>> 根据某属性值对流分组,属性为K,结果为V
Map<Emp.Status, List<Emp>> map= list.stream().collect(Collectors.groupingBy(Employee::getStatus));
partitioningBy Map<Boolean, List<T>> 根据true或false进行分区
Map<Boolean,List<Emp>> vd = list.stream().collect(Collectors.partitioningBy(Employee::getManage));
示例:Demo
import java.util.Arrays;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.Stream;
/** 终止操作 */
public class StreamTest3 {
@Test
public void test(){
System.out.println("匹配与查找\n");
List<Emp> employees = Emp.getEmployees();
// allMatch(Predicate p)——检查是否匹配所有元素。
// 练习:是否所有的员工的年龄都大于18
boolean allMatch = employees.stream().allMatch(e -> e.getAge() > 18);
System.out.println(allMatch);
// anyMatch(Predicate p)——检查是否至少匹配一个元素。
// 练习:是否存在员工的工资大于 10000
boolean anyMatch = employees.stream().anyMatch(e -> e.getSalary() > 10000);
System.out.println(anyMatch);
// noneMatch(Predicate p)——检查是否没有匹配的元素。
// 练习:是否存在员工姓“雷”
boolean noneMatch = employees.stream().noneMatch(e -> e.getName().startsWith("雷"));
System.out.println(noneMatch);
// findFirst——返回第一个元素
Optional<Emp> employee = employees.stream().findFirst();
System.out.println(employee);
// findAny——返回当前流中的任意元素
Optional<Emp> employee1 = employees.parallelStream().findAny();
System.out.println(employee1);
}
@Test
public void test2(){
List<Emp> employees = Emp.getEmployees();
// count——返回流中元素的总个数
long count = employees.stream().filter(e -> e.getSalary() > 5000).count();
System.out.println(count);
// max(Comparator c)——返回流中最大值
// 练习:返回最高的工资:
Stream<Double> salaryStream = employees.stream().map(e -> e.getSalary());
Optional<Double> maxSalary = salaryStream.max(Double::compare);
System.out.println(maxSalary);
// min(Comparator c)——返回流中最小值
// 练习:返回最低工资的员工
Optional<Emp> employee = employees.stream().min((e1, e2) -> Double.compare(e1.getSalary(), e2.getSalary()));
System.out.println(employee);
System.out.println();
// forEach(Consumer c)——内部迭代
employees.stream().forEach(System.out::println);
//使用集合的遍历操作
employees.forEach(System.out::println);
}
//2-归约
@Test
public void test3(){
// reduce(T identity, BinaryOperator)——可以将流中元素反复结合起来,得到一个值。返回 T
// 练习1:计算1-10的自然数的和
List<Integer> list = Arrays.asList(1,2,3,4,5,6,7,8,9,10);
Integer sum = list.stream().reduce(0, Integer::sum);
System.out.println(sum);
// reduce(BinaryOperator) —— 函数式接口对象——>可以将流中元素反复结合起来,得到一个值,返回 Optional<T>
// 练习2:计算公司所有员工工资的总和
List<Emp> employees = Emp.getEmployees();
Stream<Double> salaryStream = employees.stream().map(Emp::getSalary);
Optional<Double> sumMoney = salaryStream.reduce((d1,d2) -> d1 + d2);
System.out.println(sumMoney.get());
}
//3-收集
@Test
public void test4(){
// collect(Collector c)——将流转换为其他形式。接收一个 Collector接口的实现,用于给Stream中元素做汇总的方法
// 练习1:查找工资大于6000的员工,结果返回为一个List或Set
List<Emp> employees = Emp.getEmployees();
List<Emp> employeeList = employees.stream().filter(e -> e.getSalary() > 6000).collect(Collectors.toList());
employeeList.forEach(System.out::println);
System.out.println();
Set<Emp> employeeSet = employees.stream().filter(e -> e.getSalary() > 6000).collect(Collectors.toSet());
employeeSet.forEach(System.out::println);
}
}
Optional 类
Java应用中最常见的bug就是空值异常
在Java 8之前,Google Guava引入了Optionals类来解决NullPointerException
从而避免源码被各种**null**检查污染,以便开发者写出更加整洁的代码
Java 8也将Optional加入了官方库
Optional 类(java.util.Optional) 是一个容器类
就是对一共对象,的一共包装~
保证调用对象之后不会产生 空指针
它可以保存类型T的值,代表这个值存在,或者仅仅保存null,表示这个值不存在
常用方法:
创建Optional类对象的方法
- Optional.of(T t) : 创建一个 Optional 实例,t必须非空
- Optional.empty() : 创建一个空的 Optional 实例
- Optional.ofNullable(T t):t可以为null
判断Optional容器中是否包含对象
- boolean isPresent() : 判断是否包含对象
- void ifPresent(Consumer consumer) :如果有值,就执行Consumer 接口的实现代码,并且该值会作为参数传给它
获取Optional容器的对象
- T get(): 如果调用对象包含值,返回该值,否则抛异常
- T orElse(T other) :如果有值则将其返回,否则返回指定的other对象
- T orElseGet(Supplier other) :如果有值则将其返回,否则返回由 Supplier接口实现提供的对象
- T orElseThrow(Supplier exceptionSupplier) :如果有值则将其返 回,否则抛出由Supplier接口实现提供的异常
接口默认方法
这个最为简单,可以简单的理解现在的接口中方法可以定默认实现
这样做到了像以前一样的抽象方法实现接口的默认实现,也方便了我们不在需要像以前一样做抽象的模板模式
interface A{ defalut void method1(){ method2(); //默认实现 } void method2();}
java8接口中除了default method,还提供定义(并实现)静态方法
interface B{ static String method(){ return "xxx"; }}
注解的影响:
新日期API
其它新增:
Base64 加解密:
Java 8将Base64 加入到JDK库中 样不需要使用第三方库就可以进行Base64编码
import java.nio.charset.StandardCharsets;import java.util.Base64;/** Base64 加解密 */public class Base { public static void main(String[] args) { final String text = "Java慈祥,yyds!"; final String encoded = Base64.getEncoder().encodeToString( text.getBytes( StandardCharsets.UTF_8 ) ); System.out.println("加密:"+encoded); final String decoded = new String(Base64.getDecoder().decode( encoded ),StandardCharsets.UTF_8 ); System.out.println("解密:"+decoded); }}
MD5 加解密:
尽然提到加密就顺便题一下MD5 加密:
- MD5的全称是Message-Digest Algorithm
信息-摘要算法
- MD5其实不算是加密算法,而是一种信息的摘要,它的特性是
不可逆的
除了暴力破解 一般逆序算法是得不到结果的一个个实验暴力循环~
举个例子:
1+99=100
**MD5接到的字符是1和99 然后通过自己的算法最后生成100 **但知道结果是100却很难推测出是通过1+99得来的
再比如 一本书的每一页取一个字,最后通过计算得出一个MD5码
但却很难通过这个MD5码去推测出这本书的内容...
MD5加密的特点主要有以下几点:
- 针对不同长度待加密的数据、字符串等等,其都可以返回一个固定长度的MD5加密字符串
- 其加密过程几乎不可逆,除非维护一个庞大的Key-Value数据库来进行碰撞破解,否则几乎无法解开
- 对于一个固定的字符串。数字等等,MD5加密后的字符串是固定的,
也就是说不管MD5加密多少次,都是同样的结果
java.security.MessageDigest
import java.math.BigInteger;import java.security.MessageDigest;import java.security.NoSuchAlgorithmException;/** Md5加密 */public class Md5 { public static void main(String[] args) throws NoSuchAlgorithmException { // 生成一个MD5加密计算摘要 MessageDigest md = MessageDigest.getInstance("MD5"); md.update("Java.慈祥,yyds".getBytes()); //digest()最后确定返回md5 hash值,返回值为8位字符串。因为md5 hash值是16位的hex值,实际上就是8位的字符 //BigInteger函数则将8位的字符串转换成16位hex值,用字符串来表示;得到字符串形式的hash值 //一个byte是八位二进制,也就是2位十六进制字符(2的8次方等于16的2次方) System.out.println(new BigInteger(1, md.digest()).toString(16)); }}
MD5 简单的字符串加密之后可以在线解密,复杂的话解不出来的
在线解密
UUID
UUID 是Java1.5 就新增的~
- UUID是指在一台机器上生成的数字,它保证对在同一时空中的所有机器都是唯一的
UUID由以下几部分的组合:
- 当前日期和时间,UUID的第一个部分与时间有关,如果你在生成一个UUID之后,过几秒又生成一个UUID,则第一个部分不同,其余相同
- 时钟序列
- 全局唯一的IEEE机器识别号,如果有网卡,从网卡MAC地址获得,没有网卡以其他方式获得
import java.util.UUID;String uuid = UUID.randomUUID().toString();
JDK8 一文搞定👍的更多相关文章
- 一文搞定 SonarQube 接入 C#(.NET) 代码质量分析
1. 前言 C#语言接入Sonar代码静态扫描相较于Java.Python来说,相对麻烦一些.Sonar检测C#代码时需要预先编译,而且C#代码必须用MSbuid进行编译,如果需要使用SonarQub ...
- 一文搞定MySQL的事务和隔离级别
一.事务简介 事务是数据库管理系统执行过程中的一个逻辑单位,由一个有限的数据库操作序列构成. 一个数据库事务通常包含了一个序列的对数据库的读/写操作.它的存在包含有以下两个目的: 为数据库操作序列提供 ...
- 一文搞定scrapy爬取众多知名技术博客文章保存到本地数据库,包含:cnblog、csdn、51cto、itpub、jobbole、oschina等
本文旨在通过爬取一系列博客网站技术文章的实践,介绍一下scrapy这个python语言中强大的整站爬虫框架的使用.各位童鞋可不要用来干坏事哦,这些技术博客平台也是为了让我们大家更方便的交流.学习.提高 ...
- 一文搞定Spring Boot + Vue 项目在Linux Mysql环境的部署(强烈建议收藏)
本文介绍Spring Boot.Vue .Vue Element编写的项目,在Linux下的部署,系统采用Mysql数据库.按照本文进行项目部署,不迷路. 1. 前言 典型的软件开发,经过" ...
- 21.SpringCloud实战项目-后台题目类型功能(网关、跨域、路由问题一文搞定)
SpringCloud实战项目全套学习教程连载中 PassJava 学习教程 简介 PassJava-Learning项目是PassJava(佳必过)项目的学习教程.对架构.业务.技术要点进行讲解. ...
- 一文搞定FastDFS分布式文件系统配置与部署
Ubuntu下FastDFS分布式文件系统配置与部署 白宁超 2017年4月15日09:11:52 摘要: FastDFS是一个开源的轻量级分布式文件系统,功能包括:文件存储.文件同步.文件访问(文件 ...
- 一文搞定 Git 相关概念和常用指令
我几乎每天都使用 Git,但仍然无法记住很多命令. 通常,只需要记住下图中的 6 个命令就足以供日常使用.但是,为了确保使用地很顺滑,其实你应该记住 60 到 100 个命令. Git 相关术语 Gi ...
- 一文搞定十大经典排序算法(Java实现)
本文总结十大经典排序算法及变形,并提供Java实现. 参考文章: 十大经典排序算法总结(Java语言实现) 快速排序算法—左右指针法,挖坑法,前后指针法,递归和非递归 快速排序及优化(三路划分等) 一 ...
- 一文搞定Flask
Flask 一 .Flask简介 Flask是一个基于Python开发并且依赖jinja2模板和Werkzeug WSGI服务的一个微型框架,对于Werkzeug本质是Socket服务端,其用于接收h ...
随机推荐
- Docker 快速删除无用(none)镜像
Dockerfile 代码更新频繁,自然docker build构建同名镜像也频繁的很,产生了众多名为none的无用镜像. 分别执行以下三行可清除 docker ps -a | grep " ...
- CF1428A Box is Pull 题解
Content 有一个兔子拖着一个盒子在走,每秒钟可以带着盒子走一个单位,也可以不带着盒子走一个单位.当且仅当兔子和盒子的距离不超过 \(1\) 时可以带着盒子走一个单位.现给出 \(t\) 次询问, ...
- 贪心——122.买卖股票的最佳时机II
给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格. 设计一个算法来计算你所能获取的最大利润.你可以尽可能地完成更多的交易(多次买卖一支股票). 注意:你不能同时参与多笔交易(你必须在再次 ...
- python enumerate枚举用法总结
enumerate()说明 enumerate()是python的内置函数enumerate在字典上是枚举.列举的意思对于一个可迭代的(iterable)/可遍历的对象(如列表.字符串),enumer ...
- Python第三周 函数详解
def 函数名(): """注释说明"""" 执行逻辑体 return 返回值 定义不带参数的函数 带参数的函数 默认参数 这个是 ...
- yarn 过程中遇到的问题
场景 项目中打包遇到了点问题,所以想删除原先装好的依赖包,重新yarn,结果神奇的报错了,无语... 遇到的问题 (1)error An unexpected error occurred: &quo ...
- JAVA实现导出excel功能,封装通用导出工具类
引入maven <!--excel导出--> <dependency> <groupId>net.sourceforge.jexcelapi</groupId ...
- BitBake使用攻略--BitBake的语法知识一
目录 写在前面 1. BitBake中的赋值 1.1 直接赋值 1.2 间接赋值 1.3 追加与前加赋值 1.4 Override风格的赋值语法 1.5 标志赋值 1.6 内联函数赋值 1.7 其他一 ...
- 【LeetCode】12. Integer to Roman 整数转罗马数字
作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 个人公众号:负雪明烛 本文关键词:roman, 罗马数字,题解,leetcode, 力扣, ...
- 【LeetCode】694. Number of Distinct Islands 解题报告 (C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS 日期 题目地址:https://leetcod ...