迅速看一下jdk8
一直在看java并发的感觉说的有点多,就看点简单的放松一下吧!这次来简单说一下jdk8,很久没用,都陌生了,仔细看看还挺有意思的,让我们大脑转化一个角度来写代码;因为我们现在平常大部分用jdk7写代码,我们都是在想着这一步怎么做,下一步怎么做;而jdk8只需要知道这一步做什么,下一步做什么,思维的转换很有意思;
首先说说什么叫做行为参数化?简单的来说就是传递的是一个行为,可以想象成传递一个lambda表达式,其中lambda表达式就不多说了;
举个例子:
package com.example.demo.vo; import lombok.Data;
import lombok.experimental.Accessors; @Data
@Accessors(chain = true)
public class Apple {
private String color;
private int weight; }
假如一个集合中有很多苹果,我们首先要筛选出重量大于10的苹果,那么我们需要定义一个这样的方法:
static List<Apple> filterColor(List<Apple>inventory){
List<Apple> result = Lists.newArrayList();
for (Apple apple : inventory) {
if (apple.getWeight()>10) {
result.add(apple);
}
}
return result;
}
如果有一天突然需求改变了,要求我们筛选出颜色为绿色的苹果,于是我们又要定义一个这样的方法:
static List<Apple> filterGreen(List<Apple>inventory){
List<Apple> result = Lists.newArrayList();
for (Apple apple : inventory) {
if (Objects.equal("green", apple.getColor())) {
result.add(apple);
}
}
return result;
}
如果又有一天提了某某需求,于是吧啦吧啦,那么我们看看简单的看看这两个方法有什么不同啊,其实仔细一看,就是上面的for循环中if语句中筛选条件不一样,其他的代码直接复制粘贴的,我们知道复制粘贴有的时候太多了,你会看到很多重复的代码,这就很坑了,有没有比较简化一点的方法呢?
其实很容易,既然上面其他部分都是一样的,我们把一样的部分提出来,当做一个模板,以后我们只需要传递我们定制的筛选条件不就行了吗?简单吧!那么问题又来了,怎么把那些代码编程一个模板呢?在jdk8中有一些函数式接口,其中一个就是Predicate,注解@FunctionalInterface翻译一下就是函数式接口嘛!我们暂时就用它的test方法,可以看到这个方法接收一个形参,返回一个boolean类型的,上面的代码中的筛选条件本质上就是接收一个Apple类型,返回一个boolean类型嘛!
于是我们可以这样做制作一个模板出来:
static List<Apple> filterApples(List<Apple> inventory, Predicate<Apple> p) {
List<Apple> result = new ArrayList<>();
for (Apple apple : inventory) {
if (p.test(apple)) {
result.add(apple);
}
}
return result;
}
那么问题又来了,制作出来了,怎么使用呢?我们就试试用上面的模板来筛选苹果重量和苹果颜色,这里会用到Lambda表达式:
public static void main(String[] args) {
//用一个集合装三个苹果
List<Apple> apples = Lists.newArrayList();
Apple apple1 = new Apple();
apple1.setColor("red").setWeight(20);
Apple apple2 = new Apple();
apple2.setColor("green").setWeight(20);
Apple apple3 = new Apple();
apple3.setColor("black").setWeight(5);
apples.add(apple1);
apples.add(apple2);
apples.add(apple3); //筛选出重量>10的苹果,下面两种写法一样
// List<Apple> list1 = filterApples(apples,(Apple a)->a.getWeight()>10);
List<Apple> list1 = filterApples(apples,a->a.getWeight()>10);
System.out.println("重量大于10的苹果:"+list1); //筛选出颜色是绿色的苹果,下面两种写法一样
// List<Apple> list2 = filterApples(apples,(Apple a)->Objects.equal("green", a.getColor()));
List<Apple> list2 = filterApples(apples,a->Objects.equal("green", a.getColor()));
System.out.println("绿色苹果:"+list2); }
到这里就行了么?还有更加有意思的就是流,流可以说是特地为了处理集合而创造的,其实上面的代码还是太麻烦了,还要自己定义一个方法,有没有更快更快的方法,下面是使用流的方式:
//使用流筛选重量大于10的苹果
List<Apple> weightApples = apples.stream().filter(a->a.getWeight()>10).collect(Collectors.toList());
System.out.println(weightApples); //注意,每次获取流是一次性的,如果前面已经获取了流的返回值了,还想继续操作流,只能重新获取流;使用流筛选颜色是绿色的苹果
List<Apple> greenApples = apples.stream().filter(a->Objects.equal("green", a.getColor())).collect(Collectors.toList());
System.out.println(greenApples);
看到没有,使用了流之后只需要几行代码,不用再定义什么方法了,只需要将集合变成一个流,然后链式调用filter方法,传进去一个行为(这里传的是一个Lambda表达式),然后再调用collect方法收集流中的元素;
其实到这里还有没有可以优化的地方,当然有,就比如说每次都要System.out.println(xxx)方法真的很讨厌,能不能干掉,但是控制台却还是能打印出来,当然可以,这里就涉及到了一个概念叫做方法引用,什么叫做方法引用呢?即是一个函数指针,你可以这样想,当一个类被加载到jvm中了,那么我们只要知道这个类在哪个内存地址,就可以知道它的方法的内存地址了(可以想想调用静态方法的时候,直接用类名加方法名调用的),于是我们可以用这样的方式 类名::方法名 的方式调用某个方法,那么把System.out.println(xxx)修改一下就是System.out::printIn,代码修改如下,我们把两个集合分别遍历:
//使用流筛选重量大于10的苹果
apples.stream().filter(a->a.getWeight()>10).collect(Collectors.toList()).forEach(System.out::println); //使用流筛选颜色是绿色的苹果
apples.stream().filter(a->Objects.equal("green", a.getColor())).collect(Collectors.toList()).forEach(System.out::println);
上面的方法引用还只是最简单的方式,其他的后面有时间会慢慢说的;
通过迅速的看了看jdk8,应该对这种集合形式的处理了解一些了,然后我们再分块讨论,比如Lambda表达式怎么写,函数式接口有哪些,流的操作有哪些,方法引用怎么使用等等!这些大概就涵盖了jdk8的80%了,后面还有jdk8新的日期和时间api,Optional代替null,默认方法等等就简单了;
迅速看一下jdk8的更多相关文章
- 看不懂JDK8的流操作?5分钟带你入门(转)
在JDK1.8里有两个非常高级的新操作,它们分别是:Lambda 表达式和 Stream 流. Lambda表达式 让我们先说说 Lambda 表达式吧,这个表达式最大的作用就是简化语法,让代码更加易 ...
- 谈谈HashMap与HashTable
谈谈HashMap与HashTable HashMap 我们一直知道HashMap是非线程安全的,HashTable是线程安全的,可这是为什么呢?先聊聊HashMap吧,想要了解它为什么是非线程安全的 ...
- Java中常见数据结构Map之HashMap
之前很早就在博客中写过HashMap的一些东西: 彻底搞懂HashMap,HashTableConcurrentHashMap关联: http://www.cnblogs.com/wang-meng/ ...
- HashMap中的hash算法总结
前言 算法一直是我的弱项,然而面试中基本是必考的项目,刚好上次看到一个HashMap的面试题,今天也来学习下 HashMap中的hash算法是如何实现的. 数学知识回顾 << : 左移运算 ...
- Integer代码分析
我们都知道Integer是int的封装类,提供了一些类型转换等工具方法,有一个-128-127的缓存,而且是final的. ----------------------------- 干货: Inte ...
- Node.js 网页爬虫再进阶,cheerio助力
任务还是读取博文标题. 读取app2.js // 内置http模块,提供了http服务器和客户端功能 var http=require("http"); // cheerio模块, ...
- JDK1.7中HashMap死环问题及JDK1.8中对HashMap的优化源码详解
一.JDK1.7中HashMap扩容死锁问题 我们首先来看一下JDK1.7中put方法的源码 我们打开addEntry方法如下,它会判断数组当前容量是否已经超过的阈值,例如假设当前的数组容量是16,加 ...
- 面试题-关于Java线程池一篇文章就够了
在Java面试中,线程池相关知识,虽不能说是必问提,但出现的频次也是非常高的.同时又鉴于公众号"程序新视界"的读者后台留言让写一篇关于Java线程池的文章,于是就有本篇内容,本篇将 ...
- 使用Intellij idea新建Java Web项目(servlet) 原理及初步使用
准备 JDK (配置JDK_HOME\bin 和 CLASSPATH) 注:JDK8下载已经需要注册了,请使用JDK11(现在是官方长期支持的版本) 对于我们新手来说,JD ...
随机推荐
- python:复制文件及文件夹
#!/usr/bin/python# -*- coding:utf-8 -*- import shutil #shutil.copy(文件1,文件2)#将源内容复制到目标文件中.d.txt不存在则创建 ...
- CSS的快速入门
CSS的快速入门 1.CSS要学习的内容主要包括 1. CSS概念和快速入门 2.CSS选择器(重点+难点) 3.美化网页(文字.阴影.超链接.列表.渐变,等) 4.盒子模型 5.浮动 6.定位 2. ...
- Ubuntu18.04 一条命令安装caffe问题
由于caffe安装坑很多,而且caffe框架很久不更新了,微调对框架影响不大,所以对与ubuntu18.04在caffe官网提供了一条命令安装,避免很多踩坑痛苦. CPU的一条命令安装: sudo a ...
- js - 子节点
子节点数量:this.wdlgLossInfo.childNodes.length
- python3练习100题——020
原题链接:http://www.runoob.com/python/python-exercise-example20.html 题目:一球从100米高度自由落下,每次落地后反跳回原高度的一半:再落下 ...
- eclipse查看jar包源代码乱码问题解决
文章来源 今天在eclipse中查看java的jar包中的源代码时,显示的全部是乱码.起初只设置了content types还不行,还是乱码.不过问题最后解决了,配置步骤如下: 首先要知道你需要配置的 ...
- java下载和环境变量配置
初学java,以前没有接触过这方面内容,所以简要记录一下我2个月的学习流程. 首先,我在慕课上学习java的基础,浙江大学翁恺老师的课程. 下载ECLIPSE-java 进入官网:https://ww ...
- mysql(1):简介
typora-root-url: ./ SQL语法顺序和执行顺序 SQL语法顺序 SELECT [DISTINCT] <select_list> FROM <left_table&g ...
- SSH通道 Xshell
SSH是每一台Linux电脑的标准配置. SSH是一种网络协议,用于计算机之间的加密登录(安全的). 1. 登录远程主机 $ ssh user@host 2. SSH的默认端口是22, ...
- Hadoop TextInputFormat
1. TextInputFortmat TextInputFormat是默认的InputFormat.每条记录是一行输入.Key是LongWritable类型,存储该行在整个文件中的字节偏移量(不是行 ...