JAVA从零开始学习知识整理——day13—【Stream流】

一、Stream流
1.1 Stream流引入
传统的数据筛选等操作,需要根据筛选条件进行多次操作,除了典型的必需的添加、删除、获取外,其中最为常见的就是通过遍历集合等。
例如传统的方法:
三个循环完成三个筛选条件:

import java.util.ArrayList;
import java.util.List;
public class Demo02NormalFilter {
		public static void main(String[] args) {
				List<String> list = new ArrayList<>();
				list.add("张无忌");
				list.add("周芷若");
				list.add("赵敏");
				list.add("张强");
				list.add("张三丰");
				List<String> zhangList = new ArrayList<>();
				for (String name : list) {
						if (name.startsWith("张")) {
							zhangList.add(name);
						}
				}
				List<String> shortList = new ArrayList<>();
				for (String name : zhangList) {
				if (name.length() == 3) {
							shortList.add(name);
						}
				}
				for (String name : shortList) {
				System.out.println(name);
			}
		}
}

Stream的更优写法:

import java.util.ArrayList;
import java.util.List;
public class Demo03StreamFilter {
		public static void main(String[] args) {
			List<String> list = new ArrayList<>();
			list.add("张无忌");
			list.add("周芷若");
			list.add("赵敏");
			list.add("张强");
			list.add("张三丰");
			list.stream()
								.filter(s ‐> s.startsWith("张"))
								.filter(s ‐> s.length() == 3)
								.forEach(System.out::println);
			}
}

1.2 流式思想概述
在上面的例子里,过滤遍历等操作其实都是对流对象的处理,集合对象没有被真的处理,只有终结方法执行的时候,集合才会被按流程处理,这得益于lambda的延时执行特征。

**Stream(流)**是一个来自数据源的元素队列元素是特定类型的对象,形成一个队列。 Java中的Stream并不会存储元素,而是按需计算。
数据源 流的来源。 可以是集合,数组 等。
和以前的Collection操作不同, Stream操作还有两个基础的特征:
Pipelining: 中间操作都会返回流对象本身。 这样多个操作可以串联成一个管道, 如同流式风格(fluent
style)。 这样做可以对操作进行优化, 比如延迟执行(laziness)和短路( short-circuiting)。
内部迭代: 以前对集合遍历都是通过Iterator或者增强for的方式, 显式的在集合外部进行迭代, 这叫做外部迭
代。 Stream提供了内部迭代的方式,流可以直接调用遍历方法。

1.3获取流
java.util.stream.Stream 是Java 8新加入的最常用的流接口。(这并不是一个函数式接口。)
获取一个流非常简单,有以下几种常用的方式:

所有的 Collection 集合都可以通过 stream 默认方法获取流;
Stream 接口的静态方法 of 可以获取数组对应的流。

1.3.1 根据Collection获取流
首先, java.util.Collection 接口中加入了default方法 stream 用来获取流,所以其所有实现类均可获取流。

import java.util.*;
import java.util.stream.Stream;
public class Demo04GetStream {
		public static void main(String[] args) {
				List<String> list = new ArrayList<>();
				// ...
				Stream<String> stream1 = list.stream();
				Set<String> set = new HashSet<>();
				// ...
				Stream<String> stream2 = set.stream();
				Vector<String> vector = new Vector<>();
				// ...
				Stream<String> stream3 = vector.stream();
		}
}

1.3.2 根据Map获取流
java.util.Map 接口不是 Collection 的子接口,且其K-V数据结构不符合流元素的单一特征,所以获取对应的流
需要分key、value或entry等情况:

import java.util.HashMap;
import java.util.Map;
import java.util.stream.Stream;
public class Demo05GetStream {
		public static void main(String[] args) {
				Map<String, String> map = new HashMap<>();
				// ...
				Stream<String> keyStream = map.keySet().stream();
				Stream<String> valueStream = map.values().stream();
				Stream<Map.Entry<String, String>> entryStream = map.entrySet().stream();
		}
}

1.4 根据数组获取流
如果使用的不是集合或映射而是数组,由于数组对象不可能添加默认方法,所以 Stream 接口中提供了静态方法of ,使用很简单:

import java.util.stream.Stream;
public class Demo06GetStream {
public static void main(String[] args) {
			String[] array = { "张无忌", "张翠山", "张三丰", "张一元" };
			Stream<String> stream = Stream.of(array);
	}
}

1.4 常用方法
流模型的操作很丰富,这里介绍一些常用的API。这些方法可以被分成两种:
延迟方法:返回值类型仍然是 Stream 接口自身类型的方法,因此支持链式调用。(除了终结方法外,其余方法均为延迟方法。)
终结方法:返回值类型不再是 Stream 接口自身类型的方法,因此不再支持类似 StringBuilder 那样的链式调
用。本小节中,终结方法包括 count 和 forEach 方法。
逐一处理:forEach

import java.util.stream.Stream;
public class Demo12StreamForEach {
		public static void main(String[] args) {
			Stream<String> stream = Stream.of("张无忌", "张三丰", "周芷若");
			stream.forEach(name‐> System.out.println(name));
	}
}

过滤:filter

import java.util.stream.Stream;
public class Demo07StreamFilter {
		public static void main(String[] args) {
				Stream<String> original = Stream.of("张无忌", "张三丰", "周芷若");
				Stream<String> result = original.filter(s ‐> s.startsWith("张"));
		}
}

映射:map

import java.util.stream.Stream;
public class Demo08StreamMap {
		public static void main(String[] args) {
			Stream<String> original = Stream.of("10", "12", "18");
			Stream<Integer> result = original.map(str‐>Integer.parseInt(str));
	}
}

统计个数:count

import java.util.stream.Stream;
public class Demo09StreamCount {
		public static void main(String[] args) {
			Stream<String> original = Stream.of("张无忌", "张三丰", "周芷若");
			Stream<String> result = original.filter(s ‐> s.startsWith("张"));
			System.out.println(result.count()); // 2
		}
}

取用前几个:limit

import java.util.stream.Stream;
public class Demo10StreamLimit {
			public static void main(String[] args) {
				Stream<String> original = Stream.of("张无忌", "张三丰", "周芷若");
				Stream<String> result = original.limit(2);
				System.out.println(result.count()); // 2
		}
}

跳过前几个:skip

public class Demo11StreamSkip {
	public static void main(String[] args) {
			Stream<String> original = Stream.of("张无忌", "张三丰", "周芷若");
			Stream<String> result = original.skip(2);
			System.out.println(result.count()); // 1
	}
}

组合:concat

import java.util.stream.Stream;
public class Demo12StreamConcat {
		public static void main(String[] args) {
				Stream<String> streamA = Stream.of("张无忌");
				Stream<String> streamB = Stream.of("张翠山");
				Stream<String> result = Stream.concat(streamA, streamB);
		}
}

实例:
现在有两个 ArrayList 集合存储队伍当中的多个成员姓名,要求使用传统的for循环(或增强for循环)依次进行以
下若干操作步骤:

  1. 第一个队伍只要名字为3个字的成员姓名;存储到一个新集合中。
  2. 第一个队伍筛选之后只要前3个人;存储到一个新集合中。
  3. 第二个队伍只要姓张的成员姓名;存储到一个新集合中。
  4. 第二个队伍筛选之后不要前2个人;存储到一个新集合中。
  5. 将两个队伍合并为一个队伍;存储到一个新集合中。
  6. 根据姓名创建 Person 对象;存储到一个新集合中。
  7. 打印整个队伍的Person对象信息。
    代码实现:(Person类自建)
import java.util.ArrayList;
import java.util.stream.Stream;

public class demo {

        public static void main(String[] args) {
//第一支队伍
            ArrayList<String> one = new ArrayList<>();
            one.add("迪丽热巴");
            one.add("宋远桥");
            one.add("苏星河");
            one.add("石破天");
            one.add("石中玉");
            one.add("老子");
            one.add("庄子");
            one.add("洪七公");
//第二支队伍
            ArrayList<String> two = new ArrayList<>();
            two.add("古力娜扎");
            two.add("张无忌");
            two.add("赵丽颖");
            two.add("张三丰");
            two.add("尼古拉斯赵四");
            two.add("张天爱");
            two.add("张二狗");
// ....
            Stream<String> stream = one.stream();
            Stream<String> limit = stream.filter((name) -> name.length() == 3)
                    .limit(3);

            Stream<String> stream1 = two.stream();
            Stream<String> skip = stream1.filter((name) -> name.startsWith("张"))
                    .skip(2);

            Stream<String> concat = Stream.concat(limit, skip);
            ArrayList<Person> p = new ArrayList<>();
            concat.forEach((s) -> p.add(new Person(s)));

            for (Person person : p) {
                System.out.println(person);
            }


        }
    }

二、方法引用
在使用Lambda表达式的时候,我们实际上传递进去的代码就是一种解决方案:拿什么参数做什么操作。那么考虑一种情况:如果我们在Lambda中所指定的操作方案,已经有地方存在相同方案,那是否还有必要再写重复逻辑?
2.1 冗余的Lambda场景
当我们引用一个方法的时候,方法内有一个函数式接口需要lambda表达式的时候,对象和方法已经存在的条件下,我们无需进行方法调用,直接使用方法引用即可。
改进:

 public class Demo02PrintRef {
        private static void printString(Printable data) {
            data.print("Hello, World!");
        }
        public static void main(String[] args) {
            printString((s) -> System.out.println(s));
            printString(System.out::println);
        }
    }

