Java8新增的这些集合骚操作,你掌握了嘛?
Java8时Lambda表达式的出现,将行为作为参数传递进函数的函数式编程,大大简化了之前冗杂的写法。
如果你对Lambda还不了解,可以参考我之前的关于Lambda表达式的总结:Java8的Lambda表达式,你会不?
对于集合一类,我们来整理一下发生的变化叭。
Iterable的forEach
Iterable接口就是所有可迭代类型的父接口,我们熟知的Collection接口就是继承自它。Java8接口默认方法以及Lambda表达式的出现,让我们在遍历元素时对元素进行操作变得格外简单。
下面的forEach方法就是Java8新增的,它接受一个Consumer对象,是一个消费者类型的函数式接口。
default void forEach(Consumer<? super T> action) {
Objects.requireNonNull(action);
for (T t : this) {
action.accept(t);
}
}
下面这段代码遍历输出每个元素。
public void testForEach(){
List<Integer> list = new ArrayList<>(Arrays.asList(1, 2, 3, 4, 5));
list.forEach(System.out::println);
}
Iterator的forEachRemaining
java.util.Iterator
是用于遍历集合的迭代器,接口定义如下:
public interface Iterator<E> {
boolean hasNext();
E next();
default void remove() {
throw new UnsupportedOperationException("remove");
}
default void forEachRemaining(Consumer<? super E> action) {
Objects.requireNonNull(action);
while (hasNext())
action.accept(next());
}
}
default方法是Java8接口中新增的,forEachRemaining
方法接收一个Consumer,我们可以通过该方法简化我们的遍历操作:
/**
* Java8 为Iterator新增了 forEachRemaining(Consumer action) 方法
*/
public static void main(String[] args) {
List<Integer> list = new ArrayList<>(Arrays.asList(1, 2, 3, 4, 5));
Iterator<Integer> iterator = list.iterator();
iterator.forEachRemaining(System.out::println);
}
Collection的removeIf
Java8为Collection增加了默认的removeIf方法,接收一个Predicate判断,test方法结果为true,就移除对应的元素。
default boolean removeIf(Predicate<? super E> filter) {
Objects.requireNonNull(filter);
boolean removed = false;
final Iterator<E> each = iterator();
while (each.hasNext()) {
// 判断元素是否需要被移除
if (filter.test(each.next())) {
each.remove();
removed = true;
}
}
return removed;
}
下面这段代码移除列表中的偶数元素。
public void testRemoveIf() {
List<Integer> list = new ArrayList<>(Arrays.asList(1, 2, 3, 4, 5));
list.removeIf(x -> x % 2 == 0);
list.forEach(System.out::println);
}
Stream操作
具体使用可以参照Java8的StreamAPI常用方法总结
public void testStream(){
IntStream stream = IntStream.builder().add(1).add(2).add(3).build();
int max = stream.max().getAsInt();
System.out.println(max);
}
List的replaceAll
Java8为List接口增加了默认的replaceAll方法,需要UnaryOperator来替换所有集合元素,UnaryOperator是一个函数式接口。
default void replaceAll(UnaryOperator<E> operator) {
Objects.requireNonNull(operator);
final ListIterator<E> li = this.listIterator();
while (li.hasNext()) {
li.set(operator.apply(li.next()));
}
}
下面这个示例为每个元素加上3。
public void testReplaceAll(){
List<Integer> list = new ArrayList<>(Arrays.asList(1, 2, 3, 4, 5));
list.replaceAll(x -> x + 3);
list.forEach(System.out::println);
}
List的sort
Java8为List接口增加了默认的sort方法,需要Comparator对象来控制元素排,我们可以传入Lambda表达式。
default void sort(Comparator<? super E> c) {
Object[] a = this.toArray();
Arrays.sort(a, (Comparator) c);
ListIterator<E> i = this.listIterator();
for (Object e : a) {
i.next();
i.set((E) e);
}
}
下面这个例子将list逆序。
public void testSort() {
List<Integer> list = new ArrayList<>(Arrays.asList(1, 2, 3, 4, 5));
list.sort((x, y) -> y - x);
System.out.println(list);
}
Map的ForEach
Map接口在Java8同样也新增了用于遍历的方法:
default void forEach(BiConsumer<? super K, ? super V> action) {
Objects.requireNonNull(action);
for (Map.Entry<K, V> entry : entrySet()) {
K k;
V v;
try {
k = entry.getKey();
v = entry.getValue();
} catch(IllegalStateException ise) {
// this usually means the entry is no longer in the map.
throw new ConcurrentModificationException(ise);
}
action.accept(k, v);
}
}
该方法接收一个二元的参数,分别对应key和value:
private void print(Map<Integer,String> map){
map.forEach((x , y )-> {
System.out.println("x -> " + x + ", y -> " + y);
});
}
为了接下来测试方便,数据先准备一下:
Map<Integer, String> map = new HashMap<>();
{
map.put(1, "hello");
map.put(2, "summer");
map.put(3, "day");
map.put(4, "tqbx");
}
Map的remove
Java8新增了一个remove(Object key, Object value)
方法。
default boolean remove(Object key, Object value) {
// key存在且key对应的value确实是传入的value才移除
Object curValue = get(key);
if (!Objects.equals(curValue, value) ||
(curValue == null && !containsKey(key))) {
return false;
}
remove(key);
return true;
}
移除key为3且value为day的元素。
@Test
public void testRemove(){
map.remove(3,"day");
print(map);
}
Map的compute相关方法
Object compute(Object key, BiFunction remappingFunction)
:该方法使用remappingFunction根据key-value对计算一个新value,情况如下:
- 只要新value不为null,就使用新value覆盖原value。
- 如果原value不为null,但新value为null,则删除原key-value对。
- 如果原value、新value同时为null那么该方法不改变任何key-value对,直接返回null。
Object computeIfAbsent(Object key, Function mappingFunction)
:
- 如果传给该方法的key参数在Map中对应的value为null,则使用mappingFunction根据key计算个新的结果。
- 如果计算结果不为null,则用计算结果覆盖原有的value。
- 如果原Map原来不包括该key,那么该方法可能会添加一组key-value对。
Object computeIfPresent(Object key, BiFunction remappingFunction)
:
- 如果传给该方法的key参数Map中对应的value不为null,该方法将使用remappingFunction根据原key、value计算一个新的结果。
- 如果计算结果不为null,则使用该结果覆盖原来的value。
- 如果计算结果为null,则删除原key-value对。
@Test
public void testCompute() {
//key==2 对应的value存在时,使用计算的结果作为新value
map.computeIfPresent(2, (k, v) -> v.toUpperCase());
print(map);
//key==6 对应的value为null (或不存在)时,使用计算的结果作为新value
map.computeIfAbsent(6, (k) -> k + "haha");
print(map);
}
Map的getOrDefault
default V getOrDefault(Object key, V defaultValue) {
V v;
// key存在或 value存在,则返回对应的value,否则返回defaultValue
return (((v = get(key)) != null) || containsKey(key))
? v
: defaultValue;
}
@Test
public void testGetOrDefault(){
// 获取指定key的value,如果该key不存在,则返回default
String value = map.getOrDefault(5, "如果没有key==5的value,就返回这条信息");
System.out.println(value);
}
Map的merge
default V merge(K key, V value,
BiFunction<? super V, ? super V, ? extends V> remappingFunction) {
Objects.requireNonNull(remappingFunction);
Objects.requireNonNull(value);
// 先获取原值
V oldValue = get(key);
// 原值为null,新值则为传入的value,不为null,则使用function计算,得到新值
V newValue = (oldValue == null) ? value :
remappingFunction.apply(oldValue, value);
// 新值如果为null,则移除原key-value对
if(newValue == null) {
remove(key);
} else {
// 不为null,则替换之
put(key, newValue);
}
// 返回新值
return newValue;
}
@Test
public void testMerge(){
// key为2 的值 加上 +add
map.merge(2," + add",(oldval, newVal) -> oldval + newVal);
print(map);
}
Map的putIfAbsent
default V putIfAbsent(K key, V value) {
// 得到原值
V v = get(key);
// 原值为null,则替换新值
if (v == null) {
v = put(key, value);
}
// 返回原值
return v;
}
@Test
public void testPutIfAbsent(){
// key = 10 对应的value不存在, 则用101010 覆盖
String s1 = map.putIfAbsent(10, "101010");
System.out.println(s1);
print(map);
System.out.println("============================");
// key = 2 对应的value存在且为summer,返回summer
String s2 = map.putIfAbsent(2, "2222");
System.out.println(s2);
print(map);
}
输出结果
null
x -> 1, y -> hello
x -> 2, y -> summer
x -> 3, y -> day
x -> 4, y -> tqbx
x -> 10, y -> 101010
============================
summer
x -> 1, y -> hello
x -> 2, y -> summer
x -> 3, y -> day
x -> 4, y -> tqbx
x -> 10, y -> 101010
Map的replace相关方法
@Test
public void testReplace(){
//boolean 将指定的 1 -> hello 键值对的value替换为hi
map.replace(1,"hello","hi");
print(map);
System.out.println("============================");
//V 指定key对应的value替换成新value
map.replace(2,"天乔巴夏");
print(map);
System.out.println("============================");
//void 对所有的key和value 进行计算,填入value中
map.replaceAll((k,v)-> k + "-" + v.toUpperCase());
print(map);
}
输出结果:
x -> 1, y -> hi
x -> 2, y -> summer
x -> 3, y -> day
x -> 4, y -> tqbx
============================
x -> 1, y -> hi
x -> 2, y -> 天乔巴夏
x -> 3, y -> day
x -> 4, y -> tqbx
============================
x -> 1, y -> 1-HI
x -> 2, y -> 2-天乔巴夏
x -> 3, y -> 3-DAY
x -> 4, y -> 4-TQBX
Java8新增的这些集合骚操作,你掌握了嘛?的更多相关文章
- <JAVA8新增内容>关于集合的操作(Collection/Iterator/Stream)
因为下文频繁使用lambda表达式,关于Java中的lambda表达式内容请见: http://www.cnblogs.com/guguli/p/4394676.html 一.使用增强的Iterato ...
- java集合(3)-Java8新增的Stream操作集合
Java8新增了Stream,IntStream,LongStream,DoubleStream等流式API,这些API代表多个支持串行和并行聚集操作的元素.上面的4个接口中,Stream是一个通用的 ...
- java集合(3)-Java8新增的Predicate操作集合
Java8起为Collection集合新增了一个removeIf(Predicate filter)方法,该方法将批量删除符合filter条件的所有元素.该方法需要一个Predicate(谓词)对象作 ...
- Java8 新特性之集合操作Stream
Java8 新特性之集合操作Stream Stream简介 Java 8引入了全新的Stream API.这里的Stream和I/O流不同,它更像具有Iterable的集合类,但行为和集合类又有所不同 ...
- Java8 如何进行stream reduce,collection操作
Java8 如何进行stream reduce,collection操作 2014-07-16 16:42 佚名 oschina 字号:T | T 在java8 JDK包含许多聚合操作(如平均值,总和 ...
- JAVA中的集合容器操作类
目录 JAVA中的集合容器操作类 List集合 ArrayList的操作方法说明 LinkedList Stack Set Map Queue 总结 JAVA中的集合容器操作类 Java容器类库总共分 ...
- Java8增强的Map集合
Map集合简介 Map用于保存具有映射关系的数据,因此Map集合里保存着两组值,一组值用于保存Map里的key,另外一组用于保存Map里的vlaue,key和value都可以是任何引用类型的数据. M ...
- 聊聊redis实际运用及骚操作
前言 聊起 redis 咱们大部分后端猿应该都不陌生,或多或少都用过.甚至大部分前端猿都知道. 数据结构: string. hash. list. set (无序集合). setsorted(有序集合 ...
- 我天!xx.equals(null) 是什么骚操作??
问题背景 我的天,最近做 Code Review 看到一个同事的骚操作,他写了一个工具类,大概是这样的: public static boolean isNull(Object object){ re ...
随机推荐
- RedHat Linux-配置YUM仓库
范例:配置Yum仓库 Yum软件仓库的作用是为了进一步简化RPM管理软件的难度以及自动分析所需软件包及其依赖关系的技术.可以把Yum想象成是一个硕大的软件仓库,里面保存有几乎所有常用的工具,而且只需要 ...
- Elementary OS安装及开发环境配置(一)
前言 假期在家无聊,刚好把六年前的一台笔记本电脑利用起来,原来电脑虽然说配置说不上古董机器,但是运行win系统感觉还是不流畅,所幸给换成Linux桌面版系统,在网上查阅了很多,Linux桌面系统要么推 ...
- Spring @Autowired 注解自动注入流程是怎么样?
面试中碰到面试官问:"Spring 注解是如果工作的?",当前我一惊,完了这不触及到我的知识误区了吗?,还好我机智,灵机一动回了句:Spring 注解的工作流程倒还没有看到,但是我 ...
- docker 国内源切换加速
阿里云比较好: 地址: https://cr.console.aliyun.com/cn-hangzhou/instances/mirrors
- 死磕以太坊源码分析之p2p节点发现
死磕以太坊源码分析之p2p节点发现 在阅读节点发现源码之前必须要理解kadmilia算法,可以参考:KAD算法详解. 节点发现概述 节点发现,使本地节点得知其他节点的信息,进而加入到p2p网络中. 以 ...
- 又陷入知识盲区了,面试被问SpringBoot集成dubbo,我当时就懵了
前言 前两天在和粉丝聊天的时候,粉丝跟我说之前在面试的时候被问到SpringBoot这一块的知识被问的有点懵,和我问了不少这方面的东西.事后我想了想不如把这些东西分享出来吧,让更多的人看到,这样不管是 ...
- 【移动自动化】【四】获取Toast
什么是Toast Android中的Toast是一种简易的消息提示框. 如何识别Toast 使用 xpath 查找 推荐 //*[@class='android.widget.Toast'] (固定这 ...
- Unity减少构建安装包的体积(210MB减小到7MB)
概述 项目简介 由于是公司内做的项目,不方便开源,就只分享优化过程吧. 项目信息 逐日是一个移动端单机小游戏,使用Unity开发,目前已将项目使用的Unity升级到2019.4.14f1c1 (3e5 ...
- 精尽MyBatis源码分析 - SQL执行过程(四)之延迟加载
该系列文档是本人在学习 Mybatis 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释(Mybatis源码分析 GitHub 地址.Mybatis-Spring 源码分析 GitHub ...
- php8.0正式版新特性和性能优化学习
前言 PHP团队宣布PHP8正式GA(链接).php的发展又开启了新的篇章,PHP8.0.0版本引入了一些重大变更及许多新特性和性能优化机制.火速学习下~ JIT(Just in Time Compi ...