Java高级重点知识点-25-Stream流、方法引用

文章目录

  • Stream流
    • 流式思想概述
    • 获取流
    • 常用方法
  • 方法引用
    • 方法引用符
    • 通过对象名引用成员方法
    • 通过类名称引用静态方法
    • 通过super引用成员方法
    • 通过this引用成员方法
    • 类的构造器引用
    • 数组的构造器引用

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);}}
}

在这里插入图片描述
这段代码中我们可以看到,我们首先通过遍历集合list来获取到了所有姓张的人,然后我们通过遍历shortList集合来获取到了名字长度为三个字的人。最终打印出了名字三个字并且姓张的人,很显然这个过程很麻烦,下面我们来看一下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);}
}

在这里插入图片描述
这里我们可以看到我们通过filter()过滤方法过滤掉了姓张且名字长度为3的人,并且通过方法引用来输出打印。

流式思想概述

流式思想类似于工厂车间的“生产流水线”
Stream流”其实是一个集合元素的函数模型,它并不是集合,也不是数据结构,其本身并不存储任何元素(或其地址值)。
Stream(流)是一个来自数据源的元素队列

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

获取流

java.util.stream.Stream<T> 是Java 8新加入的最常用的流接口

  • 所有的 Collection 集合都可以通过 stream 默认方法获取流;
  • Stream 接口的静态方法 of 可以获取数组对应的流。
  1. 根据Collection获取流
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. 根据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. 根据数组获取流
    Stream 接口中提供了静态方法of来实现数组的流化。
import java.util.stream.Stream;
public class Demo06GetStream {public static void main(String[] args) {String[] array = { "张无忌", "张翠山", "张三丰", "张一元" };Stream<String> stream = Stream.of(array);}
}

常用方法

  • 延迟方法:返回值类型仍然是 Stream 接口自身类型的方法,因此支持链式调用。(除了终结方法外,其余方法均为延迟方法。)
  • 终结方法:返回值类型不再是 Stream 接口自身类型的方法,因此不再支持类似 StringBuilder 那样的链式调用。

逐一处理:forEach

  • void forEach(Consumer<? super T> action);
import java.util.stream.Stream;
public class StreamForEach {public static void main(String[] args) {Stream<String> stream = Stream.of("张三", "张三丰", "周姐");stream.forEach(name‐> System.out.println(name));}
}

过滤:filter

  • Stream<T> filter(Predicate<? super T> predicate);
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

  • <R> Stream<R> map(Function<? super T, ? extends R> mapper); 将流中的元素映射到另一个流中
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

  • long count(); 这里的返回值是long类型,不再和旧集合一样是int类型,也就是说可以用来获取更大的集合对象的大小。
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

  • Stream<T> limit(long maxSize); 对流进行截取,只取用前maxSize个
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

  • Stream<T> skip(long n); skip 方法获取一个截取之后的新流
import java.util.stream.Stream;
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

  • static <T> Stream<T> concat(Stream<? extends T> a, Stream<? extends T> b) 两个流合并成为一个流
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);}
}

方法引用

冗余的Lambda场景:

  1. 首先定义一个函数式接口
@FunctionalInterface
public interface Printable {void print(String str);
}
  1. 通过lambda表达式来使用该字符串
public class PrintSimple {private static void printString(Printable data) {data.print("Hello, World!");}public static void main(String[] args) {printString(s ‐> System.out.println(s));}
}

对字符串进行控制台打印输出的操作方案,明明已经有了现成的实现,那就是 System.out对象中的 println(String) 方法。因此我们不必自己动手调用该方法。使用方法引用即可。

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

方法引用符

双冒号 :: 为引用运算符,而它所在的表达式被称为方法引用。

  • Lambda表达式写法: s -> System.out.println(s); 语义:拿到参数之后经Lambda之手,继而传递给 System.out.println 方法去处理。
  • 方法引用写法: System.out::println 语义: 直接让 System.out 中的 println 方法来取代Lambda。

注意: Lambda 中 传递的参数 一定是方法引用中 的那个方法可以接收的类型,否则会抛出异常
函数式接口是Lambda的基础,而方法引用是Lambda的孪生兄弟。因此方法引用同时也是可推导可省略的;

通过对象名引用成员方法

  1. 首先定义一个函数式接口
@FunctionalInterface
public interface Printable {void print(String str);
}
  1. 创建一个实现了指定方法的类
public class MethodRefObject {public void printUpperCase(String str) {System.out.println(str.toUpperCase());}
}
  1. 测试
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);}
}

在这里插入图片描述
这里我们可以看到,我们通过方法引用了MethodRefObject中的printUpperCase(String str)方法,并没有通过lambda表达式来实现,而是使用一个已经实现了类中的方法,与我们引用System.out对象中的println()方法类似。

通过类名称引用静态方法

通过java.lang.Math 类中的静态方法 abs举例

  1. 声明一个函数式接口
@FunctionalInterface
public interface Calcable {int calc(int num);
}
  1. 使用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));}
}
  1. 使用方法引用的方式来实现
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);}
}

通过super引用成员方法

