/* * Copyright 2023-2024 the original author or authors. * * 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.collection; import java.util.Collection; import java.util.Collections; import java.util.Map; import java.util.Map.Entry; import java.util.Optional; import java.util.Set; import java.util.concurrent.ConcurrentHashMap; import java.util.function.Consumer; import java.util.function.Function; import javax.annotation.Nullable; import com.google.common.annotations.Beta; import xyz.zhouxy.plusone.commons.util.ConcurrentHashMapTools; @Beta public abstract class AbstractMapWrapper> { private final Map map; private final Consumer keyChecker; private final Consumer valueChecker; protected AbstractMapWrapper(Map map, @Nullable Consumer keyChecker, @Nullable Consumer valueChecker) { this.map = map; this.keyChecker = keyChecker; this.valueChecker = valueChecker; } public final T put(K key, V value) { if (this.keyChecker != null) { this.keyChecker.accept(key); } if (this.valueChecker != null) { this.valueChecker.accept(value); } this.map.put(key, value); return getSelf(); } public final T putAll(Map m) { m.forEach(this::put); return getSelf(); } /** * 获取 {@code map} 中的值。如果 {@code key} 不存在,则抛出异常。 * 将 {@code value}(可为 {@code null})装进 {@link Optional} 中后返回。 * 为了这碟醋包的这盘饺子。 * * @param key 键 * @return 可缺失的值 * @throws IllegalArgumentException key 不存在时抛出。 */ public Optional get(K key) { if (!this.map.containsKey(key)) { throw new IllegalArgumentException("Key does not exist"); } return Optional.ofNullable(this.map.get(key)); } @SuppressWarnings("unchecked") public final Optional getAndConvert(K key) { return get(key).map(v -> (R) v); } public final Optional getAndConvert(K key, Function mapper) { return get(key).map(mapper); } public final boolean containsKey(Object key) { return this.map.containsKey(key); } public final int size() { return this.map.size(); } public final Set keySet() { return this.map.keySet(); } public final Collection values() { return this.map.values(); } public final Set> entrySet() { return this.map.entrySet(); } public final void clear() { this.map.clear(); } public final boolean containsValue(Object value) { return this.map.containsValue(value); } public final boolean isEmpty() { return this.map.isEmpty(); } public final V remove(Object key) { return this.map.remove(key); } public final V putIfAbsent(K key, V value) { if (this.keyChecker != null) { this.keyChecker.accept(key); } if (this.valueChecker != null) { this.valueChecker.accept(value); } return this.map.putIfAbsent(key, value); } public final V computeIfAbsent(K key, Function mappingFunction) { if (this.keyChecker != null) { this.keyChecker.accept(key); } Function func = (K k) -> { V value = mappingFunction.apply(k); if (this.valueChecker != null) { this.valueChecker.accept(value); } return value; }; if (this.map instanceof ConcurrentHashMap) { return ConcurrentHashMapTools.computeIfAbsent( (ConcurrentHashMap) this.map, key, func); } else { return this.map.computeIfAbsent(key, func); } } public final Map exportMap() { return this.map; } public final Map exportUnmodifiableMap() { return Collections.unmodifiableMap(this.map); } protected abstract T getSelf(); @Override public String toString() { return this.map.toString(); } public abstract static class Builder> { protected final Map map; protected Consumer keyChecker; protected Consumer valueChecker; protected Builder(Map map) { this.map = map; } public Builder keyChecker(@Nullable Consumer keyChecker) { this.keyChecker = keyChecker; return this; } public Builder valueChecker(@Nullable Consumer valueChecker) { this.valueChecker = valueChecker; return this; } public Builder put(K key, V value) { if (this.keyChecker != null) { this.keyChecker.accept(key); } if (this.valueChecker != null) { this.valueChecker.accept(value); } this.map.put(key, value); return this; } public Builder putAll(Map m) { m.forEach(this::put); return this; } public abstract T build(); public abstract T buildUnmodifiableMap(); } }