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

com.squareup.gifencoder.HashMultiset Maven / Gradle / Ivy

Go to download

A pure Java library implementing the GIF89a specification. Suitable for use on Android.

There is a newer version: 0.10.1
Show newest version
/*
 * Copyright (C) 2015 Square, Inc.
 *
 * 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 com.squareup.gifencoder;

import java.util.AbstractCollection;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;

public final class HashMultiset extends AbstractCollection implements Multiset {
  private final Map elementCounts = new HashMap<>();
  private int size;

  public HashMultiset() {
  }

  public HashMultiset(Collection source) {
    addAll(source);
  }

  @Override public void add(E element, int n) {
    Count count = elementCounts.get(element);
    if (count != null) {
      count.value += n;
    } else {
      elementCounts.put(element, new Count(n));
    }
    size += n;
  }

  @Override public boolean add(E element) {
    add(element, 1);
    return true;
  }

  @Override public int remove(Object element, int n) {
    Count count = elementCounts.get(element);
    if (count == null) {
      return 0;
    }

    if (n < count.value) {
      count.value -= n;
      size -= n;
      return n;
    }

    elementCounts.remove(element);
    size -= count.value;
    return count.value;
  }

  @Override public boolean remove(Object element) {
    return remove(element, 1) > 0;
  }

  @Override public Iterator iterator() {
    return new HashMultisetIterator();
  }

  @Override public int size() {
    return size;
  }

  @Override public int count(Object element) {
    Count countOrNull = elementCounts.get(element);
    return countOrNull != null ? countOrNull.value : 0;
  }

  @Override public Set getDistinctElements() {
    return elementCounts.keySet();
  }

  private final class HashMultisetIterator implements Iterator {
    final Iterator> distinctElementIterator;
    E currentElement;
    int currentCount;
    boolean currentElementRemoved;

    HashMultisetIterator() {
      this.distinctElementIterator = elementCounts.entrySet().iterator();
      this.currentCount = 0;
    }

    @Override public boolean hasNext() {
      return currentCount > 0 || distinctElementIterator.hasNext();
    }

    @Override public E next() {
      if (!hasNext()) {
        throw new NoSuchElementException("iterator has been exhausted");
      }

      if (currentCount == 0) {
        Map.Entry next = distinctElementIterator.next();
        currentElement = next.getKey();
        currentCount = next.getValue().value;
      }

      --currentCount;
      currentElementRemoved = false;
      return currentElement;
    }

    @Override public void remove() {
      if (currentElement == null) {
        throw new IllegalStateException("next() has not been called");
      }
      if (currentElementRemoved) {
        throw new IllegalStateException("remove() already called for current element");
      }
      HashMultiset.this.remove(currentElement);
    }
  }

  private static final class Count {
    int value;

    Count(int value) {
      this.value = value;
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy