1.Stream流
在Java 8中,得益于Lambda所带来的函数式编程,引入了一个全新的Stream概念,用于解决已有集合类库既有的弊端
1.1 引言
传统集合的多步遍历代码
几乎所有的集合都支持直接或者间接的遍历操作。
public class Day082 {
public static void main(String[] args) {
List<String> list=new ArrayList<>();
list.add("路飞");
list.add("索隆");
list.add("山治");
list.add("乔巴");
for (String s : list) {
System.out.println(s);
}
}
}
循环遍历的弊端
Java 8的Lambda让我们可以更加专注于做什么(What),而不是怎么做(How),这点此前已经结合内部类进行了对比说明。现在,我们仔细体会一下上例代码,可以发现:
(1)for循环的语法就是“怎么做”
(2)for循环的循环体才是“做什么”
为什么使用循环?因为要进行遍历。但循环是遍历的唯一方式吗?遍历是指每一个元素逐一进行处理,而并不是从第一个到最后一个顺次处理的循环。前者是目的,后者是方式。
试想一下,如果希望对集合中的元素进行筛选过滤:
(1)试想一下,如果希望对集合中的元素进行筛选过滤:
(2)然后再根据条件二过滤为子集C。
在java8之前可能是:
public class Day082 {
public static void main(String[] args) {
List<String> list=new ArrayList<>();
list.add("路飞22");
list.add("索隆3333");
list.add("山治3333");
list.add("乔巴333");
List<String> list1=new ArrayList<>();
for (String s : list) {
if(s.length()>4){
list1.add(s);
}
}
for (String s : list1) {
System.out.println(s);
}
}
}
Stream的更优写法
public class DemoStream {
public static void main(String[] args) {
List<String> list=new ArrayList<>();
list.add("路飞22");
list.add("索隆3333");
list.add("山治3333");
list.add("乔巴333");
list.stream()
.filter(s ->s.length()>4)
.filter(s -> s.length()>5)
.forEach(System.out::println);
}
}
直接阅读代码的字面意思即可完美展示无关逻辑方式的语义:获取流、过滤长度>4、过滤长度>5、逐一打印。代码中并没有体现使用线性循环或是其他任何算法进行遍历,我们真正要做的事情内容被更好地体现在代码中。
1.2 流式思想概述:
stream流其实就是一个集合元素的函数模型,它并不是集合,也不是数据结构,其本身并不储存任何元素或者地址。
Stream流是一个来自数据源的元素队列:
元素是特定类型的对象,形成一个队列。java中的stream并不会存储元素,只是计算元素。
数据源流的来源。可以是集合、数组等。
与collection操作不同,stream流操作还有两个基础的特性。
Pipelining:中间的操作都会返回流对象本身。这样就可以多个操作串联成一个管道,就像流式风格,这样做可以对操作进行优化,比如说演示和短路。
内部迭代:以前对结合的遍历都是通过Itertor或者增强for的方式,这些都是显式的在外部迭代,叫做外部迭代。Stream提供了内部迭代的方式。流可以直接调用遍历方法。
在调用一个流对象时,包括三个步骤:获取一个数据源---->数据转换---->执行操作获取想要的结果。每次进行转换的原有的Stream对象不变,会返回一个新的Stream对象(可以进行多次转换),这就使操作可以像链一样,形成一个管道。
1.3 获取流
java.util.stream <T>是java8新加入的最常用的流接口。
获取一个流非常简单,有以下几种常用的方式:
(1)所有的Collection集合可以通过Stream的默认方法,获取流。
(2)Stream接口的静态方法of可以获取数组对应的流。
根据Collection获取流:
首先,java.util.collection接口中加入了default方法stream来获取流。
public class Demostream {
public static void main(String[] args) {
List<String> list=new ArrayList<>();
Stream<String> stream = list.stream();
Set<String> set=new HashSet<>();
Stream<String> stream1 = set.stream();
Vector<String> vector=new Vector<>();
Stream<String> stream2 = vector.stream();
}
}
根据Map获取流:
java.util.Map接口不是Collection的子接口,map结合中的数据结构为K-V不符合流数据的单一特征,因此map集合获取对应的流对象,要分key、value、或者entry等情况。
public class Demostream {
public static void main(String[] args) {
Map<String,String> map=new HashMap<>();
Stream<String> stream = map.keySet().stream();
Stream<String> stream1 = map.values().stream();
Stream<Map.Entry<String, String>> stream2 = map.entrySet().stream();
}
}
根据数组获取流:
如果使用的不是集合或者映射而是数组,因为数组对象,不能有添加默认方法,因此Stream接口中提了静态方法of:
public class Demostream {
public static void main(String[] args) {
String[] arr={"路飞","索隆","山治","乔巴"};
Stream<String> arr1 = Stream.of(arr);
}
}
of方法的参数其实是一个可变参数,所以支持数组。
1.4 Stream流的常用方法
Stream流的操作很丰富,这些操作方法分为两种:
延迟方法:返回值类型仍然是Stream接口自身类型的方法,支持链式编程。(除了终结方法外,都是延迟方法)
终结方法:返回值类型不再是Stream接口自身类型的方法,不支持链式编程,总结方法有count和forEach方法。
(1) 逐一处理:forEach
这个方法与for循环中的for-each不同
void fotEach(Consumer<? super T> action);
该方法接收一个Consumer接口函数,会将每一个流元素交给该函数处理。
Consumer接口
java.util.function.Consumer<T>接口是一个消费型接口。
Consumer接口中包含抽象方法void accept(T t),意为消费一个指定泛型的数据。
基本使用:
public class Demostream {
public static void main(String[] args) {
String[] arr={"路飞","索隆","山治","乔巴"};
Stream<String> arr1 = Stream.of(arr);
arr1.forEach(s-> System.out.println(s));
}
}
路飞
索隆
山治
乔巴
(2) 过滤:filter
可以通过filter方法将一个流转换成另一个子集流。
Stream<T> filter(Predicate<? super T> predicate);
该接口接收一个Predicate函数式接口(可以是lambda表达式或者方法引用)参数作为筛选条件。
Predicate接口:
Predicate接口中有一个抽象方法:
boolean test(T t);
test方法会产生一个boolean值结果,代表指定的条件是否满足。如果结果为true,那么filter方法将会留用元素,如果结果为false方法会舍弃元素。
基本使用:
public class Demostream {
public static void main(String[] args) {
Stream<String> stream = Stream.of("路飞", "路西", "索隆");
Stream<String> stream1 = stream.filter(s -> s.startsWith("路"));
stream1.forEach(System.out::println);
}
}
路飞
路西
(3) 映射:map
如果将流中的元素映射到另一个流中,可以使用map的方法。
<R> Stream<R> map(Function<? super T, ? extends R> mapper);
该接口需要一个Function函数式接口,可以将当前流中的T类型数据转换成另一种R类型的流。
Function函数式接口
java.util.stream.Function 函数式接口,其中唯一的抽象方法为:
R apply (T t);
这可以将一种T类型转换成R类型,这种转换的动作,就称为“映射”。
基本使用:
public class Demostream {
public static void main(String[] args) {
Stream<String> stream = Stream.of("10", "20", "10");
Stream<Integer> stream1 = stream.map(s ->Integer.parseInt(s));
stream1.forEach(System.out::println);
}
}
10
20
10
(4)统计个数
正如旧集合Collection 当中的size 方法一样,流提供count 方法来数一数其中的元素个数:
long count();
该方法返回一个long值代表元素个数(不再像旧集合那样是int值)。基本使用:
public class Demostream {
public static void main(String[] args) {
Stream<String> stream = Stream.of("张三", "张六", "李二");
Stream<String> stream1= stream.filter(s -> s.startsWith("张"));
System.out.println(stream1.count());
}
}
2
(5)取用前几个:limit
limit方法可以对流进行截取,只取用前几个。
Stream<T> limit (long maxSize);
参数是一个long型,如果集合当前长度大于参数则进行截取;否则不进行操作。基本使用
public class Demostream {
public static void main(String[] args) {
Stream<String> stream = Stream.of("张三", "张六", "李二");
Stream<String> stream1 = stream.limit(2);
stream1.forEach(System.out::println);
}
}
张三
张六
(6)跳过前几个:skip
如果希望跳过前几个元素,可以使用skip 方法获取一个截取之后的新流
Stream <T> skip(long n);
如果流的当前长度大于n,则跳过前n个;否则将会得到一个长度为0的空流。基本使用:
public class Demostream {
public static void main(String[] args) {
Stream<String> stream = Stream.of("张三", "张六", "李二");
Stream<String> stream1 = stream.skip(1);
stream1.forEach(System.out::println);
}
}
张六
李二
(7)组合:concat
如果有两个流,希望合并成为一个流,那么可以使用Stream 接口的静态方法concat :
static <T> Stream<T> concat(Stream<? extends T> a, Stream<? extends T> b)
public class Demostream {
public static void main(String[] args) {
Stream<String> stream = Stream.of("张三", "张六", "李二");
Stream<String> stream1 = Stream.of("张二", "张四", "李以");
Stream<String> stream3 = Stream.concat(stream, stream1);
stream3.forEach(System.out::println);
}
}
张三
张六
李二
张二
张四
李以
1.5 集合元素处理练习
需求:现在有两个ArrayList 集合存储队伍当中的多个成员姓名,要求使用传统的for循环(或增强for循环)依次进行以:
1. 第一个队伍只要名字为3个字的成员姓名;存储到一个新集合中。
2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。
.3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。
4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。
5. 将两个队伍合并为一个队伍;存储到一个新集合中。
6. 根据姓名创建Person 对象;存储到一个新集合中。
7. 打印整个队伍的Person对象信息。
使用传统的模式:
1. 构建Person类
public class Person {
private String name;
public Person() {
}
public Person(String name) {
this.name = name;
}
public String getName() {
return name;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
'}';
}
public void setName(String name) {
this.name = name;
}
}
2,实现代码
public class DemoArraylist {
public static void main(String[] args) {
//第一支队伍
ArrayList<String> list=new ArrayList<>();
list.add("迪丽热巴");
list.add("宋远桥");
list.add("苏星河");
list.add("石破天");
list.add("石中玉");
list.add("老子");
list.add("庄子");
list.add("洪七公");
//第二只队伍
ArrayList<String> list1=new ArrayList<>();
list1.add("古力娜扎");
list1.add("张无忌");
list1.add("赵丽颖");
list1.add("张三丰");
list1.add("尼古拉斯赵四");
list1.add("张天爱");
list1.add("张二狗");
// 第一个队伍只要名字为3个字的成员姓名;
ArrayList<String> listA=new ArrayList<>();
for (String s : list) {
if(s.length()==3){
listA.add(s);
}
}
// 第一个队伍筛选之后只要前3个人;
ArrayList<String> listB=new ArrayList<>();
for (int i = 0; i < 3; i++) {
listB.add(listA.get(i));
}
// 第二个队伍只要姓张的成员姓名;
ArrayList<String> list1A=new ArrayList<>();
for (String s : list1) {
if(s.startsWith("张")){
list1A.add(s);
}
}
// 第二个队伍筛选之后不要前2个人;
ArrayList<String> list1B=new ArrayList<>();
for (int i = 2; i < list1B.size(); i++) {
list1B.add(list1A.get(i));
}
// 将两个队伍合并为一个队伍;
ArrayList<String> newlist=new ArrayList<>();
newlist.addAll(listB);
newlist.addAll(list1B);
// 根据姓名创建Person对象;
ArrayList<Person> newlist1=new ArrayList<>();
for (String s : newlist) {
newlist1.add(new Person(s));
}
// 打印整个队伍的Person对象信息。
for (Person person : newlist1) {
System.out.println(person);
}
}
}
Person{name='宋远桥'}
Person{name='苏星河'}
Person{name='石破天'}
Person{name='张天爱'}
Person{name='张二狗'}
使用Stream方式处理:
public class DemoStream1 {
public static void main(String[] args) {
//第一支队伍
ArrayList<String> list = new ArrayList<>();
list.add("迪丽热巴");
list.add("宋远桥");
list.add("苏星河");
list.add("石破天");
list.add("石中玉");
list.add("老子");
list.add("庄子");
list.add("洪七公");
//第二只队伍
ArrayList<String> list1 = new ArrayList<>();
list1.add("古力娜扎");
list1.add("张无忌");
list1.add("赵丽颖");
list1.add("张三丰");
list1.add("尼古拉斯赵四");
list1.add("张天爱");
list1.add("张二狗");
//第一个队伍只要名字为3个字的成员姓名;第一个队伍筛选之后只要前3个人;
Stream<String> listA = list.stream().filter(s -> s.length() == 3).limit(3);
// 第二个队伍只要姓张的成员姓名;第二个队伍筛选之后不要前2个人;
Stream<String> list1A = list1.stream().filter(s -> s.startsWith("张")).skip(2);
//将两个队伍合并为一个队伍;根据姓名创建Person对象;打印整个队伍的Person对象信息。
Stream.concat(listA, list1A).map(Person::new).forEach(System.out::println);
}
}
Person{name='宋远桥'}
Person{name='苏星河'}
Person{name='石破天'}
Person{name='张天爱'}
Person{name='张二狗'}
上述的map中的lambda表达式使用的简便操作,也可这样写 map(s -> new Person(s))。
2.方法引用
在使用lambda表达式中,我们实际上传递进去的代码就是一种解决方案:拿什么参数做什么操作。那么考虑一种情况:如果我们在lambda中所指定的操作方案,已经有地方存在相同的方案,我们是够还有必要进行重写?
2.1 冗余的Lambda场景
我们来看一个简单的函数式接口以应用lambda表达式:
@FunctionalInterface
public interface println {
void print(String s);
}
在println接口中唯一的抽象方法print方法接收一个参数,目的就是为了打印这个参数,用lambda表达式很简单。
public class DemoPrintln {
public static void println(println p){
p.print("函数式接口");
}
public static void main(String[] args) {
println(s -> System.out.println(s));
}
}
函数式接口
其实println方法只管调用pritlnj接口中的print方法,并不会关心print方法的具体显示逻辑会将字符串打印到哪里。在main方法中通过lambda表达式指定了函数接口print的具体实现步骤:拿到String数据后,在控制台输出。
2.2 问题分析
这段代码的问题在于,对字符串进行控制台打印输出的操作方案,明明有了现有的实现,那就是System.out对象中的println(String)方法。既然lambda表达式希望调用的事println(String)方法,那何必自己在重写呢?
2.3 用方法引用改进代码
public class DemoPrintln {
public static void println(println p){
p.print("函数式接口");
}
public static void main(String[] args) {
println(System.out::println);
}
}
函数式接口
双冒号 “::”写法,称为方法引用,这是一种新的语法。
2.4 方法引用符
双冒号“::”为引用运算符,它所在的表达式称为方法引用。如果lambda要表达的函数方案已经存在与某在方法的实现中,那么就可以用双冒号来引用作为方法的替代者。
语法分析:
例如上例中, System.out 对象中有一个重载的println(String) 方法恰好就是我们所需要的。那么对于printString 方法的函数式接口参数,对比下面两种写法,完全等效;
(1)Lambda表达式写法: s -> System.out.println(s);
(2)方法引用写法: System.out::println
第一种语义是指:拿到参数之后经Lambda之手,继而传递给System.out.println 方法去处理。
第二种等效写法的语义是指:直接让System.out 中的println 方法来取代Lambda。两种写法的执行效果完全一样,而第二种方法引用的写法复用了已有方案,更加简洁。
注::Lambda 中 传递的参数 一定是方法引用中 的那个方法可以接收的类型,否则会抛出异常。
推导与省略:
如果使用Lambda,那么根据“可推导就是可省略”的原则,无需指定参数类型,也无需指定的重载形式——它们都将被自动推导。而如果使用方法引用,也是同样可以根据上下文进行推导。
函数式接口是Lambda的基础,而方法引用是Lambda的孪生兄弟。
下面这段代码将会调用println 方法的不同重载形式,将函数式接口改为int类型的参数:
@FunctionalInterface
public interface println {
void print(int s);
}
由于上下文变了之后可以自动推导出唯一对应的匹配重载,所以方法引用没有任何变化:
public class DemoPrintln {
public static void println(println p){
p.print(10);
}
public static void main(String[] args) {
println(System.out::println);
}
}
10
这次方法引用将会自动匹配到println(int) 的重载形式。
2.5 通过对象名引用成员方法
如果在一个类中定义了一种输出的方法;
public class Methodobject {
public void printUpperCase(String str){
System.out.println(str.toUpperCase());
}
}
此时有一个函数式接口:
public interface Printable {
void print(String str);
}
若要使用Methodobject类中的方法,来替代printable 接口中的lambda是,就需要Methodobject类的实现类对象,通过对象名引用成员方法。
public class DemoMethodobj {
public static void println(Printable p){
p.print("hello");
}
public static void main(String[] args) {
Methodobject mobj=new Methodobject();
println(mobj::printUpperCase);
}
}
HELLO
2.6 通过类名称引用静态方法。
由于在java.lang.Math 类中已经存在了静态方法abs ,所以当我们需要通过Lambda来调用该方法时,有两种写法。
首先定义 函数式接口:
@FunctionalInterface
public interface Calcable {
int calc(int num);
}
第一种写法用lambda表达式:
public class DemoLambda {
private static void method(int num,Calcable c){
System.out.println(c.calc(num));
}
public static void main(String[] args) {
method(-10,num->Math.abs(num));
}
}
10
第二种使用方法引用:
public class DemoLambda {
private static void method(int num,Calcable c){
System.out.println(c.calc(num));
}
public static void main(String[] args) {
method(-10,Math::abs);
}
}
其中:Lambda表达式: num-> Math.abs(num) 与 方法引用: Math::abs 是等价的。
2.7 通过super引用成员变量
如果存在继承关系,当Lambda中需要出现super调用时,也可以使用方法引用进行替代。
首先是函数式接口:
@FunctionalInterface
public interface Greetable {
void greet();
}
定义父类Human:
public class Human {
public void sayhello(){
System.out.println("hello");
}
}
定义子类Man,其中使用了lambda表达式
public class Man extends Human {
@Override
public void sayhello(){
System.out.println("我是子类的hello");
}
//定义方法,参数传递接口
public void method(Greetable greetable){
greetable.greet();
}
public void show(){
//调用method方法,使用lambda表达式
method(()->{
//创建Human对象,条用sayhello方法
new Human().sayhello();
});
//简化lambda
method(()->new Human().sayhello());
//使用super关键字代替父类对象
method(()->super.sayhello());
//使用this关键字代替父类对象
method(()->this.sayhello());
}
public static void main(String[] args) {
Man m=new Man();
m.show();
}
}
hello
hello
hello
我是子类的hello
定义子类Man,其中使用了方法引用
public class Man1 extends Human {
@Override
public void sayhello(){
System.out.println("我是子类的hello");
}
//定义方法,参数传递接口
public void method(Greetable greetable){
greetable.greet();
}
public void show(){
//使用super关键字代替父类对象
method(super::sayhello);
//使用this关键字代替父类对象
method(()->this.sayhello());
}
public static void main(String[] args) {
Man1 m=new Man1();
m.show();
}
}
hello
我是子类的hello
2.8 通过this引用成员方法
public class Man1 extends Human {
@Override
public void sayhello(){
System.out.println("我是子类的hello");
}
//定义方法,参数传递接口
public void method(Greetable greetable){
greetable.greet();
}
public void show(){
//使用this关键字代替父类对象
method(()->this.sayhello());
//通过方法引用
method(this::sayhello);
}
public static void main(String[] args) {
Man1 m=new Man1();
m.show();
}
}
我是子类的hello
我是子类的hello
2.9 类的构造器引用
由于构造器的名称与类名完全一样,并不固定。所以构造器引用使用类名称::new 的格式表示。
首先是一个简单的Person 类:
public class Person {
private String name;
public Person() {
}
public Person(String name) {
this.name = name;
}
public String getName() {
return name;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
'}';
}
public void setName(String name) {
this.name = name;
}
}
用来创建Person对象的函数式接口:
public interface PersonBuilder {
Person buildPerson(String name);
}
通过lambda表达式,使用和这个函数式接口:
public class DemoPerson {
public static void printName(String name,PersonBuilder pb){
System.out.println(pb.buildPerson(name).getName());
}
public static void main(String[] args) {
printName("路飞",(name)->new Person(name));
}
}
路飞
但是通过构造器引用,有更好的写法:
public class DemoPerson {
public static void printName(String name,PersonBuilder pb){
System.out.println(pb.buildPerson(name).getName());
}
public static void main(String[] args) {
printName("路飞",Person::new);
}
}
路飞
Lambda表达式: name -> new Person(name) 和方法引用: Person::new 等效。
2.10 数组的构造器引用
数组也是Object 的子类对象,所以同样具有构造器,只是语法稍有不同。如果对应到Lambda的使用场景中时,
需要一个函数式接口:
public interface Arraybuilder {
int[] buildArray();
}
在应用该接口的时候,可以通过Lambda表达式:
public class Demo {
private static int[] inttArray(int length,Arraybuilder abr){
return abr.buildArray(length);
}
public static void main(String[] args) {
inttArray(10,length -> new int[length]);
}
}
但是更好的写法是使用数组的构造器引用:
public class Demo {
private static int[] inttArray(int length,Arraybuilder abr){
return abr.buildArray(length);
}
public static void main(String[] args) {
inttArray(10,int[]::new);
}
}
Lambda表达式: length -> new int[length] 与方法引用: int[]::new 等效。