All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.vavr.collection.Maps Maven / Gradle / Ivy

There is a newer version: 1.0.0-alpha-4
Show newest version
/*  __    __  __  __    __  ___
 * \  \  /  /    \  \  /  /  __/
 *  \  \/  /  /\  \  \/  /  /
 *   \____/__/  \__\____/__/
 *
 * Copyright 2014-2019 Vavr, http://vavr.io
 *
 * 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
 *
 *     http://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 io.vavr.collection;

import io.vavr.Tuple;
import io.vavr.Tuple2;
import io.vavr.control.Option;

import java.util.Comparator;
import java.util.Objects;
import java.util.function.*;

import static io.vavr.API.Tuple;

/**
 * INTERNAL: Common {@code Map} functions (not intended to be public).
 *
 * @author Ruslan Sennov, Daniel Dietrich
 */
final class Maps {

    private Maps() {
    }

    @SuppressWarnings("unchecked")
    static > Tuple2 computeIfAbsent(M map, K key, Function mappingFunction) {
        Objects.requireNonNull(mappingFunction, "mappingFunction is null");
        final Option value = map.get(key);
        if (value.isDefined()) {
            return Tuple.of(value.get(), map);
        } else {
            final V newValue = mappingFunction.apply(key);
            final M newMap = (M) map.put(key, newValue);
            return Tuple.of(newValue, newMap);
        }
    }

    @SuppressWarnings("unchecked")
    static > Tuple2, M> computeIfPresent(M map, K key, BiFunction remappingFunction) {
        final Option value = map.get(key);
        if (value.isDefined()) {
            final V newValue = remappingFunction.apply(key, value.get());
            final M newMap = (M) map.put(key, newValue);
            return Tuple.of(Option.of(newValue), newMap);
        } else {
            return Tuple.of(Option.none(), map);
        }
    }

    static > M distinct(M map) {
        return map;
    }

    static > M distinctBy(M map, OfEntries ofEntries,
            Comparator> comparator) {
        Objects.requireNonNull(comparator, "comparator is null");
        return ofEntries.apply(map.iterator().distinctBy(comparator));
    }

    static > M distinctBy(
            M map, OfEntries ofEntries, Function, ? extends U> keyExtractor) {
        Objects.requireNonNull(keyExtractor, "keyExtractor is null");
        return ofEntries.apply(map.iterator().distinctBy(keyExtractor));
    }

    static > M drop(M map, OfEntries ofEntries, Supplier emptySupplier, int n) {
        if (n <= 0) {
            return map;
        } else if (n >= map.size()) {
            return emptySupplier.get();
        } else {
            return ofEntries.apply(map.iterator().drop(n));
        }
    }

    static > M dropRight(M map, OfEntries ofEntries, Supplier emptySupplier,
            int n) {
        if (n <= 0) {
            return map;
        } else if (n >= map.size()) {
            return emptySupplier.get();
        } else {
            return ofEntries.apply(map.iterator().dropRight(n));
        }
    }

    static > M dropUntil(M map, OfEntries ofEntries,
            Predicate> predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return dropWhile(map, ofEntries, predicate.negate());
    }

    static > M dropWhile(M map, OfEntries ofEntries,
            Predicate> predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return ofEntries.apply(map.iterator().dropWhile(predicate));
    }

    static > M filter(M map, OfEntries ofEntries,
            BiPredicate predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return filter(map, ofEntries, t -> predicate.test(t._1, t._2));
    }

    static > M filter(M map, OfEntries ofEntries,
            Predicate> predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return ofEntries.apply(map.iterator().filter(predicate));
    }

    static > M filterKeys(M map, OfEntries ofEntries,
            Predicate predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return filter(map, ofEntries, t -> predicate.test(t._1));
    }

    static > M filterValues(M map, OfEntries ofEntries,
            Predicate predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return filter(map, ofEntries, t -> predicate.test(t._2));
    }

    static > Map groupBy(M map, OfEntries ofEntries,
            Function, ? extends C> classifier) {
        return Collections.groupBy(map, classifier, ofEntries);
    }

    static > Iterator grouped(M map, OfEntries ofEntries, int size) {
        return sliding(map, ofEntries, size, size);
    }

    @SuppressWarnings("unchecked")
    static > Option initOption(M map) {
        return map.isEmpty() ? Option.none() : Option.some((M) map.init());
    }

    static > M merge(M map, OfEntries ofEntries,
            Map that) {
        Objects.requireNonNull(that, "that is null");
        if (map.isEmpty()) {
            return ofEntries.apply(Map.narrow(that));
        } else if (that.isEmpty()) {
            return map;
        } else {
            return that.foldLeft(map, (result, entry) -> !result.containsKey(entry._1) ? put(result, entry) : result);
        }
    }

    @SuppressWarnings("unchecked")
    static > M merge(
            M map, OfEntries ofEntries,
            Map that, BiFunction collisionResolution) {
        Objects.requireNonNull(that, "that is null");
        Objects.requireNonNull(collisionResolution, "collisionResolution is null");
        if (map.isEmpty()) {
            return ofEntries.apply(Map.narrow(that));
        } else if (that.isEmpty()) {
            return map;
        } else {
            return that.foldLeft(map, (result, entry) -> {
                final K key = entry._1;
                final U value = entry._2;
                final V newValue = result.get(key).map(v -> (V) collisionResolution.apply(v, value)).getOrElse(value);
                return (M) result.put(key, newValue);
            });
        }
    }

    @SuppressWarnings("unchecked")
    static > M ofStream(M map, java.util.stream.Stream stream,
            Function keyMapper,
            Function valueMapper) {
        Objects.requireNonNull(stream, "stream is null");
        Objects.requireNonNull(keyMapper, "keyMapper is null");
        Objects.requireNonNull(valueMapper, "valueMapper is null");
        return Stream.ofAll(stream).foldLeft(map, (m, el) -> (M) m.put(keyMapper.apply(el), valueMapper.apply(el)));
    }

