Stream流式思想
使用Stream流式思想遍历集合进行过滤Stream流式思想:
[Java] 纯文本查看 复制代码 public class Demo02Stream {
public static void main(String[] args) {
//创建一个List集合,存储姓名
List<String> list = new ArrayList<>();
list.add("张无忌");
list.add("周芷若");
list.add("赵敏");
list.add("张强");
list.add("张三丰");
// 使用Stream流式操作
list.stream()
.filter(name -> name.startsWith("张")) // 第一次过滤: 只要以张开头的元素
.filter(name -> name.length()==3) // 第二次过滤: 只要姓名长度为3的人
.forEach(name -> System.out.println(name)); // 遍历查看结果
}
}
Stream流式思想处理数据的方式:
让代码的执行像流水线一样, 先设计好处理方案, 然后一按开关开始执行
流相比于集合的2个优点:
1. Pipelining(管道特性): "可以链式调用"
Stream流对象的 延迟方法 调用后, 会返回新的Stream流对象, 可以链式调用
每个方法类似于一条一条的管道, 衔接了不同的处理方案
2. 内部迭代特性: "不用写for循环"
集合遍历通过 Iterator 或者 增强for, 显式的在集合外部进行迭代, 这叫做外部迭代
Stream提供了内部迭代的方法 forEach(Consumer c), 可以直接调用遍历方法
使用Stream流的3个步骤:
1. 获取数据源 (从"集合"或"数组"转换为"Stream"对象)
2. 数据处理 (调用延迟方法, 编写处理方案)
3. 获得结果 (调用终结方法, 启动开关)
2种获取Stream流的方式
获取Stream流对象的2种方式:
1. 利用Collection接口中的默认方法 default Stream<E> stream() 方法: 集合转Stream对象
2. 利用Stream接口中的静态方法 static<T> Stream<T> of(T... values): 数组转Stream对象
java.util.Collection<E>接口:
// 默认方法
default Stream<E> stream(): 将"集合"转换为Stream对象
java.util.stream.Stream<T>接口: 管道接口, 泛型为流中元素的类型
// 静态方法
static<T> Stream<T> of(T... values): 将"数组"转换为Stream对象
[Java] 纯文本查看 复制代码 // 集合转换为Stream流对象
List<String> list = new ArrayList<>();
Stream<String> listStream = list.stream(); // List集合的Stream
Set<String> set = new HashSet<>();
Stream<String> setStream = set.stream(); // Set集合的Stream
Map<String,String> map = new HashMap<>();
Stream<String> keyStream = map.keySet().stream(); // 键的集合的Stream
Stream<String> valueStream = map.values().stream(); // 值的集合的Stream
Stream<Map.Entry<String, String>> entryStream = map.entrySet().stream(); // 键值对的集合的Stream[/font][/size]
[size=2][font=微软雅黑]
[/font][/size]
[size=2][font=微软雅黑]// 数组转换为Stream流对象
Stream<Integer> stream = Stream.of(1, 2, 3, 4, 5);
String[] array = {"a", "b", "c"};
Stream<String> arrayStream = Stream.of(array);
Stream API: 方法分类, forEach()遍历
延迟方法: (具有延迟执行的特性)
返回值类型 是 Stream接口自身类型的方法, 支持链式调用
filter(): 过滤
map(): 映射/转换
limit(): 截取
skip(): 跳过
终结方法:
返回值类型 不是 Stream接口自身类型的方法, 不支持链式调用
forEach(): 遍历
count(): 统计
------------------------------------------------------
创建数据源 -> filter()过滤 | map()映射 | limit()截取 | skip()跳过 -> forEach() | count()结果
------------------------------------------------------
获取流 转换 聚合
延迟方法 终结方法
new ArrayList().filter(...).map(...).limit(...).skip(...).forEach(...);
Stream.of(1,2,3).filter(...).map(...).limit(...).skip(...).count();
Stream API: forEach()遍历集合
java.util.stream.Stream<T>接口: 管道接口 // 抽象方法
void forEach(Consumer<? super T> action): 遍历流中的元素进行逐一消费. 并不保证元素的逐一消费动作在流中是被有序执行的
将数组转换为Stream对象, 并使用forEach()方法遍历数组, 打印每个元素到控制台
[Java] 纯文本查看 复制代码 public class Test {
public static void main(String[] args) {
String[] arr = {"张三", "李四", "王五", "赵六", "田七"};
// 将数组转换为流对象
Stream<String> stream = Stream.of(arr);
// 遍历流
stream.forEach(new Consumer<String>() {
@Override
public void accept(String s) {
System.out.println(s);
}
});
// 使用链式调用+Lambda表达式的标准格式
Stream.of("张三", "李四", "王五", "赵六", "田七")
.forEach((String name)-> {
System.out.println(name);
});
// Lambda的省略格式
Stream.of("张三", "李四", "王五", "赵六", "田七")
.forEach(name-> System.out.println(name));
}
}
Stream API: filter()过滤
java.lang.String类:
boolean startsWith(String prefix): 判断当前字符串是否以参数字符串开头
java.util.stream.Stream<T>接口: 管道接口
// 抽象方法
Stream<T> filter(Predicate<? super T> predicate): 过滤符合条件的结果. 返回过滤后的流
boolean test(String name)
定义字符串数组: {"张三丰", "张翠山", "赵敏", "周芷若", "张无忌"}
将数组转换为Stream对象, 并过滤出姓张的, 并将元素打印出来
代码:
[Java] 纯文本查看 复制代码 public class Test {
public static void main(String[] args) {
// 普通方式
String[] arr = {"张三丰", "张翠山", "赵敏", "周芷若", "张无忌"};
Stream<String> stream1 = Stream.of(arr);
Stream<String> stream2 = stream1.filter(new Predicate<String>() {
@Override
public boolean test(String name) {
return name.startsWith("张");
}
});
stream2.forEach(new Consumer<String>() {
@Override
public void accept(String name) {
System.out.println(name);
}
});
// 链式调用+Lambda表达式
Stream.of("张三丰", "张翠山", "赵敏", "周芷若", "张无忌")
.filter(name->name.startsWith("张"))
.forEach(name-> System.out.println(name));
}
}
Stream API: map()映射转换
java.util.stream.Stream<T>接口: 管道接口
// 抽象方法
<R> Stream<R> map(Function<T, R> mapper): 将当前流中的T类型的元素, 转换R类型元素, 放入新流并返回
R apply(T t)
定义字符串数组: {"1", "2", "3", "4"}
将数组转换为Stream流, 使用map()方法将流中的字符串都转换为int, 遍历输出转换后的结果
代码:
[Java] 纯文本查看 复制代码 public class Test {
public static void main(String[] args) {
String[] arr = {"1", "2", "3", "4"};
Stream<String> stream1 = Stream.of(arr);
Stream<Integer> stream2 = stream1.map(new Function<String, Integer>() {
@Override
public Integer apply(String s) {
return Integer.parseInt(s);
}
});
stream2.forEach(i-> System.out.println(i));
// 链式调用+Lambda表达式
Stream.of("1", "2", "3", "4")
.map(s -> Integer.parseInt(s))
.forEach(i-> System.out.println(i));
}
}
Stream API: count()统计流中元素个数
java.util.stream.Stream<T>接口: 管道接口
// 抽象方法
long count(): 获取流中的元素个数 (终结方法)
创建一个ArrayList集合, 存储整数: 1,2,3,4,5,6,7
将集合转换为Stream流对象, 调用count()获取流中元素的个数, 打印到控制台
代码:
[Java] 纯文本查看 复制代码 public class Test {
public static void main(String[] args) {
ArrayList<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
list.add(4);
list.add(5);
list.add(6);
list.add(7);
long count = list.stream().count();
System.out.println(count);
// 利用JDK9提供的快速创建集合方法
long count2 = List.of(1, 2, 3, 4, 5, 6, 7)
.stream()
.count();
System.out.println(count2);
}
}
Stream API: limit()获取前n个(只要前n个)
java.util.stream.Stream<T>接口: 管道接口
// 抽象方法
Stream<T> limit(long maxSize): 从流中获取前maxSize个. 如果maxSize大于等于元素个数, 则返回所有元素的流
创建字符串数组: {"美羊羊", "喜洋洋", "懒洋洋", "灰太狼", "红太狼"}
将数组转换为Stream流, 从流中获取前3个元素, 将截取后的流遍历打印输出到控制台
代码:
[Java] 纯文本查看 复制代码 public class Test {
public static void main(String[] args) {
String[] arr = {"美羊羊", "喜洋洋", "懒洋洋", "灰太狼", "红太狼"};
// Stream.of(arr)
// .limit(3)
// .forEach(name-> System.out.println(name));
Stream.of("美羊羊", "喜洋洋", "懒洋洋", "灰太狼", "红太狼")
.limit(300) // 超出实际大小, 有多少返回多少
.forEach(name-> System.out.println(name));
}
}
Stream API: skip()跳过前n个(不要前n个)
java.util.stream.Stream<T>接口: 管道接口
// 抽象方法
Stream<T> skip(long n): 从流中跳过n个元素, 获取后面的元素. 如果n大于等于元素个数, 则全都跳过
需求:
创建字符串数组: {"美羊羊", "喜洋洋", "懒洋洋", "灰太狼", "红太狼"}
将数组转换为Stream流, 从流中跳过前3个元素, 遍历打印输出到控制台
[Java] 纯文本查看 复制代码 public class Test {
public static void main(String[] args) {
String[] arr = {"美羊羊", "喜洋洋", "懒洋洋", "灰太狼", "红太狼"};
Stream.of(arr)
.skip(3)
.forEach(name-> System.out.println(name));
System.out.println("-----------");
Stream.of("美羊羊", "喜洋洋", "懒洋洋", "灰太狼", "红太狼")
.skip(3)
.forEach(name-> System.out.println(name));
}
}
Stream API: 静态方法concat()合并两个流
java.util.stream.Stream<T>接口: 管道接口
// 静态方法
static <T> Stream<T> concat(Stream<? extends T> a, Stream<? extends T> b): 合并两个流的元素, 变成一个新的流. 两个流中的元素类型必须相同, 或有共同的父类
需求:
创建2个字符串数组:
{"张三丰", "张翠山", "赵敏", "周芷若", "张无忌"}
{"美羊羊", "喜洋洋", "懒洋洋", "灰太狼", "红太狼"}
将2个数组分别转换为Stream流对象, 并使用Stream的静态方法concat()将2个流拼接为新的流, 遍历打印新流的元素
[Java] 纯文本查看 复制代码 public class Test {
public static void main(String[] args) {
String[] arr1 = {"张三丰", "张翠山", "赵敏", "周芷若", "张无忌"};
String[] arr2 = {"美羊羊", "喜洋洋", "懒洋洋", "灰太狼", "红太狼"};
Stream<String> stream1 = Stream.of(arr1);
Stream<String> stream2 = Stream.of(arr2);
Stream<String> concat = Stream.concat(stream1, stream2);
concat.forEach(name-> System.out.println(name));
System.out.println("---------");
// 简化
Stream.concat(
Stream.of("张三丰", "张翠山", "赵敏", "周芷若", "张无忌"),
Stream.of("美羊羊", "喜洋洋", "懒洋洋", "灰太狼", "红太狼")
).forEach(name-> System.out.println(name));
}
}
02-方法引用
方法引用能简化以下场景: (方法名后不要写小括号)
1. 通过对象名引用成员方法 对象名::成员方法名 System.out::println
2. 通过类名引用静态方法 类名::静态方法名 i -> Math.abs(i) Math::abs
3. 通过super引用父类成员方法 super::父类方法名 ()->super.eat(); super::eat
4. 通过this引用本类成员方法 this::本类方法名 ()->this.eat(); this::eat
5. 引用某个类的构造方法 类名::new name->new Person(name) Person::new
6. 引用创建数组的方法 数据类型[]::new length->new int[length]; int[]::new
1. 通过对象名引用成员方法 对象名::成员方法名 System.out::println
[Java] 纯文本查看 复制代码 // 定义函数式接口: 模拟Java中已经提供的函数式接口
@FunctionalInterface
public interface Printable {
// 定义功能: 打印s. 功能怎么实现先不管
void print(String s);
}
// 定义一个类: 模拟Java中已经定义好的类和方法
public class MethodRefObject {
// 将字符串变大写再打印出来
public void printUpperCaseString(String str){
System.out.println(str.toUpperCase());
}
}
public class Test {
// 定义方法用于打印Hello, 参数为 打印字符串的方式
public static void printString(Printable lambda){
lambda.print("Hello");
}
public static void main(String[] args) {
// 普通Lambda方式
printString((s)->{
// 创建对象, 调用已经存在方法printUpperCase
MethodRefObject obj = new MethodRefObject();
obj.printUpperCaseString(s);
});
// 使用方法引用简化
MethodRefObject obj = new MethodRefObject();
printString(obj::printUpperCaseString);
// 直接引用obj的printUpperCaseString方法作为抽象方法print的实现
}
}
方法引用: 通过类名引用静态方法通过类名引用静态方法
通过类名引用静态方法 类名::静态方法名 i -> Math.abs(i) Math::abs
[Java] 纯文本查看 复制代码 public class Test {
//定义方法用来计算一个数的结果. 参数: 被计算的整数 和 计算的方式
public static int method(int number, Calcable c){
return c.calcAbs(number);
}
public static void main(String[] args) {
// Lambda方式调用method方法: 传递-10和Lambda, 调用Math.abs()方法计算出结果, 并返回
int result1 = method(-10, (number)->{return Math.abs(number);});
System.out.println(result1); // 10
// 方法引用方式调用method方法: 传递-10, 直接引用Math类中的abs方法
int result2 = method(-10, Math::abs);
System.out.println(result2); // 10
}
}
方法引用: 通过super引用父类成员方法通过super引用父类成员方法
super::父类方法名 只能在子类中写,不能在main方法中写
[Java] 纯文本查看 复制代码 // 定义见面打招呼的函数式接口: 模拟Java中已经提供的函数式接口
@FunctionalInterface
public interface Greetable {
// 见面打招呼的方法
void greet();
}
// 定义父类: 人类
public class Human {
//定义一个sayHello的方法
public void sayHello(){
System.out.println("Hello 我是Human!");
}
}
// 定义子类: 男人类, 继承人类
public class Man extends Human {
// 子类重写了父类sayHello的方法
@Override
public void sayHello() {
System.out.println("Hello 我是Man!");
}
//定义方法用来见面打招呼. 参数是见面打招呼的方式
public void method(Greetable g){
g.greet();
}
// 因为要使用super, 所以定义了一个成员方法
public void show(){
// Lambda方式: 创建父类对象, 用父类对象调用方法
method(()->{
Human h = new Human(); // 创建父类Human对象
h.sayHello(); // 调用父类的sayHello方法
});
// Lambda方式: 直接使用super调用父类方法, 省去创建父类对象的麻烦
method(()->{
super.sayHello(); // 直接使用super调用父类的sayHello方法
});
// 方法引用方式: 直接引用父类的sayHello方法作为greet抽象方法的实现
method(super::sayHello); // 其实就是 对象名::成员方法名 只不过对象用super代表
}
public static void main(String[] args) {
new Man().show();
}
}
方法引用: 通过this引用本类成员方法
通过this引用本类成员方法
this::本类方法名
[Java] 纯文本查看 复制代码 // 定义一个表示富有的函数式接口, 模拟Java中提供的函数式接口
@FunctionalInterface
public interface Richable {
//定义一个想买什么就买什么的方法
void buy();
}
// Husband类, 表示一个要买房结婚丈夫
public class Husband {
// 本类中已经存在的方法: 买房子的方法
public void buyHouse(){
System.out.println("北京二环内买一套四合院!");
}
//定义结婚的方法, 参数: 传递Richable接口
public void marry(Richable r){
r.buy();
}
//定义非常高兴的方法: 因为要调用this, 所以定义这个非静态方法
public void soHappy(){
// Lambda方式: 使用this调用本类已经存在的方法
marry(()->{
this.buyHouse();
});
// 方法引用方式: 直接通过this引用本类中的buyHouse方法
marry(this::buyHouse); // 其实也是 对象名::成员方法名 只不过对象名用this表示
}
public static void main(String[] args) {
new Husband().soHappy();
}
}
方法引用: 类的构造方法引用引用某个类的构造方法
类名::new
[Java] 纯文本查看 复制代码 // Person类
public class Person {
private String name;
public Person() {
}
public Person(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
// 定义用于创建Person对象的函数式接口, 模拟Java中已经提供的函数式接口
@FunctionalInterface
public interface PersonBuilder { // Function转换
// 根据姓名创建Person对象, 怎么创建需要Lambda来实现
Person builderPerson(String name);
}
// 测试类
public class Test {
// 定义方法, 用于根据传入的姓名创建一个Person对象并打印名称
// 参数: 姓名 和 创建Person对象的方式
public static void printName(String name, PersonBuilder pb){
Person person = pb.builderPerson(name);
System.out.println(person.getName());
}
public static void main(String[] args) {
// Lambda方式: 传递姓名和创建对象的方式
printName("迪丽热巴",(String name)->{
return new Person(name); // 仅仅调用了Person类的构造方法
});
// 方法引用方式: 直接引用Person类的构造方法
printName("古力娜扎",Person::new);
}
}
方法引用: 数组的构造方法引用
引用创建数组的方法
数据类型[]::new
[Java] 纯文本查看 复制代码 // 定义创建数组的函数式接口, 模拟Java中已经提供的函数式接口
@FunctionalInterface
public interface ArrayBuilder {
// 根据指定length长度, 创建一个int[]数组. 怎么创建需要我们传递Lambda实现
int[] builderArray(int length);
}
// 测试类
public class Test {
// 定义方法用于创建数组
// 参数: 数组的长度 和 创建数组的方式
public static int[] createArray(int length, ArrayBuilder ab){
return ab.builderArray(length);
}
public static void main(String[] args) {
// Lambda方式: 传递长度10, 以及创建数组的Lambda表达式
int[] arr1 = createArray(
10,
(len)->{
return new int[len];
});
System.out.println(arr1.length); //10
// 方法引用方式: 传递长度10, 和数组的构造方法
int[] arr2 = createArray(10, int[]::new);
System.out.println(arr2.length); //10
}
}
|
|