2023-08-09 20:23:31 +08:00
|
|
|
|
/*
|
2024-10-12 00:57:35 +08:00
|
|
|
|
* Copyright 2023-2024 the original author or authors.
|
2023-08-09 20:23:31 +08:00
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
*
|
|
|
|
|
* https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
package xyz.zhouxy.plusone.commons.util;
|
|
|
|
|
|
|
|
|
|
import java.util.Objects;
|
|
|
|
|
import java.util.concurrent.ConcurrentHashMap;
|
|
|
|
|
import java.util.function.Function;
|
|
|
|
|
|
|
|
|
|
import xyz.zhouxy.plusone.commons.base.JRE;
|
2023-10-30 09:09:32 +08:00
|
|
|
|
import xyz.zhouxy.plusone.commons.collection.SafeConcurrentHashMap;
|
2023-08-09 20:23:31 +08:00
|
|
|
|
|
2023-10-30 09:09:32 +08:00
|
|
|
|
/**
|
2024-04-03 16:20:41 +08:00
|
|
|
|
* ConcurrentHashMapTools
|
2023-10-30 09:09:32 +08:00
|
|
|
|
*
|
|
|
|
|
* <p>
|
|
|
|
|
* Java 8 的 {@link ConcurrentHashMap#computeIfAbsent(Object, Function)} 方法有 bug,
|
|
|
|
|
* 可使用这个工具类的 {@link computeIfAbsentForJava8} 进行替换。
|
|
|
|
|
*
|
|
|
|
|
* <p>
|
|
|
|
|
* <b>NOTE: 方法来自Dubbo,见:issues#2349</b>
|
2024-10-21 23:17:52 +08:00
|
|
|
|
*
|
2023-10-30 09:09:32 +08:00
|
|
|
|
* @author <a href="http://zhouxy.xyz:3000/ZhouXY108">ZhouXY</a>
|
|
|
|
|
* @since 1.0
|
|
|
|
|
* @see ConcurrentHashMap
|
|
|
|
|
* @see SafeConcurrentHashMap
|
|
|
|
|
*/
|
2024-04-03 16:20:41 +08:00
|
|
|
|
public class ConcurrentHashMapTools {
|
2023-08-09 20:23:31 +08:00
|
|
|
|
|
2024-04-03 16:20:41 +08:00
|
|
|
|
public static <K, V> V computeIfAbsent(
|
|
|
|
|
ConcurrentHashMap<K, V> map, final K key, // NOSONAR
|
2023-09-09 13:54:14 +08:00
|
|
|
|
final Function<? super K, ? extends V> mappingFunction) {
|
2023-10-30 09:09:32 +08:00
|
|
|
|
Objects.requireNonNull(map, "map");
|
2023-09-09 13:54:14 +08:00
|
|
|
|
return JRE.isJava8()
|
|
|
|
|
? computeIfAbsentForJava8(map, key, mappingFunction)
|
|
|
|
|
: map.computeIfAbsent(key, mappingFunction);
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-03 16:20:41 +08:00
|
|
|
|
public static <K, V> V computeIfAbsentForJava8(
|
|
|
|
|
ConcurrentHashMap<K, V> map, final K key, // NOSONAR
|
2023-09-09 13:54:14 +08:00
|
|
|
|
final Function<? super K, ? extends V> mappingFunction) {
|
2023-10-30 09:09:32 +08:00
|
|
|
|
Objects.requireNonNull(key);
|
2023-09-09 13:54:14 +08:00
|
|
|
|
Objects.requireNonNull(mappingFunction);
|
|
|
|
|
V v = map.get(key);
|
|
|
|
|
if (null == v) {
|
|
|
|
|
v = mappingFunction.apply(key);
|
2023-08-09 20:23:31 +08:00
|
|
|
|
if (null == v) {
|
2023-09-09 13:54:14 +08:00
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
final V res = map.putIfAbsent(key, v);
|
|
|
|
|
if (null != res) {
|
|
|
|
|
return res;
|
2023-08-09 20:23:31 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-09 13:54:14 +08:00
|
|
|
|
return v;
|
2023-08-09 20:23:31 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-04-03 16:20:41 +08:00
|
|
|
|
private ConcurrentHashMapTools() {
|
2023-08-09 20:23:31 +08:00
|
|
|
|
throw new IllegalStateException("Utility class");
|
|
|
|
|
}
|
|
|
|
|
}
|