    @SuppressWarnings("unchecked")
    static > M ofStream(M map, java.util.stream.Stream stream,
            Function> entryMapper) {
        Objects.requireNonNull(stream, "stream is null");
        Objects.requireNonNull(entryMapper, "entryMapper is null");
        return Stream.ofAll(stream).foldLeft(map, (m, el) -> (M) m.put(entryMapper.apply(el)));
    }

    static > Tuple2 partition(M map, OfEntries ofEntries,
            Predicate> predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        final Tuple2>, Iterator>> p = map.iterator().partition(predicate);
        return Tuple.of(ofEntries.apply(p._1), ofEntries.apply(p._2));
    }

    static > M peek(M map, Consumer> action) {
        Objects.requireNonNull(action, "action is null");
        if (!map.isEmpty()) {
            action.accept(map.head());
        }
        return map;
    }

    @SuppressWarnings("unchecked")
    static > M put(M map, K key, U value,
            BiFunction merge) {
        Objects.requireNonNull(merge, "the merge function is null");
        final Option currentValue = map.get(key);
        if (currentValue.isEmpty()) {
            return (M) map.put(key, value);
        } else {
            return (M) map.put(key, merge.apply(currentValue.get(), value));
        }
    }

    @SuppressWarnings("unchecked")
    static > M put(M map, Tuple2 entry) {
        Objects.requireNonNull(entry, "entry is null");
        return (M) map.put(entry._1, entry._2);
    }

    static > M put(M map, Tuple2 entry,
            BiFunction merge) {
        Objects.requireNonNull(merge, "the merge function is null");
        final Option currentValue = map.get(entry._1);
        if (currentValue.isEmpty()) {
            return put(map, entry);
        } else {
            return put(map, entry.map2(value -> merge.apply(currentValue.get(), value)));
        }
    }

    static > M reject(M map, OfEntries ofEntries,
            Predicate> predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return filter(map, ofEntries, predicate.negate());
    }

    static > M reject(M map, OfEntries ofEntries,
            BiPredicate predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return filter(map, ofEntries, predicate.negate());
    }

    static > M rejectKeys(M map, OfEntries ofEntries,
            Predicate predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return filterKeys(map, ofEntries, predicate.negate());
    }

    static > M rejectValues(M map, OfEntries ofEntries,
            Predicate predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return filterValues(map, ofEntries, predicate.negate());
    }

    @SuppressWarnings("unchecked")
    static > M replace(M map, K key, V oldValue, V newValue) {
        return map.contains(Tuple(key, oldValue)) ? (M) map.put(key, newValue) : map;
    }

    @SuppressWarnings("unchecked")
    static > M replace(M map, Tuple2 currentElement, Tuple2 newElement) {
        Objects.requireNonNull(currentElement, "currentElement is null");
        Objects.requireNonNull(newElement, "newElement is null");
        return (M) (map.containsKey(currentElement._1) ? map.remove(currentElement._1).put(newElement) : map);
    }

    @SuppressWarnings("unchecked")
    static > M replaceAll(M map, BiFunction function) {
        return (M) map.map((k, v) -> Tuple(k, function.apply(k, v)));
    }

    static > M replaceAll(M map, Tuple2 currentElement, Tuple2 newElement) {
        return replace(map, currentElement, newElement);
    }

    @SuppressWarnings("unchecked")
    static > M replaceValue(M map, K key, V value) {
        return map.containsKey(key) ? (M) map.put(key, value) : map;
    }

    @SuppressWarnings("unchecked")
    static > M scan(M map, Tuple2 zero,
            BiFunction, ? super Tuple2, ? extends Tuple2> operation,
            Function>, Traversable>> finisher) {
        return (M) Collections.scanLeft(map, zero, operation, finisher);
    }

    static > Iterator slideBy(M map, OfEntries ofEntries,
            Function, ?> classifier) {
        return map.iterator().slideBy(classifier).map(ofEntries);
    }

    static > Iterator sliding(M map, OfEntries ofEntries, int size) {
        return sliding(map, ofEntries, size, 1);
    }

    static > Iterator sliding(M map, OfEntries ofEntries, int size, int step) {
        return map.iterator().sliding(size, step).map(ofEntries);
    }

    static > Tuple2 span(M map, OfEntries ofEntries,
            Predicate> predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        final Tuple2>, Iterator>> t = map.iterator().span(predicate);
        return Tuple.of(ofEntries.apply(t._1), ofEntries.apply(t._2));
    }

    @SuppressWarnings("unchecked")
    static > Option tailOption(M map) {
        return map.isEmpty() ? Option.none() : Option.some((M) map.tail());
    }

    static > M take(M map, OfEntries ofEntries, int n) {
        if (n >= map.size()) {
            return map;
        } else {
            return ofEntries.apply(map.iterator().take(n));
        }
    }

    static > M takeRight(M map, OfEntries ofEntries, int n) {
        if (n >= map.size()) {
            return map;
        } else {
            return ofEntries.apply(map.iterator().takeRight(n));
        }
    }

    static > M takeUntil(M map, OfEntries ofEntries,
            Predicate> predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        return takeWhile(map, ofEntries, predicate.negate());
    }

    static > M takeWhile(M map, OfEntries ofEntries,
            Predicate> predicate) {
        Objects.requireNonNull(predicate, "predicate is null");
        final M taken = ofEntries.apply(map.iterator().takeWhile(predicate));
        return taken.size() == map.size() ? map : taken;
    }

    @FunctionalInterface
    interface OfEntries> extends Function>, M> {
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy