JDK新特性(Lambda表达式,Stream流)

Lambda表达式:

Lambda 表达式背后的思想是函数式编程(Functional Programming)思想。在传统的面向对象编程中,程序主要由对象和对象之间的交互(方法调用)构成;而在函数式编程中,重点在于函数的应用和组合。

以上就是Lambda表示的背后思想,做为了解即可

主要要看怎么用:

Lambda表达式的格式:

 ()->{}

        () : 重写方法的参数位置
        -> : 将参数传递到方法体中
        {} : 重写方法的方法体 

直接看代码:

public class Test03 {public static void main(String[] args) {ArrayList<Integer> list = new ArrayList<>();list.add(1);list.add(2);list.add(8);list.add(0);list.add(4);Collections.sort(list, new Comparator<Integer>() {@Overridepublic int compare(Integer o1, Integer o2) {return o1-o2;}});System.out.println(list);Collections.sort(list,((o1, o2) -> o1-o2));System.out.println(list);}
}

上面是用Comparator比较器来定义排序规则

下面则是用Lambda表达式

我们对照起来看,就能发现(o1,o2)就是方法的参数  -> 后面就是方法体。

Lambda表达式使用前提:

在使用Lambda表达式之前,我们需要确认必须是函数式接口做为参数传递:

什么是函数式接口?

有且只有一个抽象方法的接口,用@FunctionalInterface去检测

我们可以点击这个Comparator这个方法

 还有我们在线程章节经常用的Runnable接口

new Thread(()->System.out.println(Thread.currentThread().getName())).start();

这时候可能有人会想,Lambda表达式既然只能用函数式接口做为参数,那这个不是很鸡肋嘛

难道我要自己去写一个接口,里面专门再只放一个抽象方法,那这不是更麻烦了嘛

反正我刚刚学的时候,我确实是有这样的疑问

后面问了GPT,其实这个lambda表达式大部分应该是和后面的Stream流一起使用的


Stream流:

Stream流中的"流"不是特指"IO流",它是一种"流式编程"(编程方式),可以看做是"流水线"

个人对流的理解:

在我学习了流的大致用法之后,我其实觉得这个流就是一个操作数组和列表的工具类

里面提供了一系列的方法来操作数组和列表

下面介绍一下Stream流的方法:

Stream流的获取:

这个Stream流的获取就是把数组或者列表转化成流

public class Demo02Stream {public static void main(String[] args) {//1.针对集合:Collection中的方法//Stream<E> stream()ArrayList<String> list = new ArrayList<>();list.add("张三");list.add("李四");list.add("王五");Stream<String> stream = list.stream();System.out.println(stream);//2.针对数组:Stream接口中的静态方法://static <T> Stream<T> of(T... values)Stream<String> stream1 = Stream.of("金莲", "三上", "松下");System.out.println(stream1);}
}
常用方法:
  1. filter(Predicate):过滤方法,根据指定的条件对元素进行筛选。

  2. map(Function):映射方法,将元素按照指定的映射规则进行转换。

  3. forEach(Consumer):遍历方法,对流中的每个元素执行指定的操作。

  4. collect(Collectors):收集方法,将流中的元素收集到一个集合中。

  5. distinct():去除流中的重复元素。

  6. limit(long):截取流中的前几个元素。

  7. skip(long):跳过流中的前几个元素。

  8. count():统计流中元素的个数。

 直接上一段代码:

public class Test01 {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("嫪毐");//1:筛选出姓张的人System.out.println("筛选出姓张的人");list.stream().filter(s -> s.startsWith("张")).forEach(s -> System.out.print(s+" "));//2:筛选出姓张且名字为两个字的人System.out.println();System.out.println("筛选出姓张且名字为两个字的人");list.stream().filter(s -> s.startsWith("张")).filter(s -> s.length()==2).forEach(s -> System.out.print(s+" "));//3:计算列表的个数System.out.println();System.out.println("计算列表的个数");long count = list.stream().count();System.out.println(count);//4:返回流的前n个对象System.out.println();System.out.println("返回流的前2个对象");Stream<String> limit = list.stream().limit(2);limit.forEach(s -> System.out.print(s+" "));//5:跳过Stream流对象中的前n个元素,返回一个新的Stream流对象System.out.println();System.out.println("跳过Stream流对象中的前n个元素,返回一个新的Stream流对象");list.stream().skip(2).forEach(s-> System.out.print(s+" "));//6:两个流合成一个流System.out.println();System.out.println("两个流合成一个流");ArrayList<String> newlist = new ArrayList<>(Collections.nCopies(list.size(), ""));Collections.copy(newlist,list);System.out.println(newlist);Stream.concat(newlist.stream(),list.stream()).forEach(s -> System.out.print(s+" "));//7:转换流中的类型System.out.println();System.out.println("转换流中的类型");Stream<Integer> stream = Stream.of(1, 2, 3, 4, 5, 6);stream.map(Integer -> Integer+"").forEach(s -> System.out.print(s+" "));}
}

下面来一个项目中的稍微复杂一点的代码加深印象:

在学这个JDK新特性这一章的时候,刚好在写伙伴匹配系统

下面这段鱼皮老师写的代码,我那个时候一直看不懂

现在算是能看懂了

这里的userList就是在数据库中查出来的所有用户列表

这里的tagNameList就是前端传过来的标签列表

所以这一段的逻辑就是根据前端传过来的标签列表进行查找有相同标签的用户

return userList.stream().filter(user -> {String tagsStr = user.getTags();Set<String> tempTagNameSet = gson.fromJson(tagsStr, new TypeToken<Set<String>>() {}.getType());tempTagNameSet = Optional.ofNullable(tempTagNameSet).orElse(new HashSet<>());for (String tagName : tagNameList) {if (!tempTagNameSet.contains(tagName)) {return false;}}return true;}).map(this::getSaftyUser).collect(Collectors.toList());

我们来分析一下整体的代码逻辑:

  1. 将userList转成流对象
  2. 转成流对象之后再从对应的用户中取出标签,因为我这里将标签定义成了一个String类型所以后面就将这个String转成了JSON格式的数据存到这个Set集合中
  3. 用了Optional来对这个转换后的集合做了一个判空处理,如果为空,就new一个空集合
  4. 然后遍历前端传过来的标签列表,如果判断标签列表中有和用户的标签列表不同的元素直接返回false,这样就被流给过滤掉了
  5. 接着再对符合要求的用户做一个映射:map(this::getSaftyUser),这个就相当于:map(this::getSaftyUser),将当前对象(this)的 getSaftyUser 方法应用到 Stream 中的每个元素上,当我们执行完这个filter方法之后,里面的用户就是我们需要的用户了,我们需要对用户进行一个脱敏将一些重要信息隐藏起来。
  6. 然后再将我们粉装好的流对象转成列表返回即可

如果上面的逻辑有点难,举个例子,因为我自己一开始也想了很久

比如有三个用户ABC

他们的标签分别是

A : Java ,男,大一

B : Python ,男,大二

C : C++ ,女,大三

然后这个时候我们传入的tagNameList是:男,大一

执行流程就是:我们会先将上面的三个用户取出来成一个列表就是userList

然后将这个列表转换成流对象,接着将这个列表中的每一个对象都执行后面Lambda表达式中的操作,有点像遍历一遍这个列表,比如将A用户的标签取出来 Java ,男,大一,封装成一个Set<String>集合,再遍历tagNameList这个列表:男,大一,挨个判断,男是否在这个集合中,在就继续判断大一这个标签是否在这个集合中,在就结束循环,返回true,就把这个A用户装到流中。

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

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

相关文章

postman给全部接口添加请求头数据(如token)

如果给没有一个接口添加请求头token就太慢了&#xff0c;如下图。可以点击所有接口的所属的目录。点击“Scripts”&#xff0c;点击Pre-request按钮。加入代码&#xff1a; pm.request.addHeader("Authorization:eyJhbGciOiJIUzI1NiIsInR5cCI111pXVCJ9.eyJjbGFpbXMiOnsiaW…

Nginx负载均衡策略

upstream机制提供了负载均衡的功能,可以讲请求负载分担到集群服务器的某个服务器上 打包时候到时一个8085 一个8090 一个8095 nohup /usr/local/develop/jdk-17.0.10/bin/java -Xmx256m -Xms256m -jar nginx-demo-8085.jar > server8085.log 2>&1 & nohup /u…

56_Redis简单命令

一、引言 1.1 数据库压力过大 由于用户量增大&#xff0c;请求数量也随之增大&#xff0c;数据压力过大 一个请求的url 背后可能有有4-5个 sql的操作 每秒钟 qps&#xff08;并发数&#xff09; 1000 背后的sql操作 4000-5000mysql 单机并发量读写 8000-10000 &#x…

鸿蒙配置Version版本号,并获取其值

app.json5中配置版本号&#xff1a; 获取版本号&#xff1a; bundleManager.getBundleInfoForSelf(bundleManager.BundleFlag.GET_BUNDLE_INFO_WITH_APPLICATION).then((bundleInfo) > {let versionName bundleInfo.versionName; //应用版本号}).catch((error: BusinessE…

【Vulnhub系列】Vulnhub_DC-1靶场渗透(原创)

【Vulnhub系列靶场】Vulnhub_DC-1靶场渗透 原文转载已经过授权 原文链接&#xff1a;Lusen的小窝 - 学无止尽&#xff0c;不进则退 (lusensec.github.io) 一、环境准备 1、在百度网盘中下载DC-1靶场。DC-1靶场受virtual box 的影响&#xff0c;在VM中直接打开是扫描不到IP 的…

jmeter录制

1、添加代理服务器 添加方法&#xff1a;“测试计划”右键 -> 添加 -> 非测试元件 -> HTTP代理服务器 2、添加线程组 添加方法&#xff1a;“测试计划”右键->添加->线程&#xff08;用户&#xff09;->线程组 3、配置http代理服务器 &#xff08;1&a…

第三方软件测试报告可做哪些用途?

1. 评估软件质量&#xff1a;第三方软件测试报告通过对软件的各项性能指标进行测试和分析&#xff0c;能够客观地评估软件的质量水平。这份报告可以为软件的开发团队提供反馈&#xff0c;帮助他们发现和修复潜在的问题&#xff0c;提高软件的质量和稳定性。 2. 验证软件功能&a…

<数据集>手机识别数据集<目标检测>

数据集格式&#xff1a;VOCYOLO格式 图片数量&#xff1a;16172张 标注数量(xml文件个数)&#xff1a;16172 标注数量(txt文件个数)&#xff1a;16172 标注类别数&#xff1a;1 标注类别名称&#xff1a;[Phone] 使用标注工具&#xff1a;labelImg 标注规则&#xff1a;…

探索分布式光伏运维系统的组成 需要几步呢?

前言 随着光伏发电的不断发展&#xff0c;对于光伏发电监控系统的需求也日益迫切&#xff0c;“互联网”时代&#xff0c;“互联网”的理念已经转化为科技生产的动力&#xff0c;促进了产业的升级发展&#xff0c;本文结合“互联网”技术提出了一种针对分散光伏发电站运行数据…

c#调用python代码,实现读取npy的数据并显示图像

本例子实现的功能是&#xff1a; 根据stat.npy、ops.npy两个npy文件的内容&#xff0c;显示图形 1. 用python代码实现读取两个文件&#xff0c;文件名为read_npy.py&#xff0c;代码如下&#xff1a; import numpy as npdef read_npy_files(stat_file, ops_file):stat np.lo…

NSL-KDD入侵检测系统的设计与实现系列预告

每日进阶-基于机器学习的入侵检测系统——打怪升级之道 在当今的数字时代&#xff0c;网络安全不仅是防御&#xff0c;更是主动出击。你是否想知道如何用机器学习技术设计一套入侵检测系统&#xff08;IDS&#xff09;&#xff0c;让黑客无所遁形&#xff1f;本系列文章将为您揭…

【全志H616开发】Linux守护进程

文章目录 守护进程简介基本特点创建一个守护进程通常涉及以下步骤&#xff1a;进程查看指令&#xff1a; 守护进程开发代码示例&#xff1a; 开机自动启动 守护进程 简介 Linux Daemon&#xff08;守护进程&#xff09;是运行在后台的一种特殊进程。它独立于控制终端并且周期性…

VScode | 我的常用插件分享

系列文章目录 本系列文章主要分享作位前端开发的工具之------VScode的使用分享。 文章目录 目录 系列文章目录 文章目录 前言 一、Vetur 三、别名路径跳转 四、Prettier 五、koroFileHeader 六、vue-helper 总结 前言 本文主要分享VScode的好用插件。 一、Vetur Vue的Vetur插…

【Vulnhub系列】Vulnhub_Raven2靶场渗透(原创)

【Vulnhub系列靶场】Vulnhub_Raven2 渗透 原文转载已经过授权 原文链接&#xff1a;Lusen的小窝 - 学无止尽&#xff0c;不进则退 (lusensec.github.io) 一、环境准备 从网盘下载该靶机&#xff0c;在vm中选择【打开】 然后设置好存储路径&#xff0c;开机后检查靶机的网络连…

谷粒商城实战笔记-84-商品服务-API-新增商品-获取分类关联的品牌

文章目录 一&#xff0c;品牌查询接口的后台实现二&#xff0c;编码经验总结1&#xff0c;Controller层的作用1.1 参数处理1.2 调用Service1.3 处理Service返回结果实例 2&#xff0c;VO的封装时机3&#xff0c;Service中最好注入Service&#xff0c;不要直接依赖Dao 问题记录 …

vue2 vue3 props 的处理机制

在 Vue 2 中&#xff0c;props 是单向数据流&#xff0c;父组件向子组件传递的 props 默认情况下是不具有响应式特性的。这意味着当父组件的数据发生变化时&#xff0c;如果传递给子组件的 props 发生变化&#xff0c;子组件不会自动更新视图。 具体来说&#xff0c;在 Vue 2 …

增量学习中Task incremental、Domain incremental、Class incremental 三种学习模式的概念及代表性数据集?

1 概念 在持续学习领域&#xff0c;Task incremental、Domain incremental、Class incremental 是三种主要的学习模式&#xff0c;它们分别关注不同类型的任务序列和数据分布变化。 1.1 Task Incremental Learning (Task-incremental) 任务增量学习&#xff0c;也称为任务增…

【论文共读】【翻译】【GAN】Generative Adversarial Nets

论文原文地址&#xff1a;https://arxiv.org/pdf/1406.2661 翻译&#xff1a;Generative Adversarial Nets 生成对抗网络 0. 摘要 提出了一种新的对抗过程估计生成模型的框架&#xff0c;其中我们同时训练两个模型&#xff1a;一个是捕获数据分布的生成模型G&#xff0c;另一…

燃气安全无小事,一双专业劳保鞋让你步步安心!

燃气作为我们日常生活中不可或缺的能源之一&#xff0c;为我们的生活提供了极大便利&#xff0c;其安全性往往被忽视在忙碌的日常生活背后。然而&#xff0c;燃气事故一旦发生&#xff0c;后果往往不堪设想&#xff0c;轻则财产损失&#xff0c;重则危及生命。因此&#xff0c;…