如果存在继承关系,当Lambda中需要出现super调用时,也可以使用方法引用进行替代。

  1. 定义函数式接口
@FunctionalInterface
public interface Greetable {void greet();
}
  1. 定义父类
public class Human {public void sayHello() {System.out.println("Hello!");}
}
  1. 定义子类(Lambda)
public class Man extends Human {@Overridepublic 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();});//简化Lambdamethod(()->new Human().sayHello());//使用super关键字代替父类对象method(()->super.sayHello());}
}
  1. 定义子类(方法引用)
public class Man extends Human {@Overridepublic void sayHello() {System.out.println("大家好,我是Man!");}//定义方法method,参数传递Greetable接口public void method(Greetable g){g.greet();}public void show(){method(super::sayHello);}
}

通过this引用成员方法

这里与父类super类似,直接展示通过方法引用的写法

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

类的构造器引用

  1. 定义一个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;}
}
  1. 定义用来创建Person 对象的函数式接口
public interface PersonBuilder {Person buildPerson(String name);
}
  1. 使用Lambda表达式创建指定名称的Person对象
public class Demo09Lambda {public static void printName(String name, PersonBuilder builder) {System.out.println(builder.buildPerson(name).getName());}public static void main(String[] args) {printName("赵丽颖", name ‐> new Person(name));}
}
  1. 使用方法引用创建指定名称的Person对象
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);}
}

数组的构造器引用

数组也是 Object 的子类对象,所以同样具有构造器

  1. 定义一个函数式接口
@FunctionalInterface
public interface ArrayBuilder {int[] buildArray(int length);
}
  1. 使用lambda表达式应用该接口
public class Demo11ArrayInitRef {private static int[] initArray(int length, ArrayBuilder builder) {return builder.buildArray(length);}public static void main(String[] args) {int[] array = initArray(10, length ‐> new int[length]);}
}
  1. 使用方法引用应用该接口
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);}
}

欢迎java热爱者了解文章,作者将会持续更新中,期待各位友友的关注和收藏。。。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/45205.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

实现Android夜间模式主题:从入门到精通

实现Android夜间模式主题:从入门到精通 随着用户对夜间模式的需求越来越高,Android开发者需要掌握如何在应用中实现夜间模式。本文将详细介绍在Android中实现夜间模式的步骤,包括配置、实现、以及一些最佳实践,帮助开发者创建更具吸引力和用户友好的应用。 夜间模式的优势…

Redis基础教程(二十):Java使用Redis

&#x1f49d;&#x1f49d;&#x1f49d;首先&#xff0c;欢迎各位来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里不仅可以有所收获&#xff0c;同时也能感受到一份轻松欢乐的氛围&#xff0c;祝你生活愉快&#xff01; &#x1f49d;&#x1f49…

华贝甄选干细胞科技,揭秘生命修复的奥秘

在探索生命奥秘的漫漫征途中&#xff0c;华贝甄选凭借干细胞科技的神奇力量&#xff0c;为您点亮健康与活力的希望之光。 我们深知&#xff0c;细胞是生命的基石&#xff0c;而干细胞则是这基石中蕴含的无限潜能。华贝甄选精心打造的干细胞疗法&#xff0c;如同神奇的魔法&…

2024SpringCloud学习笔记

远程调用Rest Template 服务注册与发现&分布式配置管理 Consul 下载安装 官网https:/ldeveloper.hashicorp.com/consul/downloads 开发者模式启动consul agennt -dev 浏览器访问本地端口:8500 服务注册与发现 Maven引入 <!--SpringCloud consul discovery -->…

【Python实战因果推断】31_双重差分2

目录 Canonical Difference-in-Differences Diff-in-Diff with Outcome Growth Canonical Difference-in-Differences 差分法的基本思想是&#xff0c;通过使用受治疗单位的基线&#xff0c;但应用对照单位的结果&#xff08;增长&#xff09;演变&#xff0c;来估算缺失的潜…

小阿轩yx-NoSQL 之 Redis 配置与优化

小阿轩yx-NoSQL 之 Redis 配置与优化 Redis 数据库介绍 是一个非关系型数据库 关系数据库与非关系型数据库 按照数据库结构划分的 关系型数据库 是一个结构化的数据库&#xff0c;创建在关系模型基础上&#xff0c;一般面向于记录借助集合代数等数学概念和方法处理数据库…

215.Mit6.S081-实验三-page tables

在本实验室中&#xff0c;您将探索页表并对其进行修改&#xff0c;以简化将数据从用户空间复制到内核空间的函数。 一、实验准备 开始编码之前&#xff0c;请阅读xv6手册的第3章和相关文件&#xff1a; kernel/memlayout.h&#xff0c;它捕获了内存的布局。kernel/vm.c&…

Python:Python基础知识(注释、命名、数据类型、运算符)

.注释 Python有两种注释方法&#xff1a;单行注释和多行注释。单行注释以#开头&#xff0c;多行注释以三个单引号 或三个双引号 """ 开头和结尾。 2.命名规则 命名规则: 大小写字母、数字、下划线和汉字等字符及组合&#xff1b; 注意事项: 大小写敏感、首…

Linux环境下Oracle 11g的离线安装与配置历程

在成功体验了 Windows 版本的Oracle 11g 后&#xff0c;这几天心血来潮&#xff0c;决定再挑战一下Linux 环境下的安装&#xff0c;特别是在考虑到部门内部虚拟机无法联网的情况下&#xff0c;我选择了在CentOS 7上进行离线安装。这次安装之旅&#xff0c;主要参考了下面大佬的…

【计算机科学】CCF-C特刊征稿合集,见刊快,期刊质量高,速投!

期刊推荐 期刊名称&#xff1a;ACTA INFORMATICA 主题包括以下项目的理论方面。 算法及其分析 自动机和形式语言 可计算性和复杂性 数据处理 离散数学 逻辑学&#xff08;计算机科学&#xff09; 人工智能的数学基础 编程语言理论 安全 系统理论 验证 中科院四区 …

STM32智能物流机器人系统教程

目录 引言环境准备智能物流机器人系统基础代码实现&#xff1a;实现智能物流机器人系统 4.1 数据采集模块 4.2 数据处理与导航算法 4.3 通信与网络系统实现 4.4 用户界面与数据可视化应用场景&#xff1a;物流机器人管理与优化问题解决方案与优化收尾与总结 1. 引言 智能物流…

mindspore打卡23天之微调本地MindNLP ChatGLM-6B StreamChat

MindNLP ChatGLM-6B StreamChat 本案例基于MindNLP和ChatGLM-6B实现一个聊天应用。 1 环境配置 %%capture captured_output # 实验环境已经预装了mindspore2.2.14&#xff0c;如需更换mindspore版本&#xff0c;可更改下面mindspore的版本号 !pip uninstall mindspore -y !p…

基于JavaSpringBoot+Vue+uniapp微信小程序校园宿舍管理系统设计与实现(7000字论文参考+源码+LW+部署讲解)

博主介绍&#xff1a;硕士研究生&#xff0c;专注于信息化技术领域开发与管理&#xff0c;会使用java、标准c/c等开发语言&#xff0c;以及毕业项目实战✌ 从事基于java BS架构、CS架构、c/c 编程工作近16年&#xff0c;拥有近12年的管理工作经验&#xff0c;拥有较丰富的技术架…

Linux:NFS共享存储

目录 一、NFS基本概述 二、NFS共享文件实验 2.1、安装nfs和rpcbind软件 2.2、修改配置文件设置共享 2.3、创建共享目录 ​编辑 2.4、开启服务 2.5、客户端验证共享目录可访问 三、tcpdump命令 3.1、概述 3.2、简单表达 3.3、过滤规则 ​编辑 3.4、tcpdump常见参数…

强化学习实战2:动手写迷宫环境

迷宫环境介绍与创建 迷宫环境图示如下&#xff1a; 如图所示&#xff0c;其为一个 三乘三 的网格世界&#xff0c;我们要让 agent 从 S0 采取策略出发&#xff0c;然后走到 S8&#xff0c;图中红线部分表示障碍不能逾越&#xff0c;其中 S1 和 S4 之间有一个障碍&#xff0c;S…

C语言有哪些特点?

C语言是一种结构化语言&#xff0c;它有着清晰的层次&#xff0c;可按照模块的方式对程序进行编写&#xff0c;十分有利于程序的调试&#xff0c;且c语言的处理和表现能力都非常的强大&#xff0c;依靠非常全面的运算符和多样的数据类型&#xff0c;可以轻易完成各种数据结构的…

Kotlin MultiPlatform(KMP)

Kotlin MultiPlatform 1.KMP 是什么 Kotlin Multiplatform 是一个工具&#xff0c;它让我们用同一种编程语言&#xff08;Kotlin&#xff09;写代码&#xff0c;这些代码可以同时在不同的设备上运行&#xff0c;比如手机、电脑和网页。这样做可以节省时间&#xff0c;因为你不…

1、项目目录设计

文章目录 前言一、项目目录设计 前言 本项目我们将会完成一个Go项目开发框架&#xff0c;该项目不会包含具体的CRUD业务代码&#xff0c;而是从头搭建一个工作中实用的开发框架。让开发者能够熟悉整个项目的搭建流程&#xff0c;能够独立完成项目从0到1的搭建&#xff0c;而且…

【RHCE】实验(HTTP,DNS,SELinux,firewalld的运用)

一、题目 二、主服务器配置 1.下载HTTP服务&#xff0c;DNS服务 [rootlocalhost ~]# yum install -y httpd bind 2.开启防火墙&#xff0c;放行服务 # 开启防火墙 [rootlocalhost ~]# systemctl start firewalld # 放行服务 [rootlocalhost ~]# firewall-cmd --add-service…

上班摸鱼吗?一文详解代码生成神器-Velocity

引言 “我不是在教你学坏,而是教你如何提高生产效率。” ----------- 牛顿 人类社会能够一直进步发展出现在的文明世界,最大的一个原因就是这个世界上懒人居多,懒人为了偷懒就需要提高生产效率,效率提高节省下来的时间才能创造出艺术、娱乐以及更高效率的科学技术。程序员…