2.2 方法引用符—— ::
语义分析
例如上例中, System.out 对象中有一个重载的 println(String) 方法恰好就是我们所需要的。那么对于
printString 方法的函数式接口参数,对比下面两种写法,完全等效:
Lambda表达式写法: s -> System.out.println(s);
方法引用写法: System.out::println

2.3 通过对象名引用成员方法
一个类有一个成员方法:

public class MethodRefObject {
			public void printUpperCase(String str) {
				System.out.println(str.toUpperCase());
		}
}

函数式接口:

@FunctionalInterface
	public interface Printable {
		void print(String str);
}

那么当需要使用这个 printUpperCase 成员方法来替代 Printable 接口的Lambda的时候,已经具有了
MethodRefObject 类的对象实例,则可以通过对象名引用成员方法,代码为:

public class Demo04MethodRef {
    private static void printString(Printable lambda) {
        lambda.print("Hello");
    }
    public static void main(String[] args) {
        MethodRefObject obj = new MethodRefObject();
        printString(obj::printUpperCase);
    }
}

2.4 通过类名称引用静态方法
由于在 java.lang.Math 类中已经存在了静态方法 abs ,所以当我们需要通过Lambda来调用该方法时,有两种写法。
首先是函数式接口:

@FunctionalInterface
		public interface Calcable {
			int calc(int num);
}

lambda表达式:

public class Demo05Lambda {
    private static void method(int num, Calcable lambda) {
        System.out.println(lambda.calc(num));
    }
    public static void main(String[] args) {
        method(‐10, n ‐> Math.abs(n));
    }
}

方法引用:

public class Demo06MethodRef {
    private static void method(int num, Calcable lambda) {
        System.out.println(lambda.calc(num));
    }
    public static void main(String[] args) {
        method(‐10, Math::abs);
    }
}

2.5 通过super引用成员方法
如果存在继承关系,当Lambda中需要出现super调用时,也可以使用方法引用进行替代。首先是函数式接口:

@FunctionalInterface
	public interface Greetable {
			void greet();
}

父类:

public class Human {
		public void sayHello() {
				System.out.println("Hello!");
		}
}

子类,lambda表达式:

public class Man extends Human {
    @Override
    public void sayHello() {
        System.out.println("大家好,我是Man!");
    }
    //定义方法method,参数传递Greetable接口
    public void method(Greetable g){
        g.greet();
    }
    public void show(){
//调用method方法,使用Lambda表达式
        method(()‐>{
//创建Human对象,调用sayHello方法
                new Human().sayHello();
});
//简化Lambda
        method(()‐>new Human().sayHello());
//使用super关键字代替父类对象
        method(()‐>super.sayHello());
    }
}

子类,方法引用:

public class Man extends Human {
    @Override
    public void sayHello() {
        System.out.println("大家好,我是Man!");
    }
    //定义方法method,参数传递Greetable接口
    public void method(Greetable g){
        g.greet();
    }
    public void show(){
        method(super::sayHello);
    }
}

2.6 通过this引用成员方法
函数式接口:

@FunctionalInterface
		public interface Richable {
			void buy();
}

Husband类,lambda:

public class Husband {
    private void buyHouse() {
        System.out.println("买套房子");
    }
    private void marry(Richable lambda) {
        lambda.buy();
    }
    public void beHappy() {
        marry(() ‐> this.buyHouse());
    }
}

方法引用:

public class Husband {
    private void buyHouse() {
        System.out.println("买套房子");
    }
    private void marry(Richable lambda) {
        lambda.buy();
    }
    public void beHappy() {
        marry(this::buyHouse);
  	  }
    }

2.7 类的构造器引用:
Person类:

public class Person {
    private String name;
    public Person(String name) {
        this.name = name;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
}

创建类的接口:

public interface PersonBuilder {
		Person buildPerson(String name);
	}

方法引用:

public class Demo10ConstructorRef {
    public static void printName(String name, PersonBuilder builder) {
        System.out.println(builder.buildPerson(name).getName());
    }
    public static void main(String[] args) {
        printName("赵丽颖", Person::new);
    }
}

2.8 数组的构造器引用

public class Demo12ArrayInitRef {
    private static int[] initArray(int length, ArrayBuilder builder) {
        return builder.buildArray(length);
    }
    public static void main(String[] args) {
        int[] array = initArray(10, int[]::new);
    }
}

猜你喜欢

转载自blog.csdn.net/weixin_44167508/article/details/86080786