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

com.yahoo.sketches.frequencies.LongsSketch Maven / Gradle / Ivy

There is a newer version: 0.13.4
Show newest version
/*
 * Copyright 2016, Yahoo! Inc. Licensed under the terms of the Apache License 2.0. See LICENSE file
 * at the project root for terms.
 */

package com.yahoo.sketches.frequencies;

import static com.yahoo.sketches.Util.LS;
import static com.yahoo.sketches.Util.toLog2;
import static com.yahoo.sketches.frequencies.PreambleUtil.EMPTY_FLAG_MASK;
import static com.yahoo.sketches.frequencies.PreambleUtil.SER_VER;
import static com.yahoo.sketches.frequencies.PreambleUtil.extractActiveItems;
import static com.yahoo.sketches.frequencies.PreambleUtil.extractFamilyID;
import static com.yahoo.sketches.frequencies.PreambleUtil.extractFlags;
import static com.yahoo.sketches.frequencies.PreambleUtil.extractLgCurMapSize;
import static com.yahoo.sketches.frequencies.PreambleUtil.extractLgMaxMapSize;
import static com.yahoo.sketches.frequencies.PreambleUtil.extractPreLongs;
import static com.yahoo.sketches.frequencies.PreambleUtil.extractSerVer;
import static com.yahoo.sketches.frequencies.PreambleUtil.insertActiveItems;
import static com.yahoo.sketches.frequencies.PreambleUtil.insertFamilyID;
import static com.yahoo.sketches.frequencies.PreambleUtil.insertFlags;
import static com.yahoo.sketches.frequencies.PreambleUtil.insertLgCurMapSize;
import static com.yahoo.sketches.frequencies.PreambleUtil.insertLgMaxMapSize;
import static com.yahoo.sketches.frequencies.PreambleUtil.insertPreLongs;
import static com.yahoo.sketches.frequencies.PreambleUtil.insertSerVer;
import static com.yahoo.sketches.frequencies.Util.LG_MIN_MAP_SIZE;
import static com.yahoo.sketches.frequencies.Util.SAMPLE_SIZE;

import java.util.ArrayList;
import java.util.Comparator;

import com.yahoo.memory.Memory;
import com.yahoo.memory.NativeMemory;
import com.yahoo.sketches.Family;
import com.yahoo.sketches.SketchesArgumentException;
import com.yahoo.sketches.SketchesStateException;

/**
 * 

This sketch is useful for tracking approximate frequencies of long items with optional * associated counts (long item, long count) that are members of a multiset of * such items. The true frequency of an item is defined to be the sum of associated counts.

* *

This implementation provides the following capabilities:

*
    *
  • Estimate the frequency of an item.
  • *
  • Return upper and lower bounds of any item, such that the true frequency is always * between the upper and lower bounds.
  • *
  • Return a global maximum error that holds for all items in the stream.
  • *
  • Return an array of frequent items that qualify either a NO_FALSE_POSITIVES or a * NO_FALSE_NEGATIVES error type.
  • *
  • Merge itself with another sketch object created from this class.
  • *
  • Serialize/Deserialize to/from a String or byte array.
  • *
* *

Space Usage

* *

The sketch is initialized with a maxMapSize that specifies the maximum physical * length of the internal hash map of the form (long item, long count). * The maxMapSize must be a power of 2.

* *

The hash map starts at a very small size (8 entries), and grows as needed up to the * specified maxMapSize.

* *

The internal memory space usage of this sketch is 18 * mapSize bytes, plus a small * constant number of additional bytes. The internal memory space usage of this sketch will never * exceed 18 * maxMapSize bytes, plus a small constant number of additional bytes.

* *

Maximum Capacity of the Sketch

* *

The LOAD_FACTOR for the hash map is internally set at 75%, * which means at any time the map capacity of (item, count) pairs is mapCap = * 0.75 * mapSize. * The maximum capacity of (item, count) pairs of the sketch is maxMapCap = * 0.75 * maxMapSize.

* *

Updating the sketch with (item, count) pairs

* *

If the item is found in the hash map, the mapped count field (the "counter") is * incremented by the incoming count, otherwise, a new counter "(item, count) pair" is * created. If the number of tracked counters reaches the maximum capacity of the hash map * the sketch decrements all of the counters (by an approximately computed median), and * removes any non-positive counters.

* *

Accuracy

* *

If fewer than 0.75 * maxMapSize different items are inserted into the sketch the * estimated frequencies returned by the sketch will be exact.

* *

The logic of the frequent items sketch is such that the stored counts and true counts are * never too different. * More specifically, for any item, the sketch can return an estimate of the * true frequency of item, along with upper and lower bounds on the frequency * (that hold deterministically).

* *

For this implementation and for a specific active item, it is guaranteed that * the true frequency will be between the Upper Bound (UB) and the Lower Bound (LB) computed for * that item. Specifically, (UB- LB) ≤ W * epsilon, where W denotes the * sum of all item counts, and epsilon = 3.5/M, where M is the maxMapSize.

* *

This is a worst case guarantee that applies to arbitrary inputs.1 * For inputs typically seen in practice (UB-LB) is usually much smaller. *

* *

Background

* *

This code implements a variant of what is commonly known as the "Misra-Gries * algorithm". Variants of it were discovered and rediscovered and redesigned several times * over the years:

*
  • "Finding repeated elements", Misra, Gries, 1982
  • *
  • "Frequency estimation of Internet packet streams with limited space" Demaine, * Lopez-Ortiz, Munro, 2002
  • *
  • "A simple algorithm for finding frequent elements in streams and bags" Karp, Shenker, * Papadimitriou, 2003
  • *
  • "Efficient Computation of Frequent and Top-k Elements in Data Streams" Metwally, * Agrawal, Abbadi, 2006
  • *
* * 1 For speed we do employ some randomization that introduces a small probability that * our proof of the worst-case bound might not apply to a given run. However, we have ensured * that this probability is extremely small. For example, if the stream causes one table purge * (rebuild), our proof of the worst case bound applies with probability at least 1 - 1E-14. * If the stream causes 1E9 purges, our proof applies with probability at least 1 - 1E-5. * * @author Justin Thaler * @author Lee Rhodes */ public class LongsSketch { private static final int STR_PREAMBLE_TOKENS = 6; /** * Log2 Maximum length of the arrays internal to the hash map supported by the data * structure. */ private int lgMaxMapSize; /** * The current number of counters supported by the hash map. */ private int curMapCap; //the threshold to purge /** * Tracks the total of decremented counts. */ private long offset; /** * The sum of all frequencies of the stream so far. */ private long streamLength = 0; /** * The maximum number of samples used to compute approximate median of counters when doing * decrement */ private int sampleSize; /** * Hash map mapping stored items to approximate counts */ private ReversePurgeLongHashMap hashMap; /** * Construct this sketch with the parameter maxMapSize and the default initialMapSize (8). * * @param maxMapSize Determines the physical size of the internal hash map managed by this * sketch and must be a power of 2. The maximum capacity of this internal hash map is * 0.75 times * maxMapSize. Both the ultimate accuracy and size of this sketch are a * function of maxMapSize. */ public LongsSketch(final int maxMapSize) { this(toLog2(maxMapSize, "maxMapSize"), LG_MIN_MAP_SIZE); } /** * Construct this sketch with parameter lgMapMapSize and lgCurMapSize. This internal * constructor is used when deserializing the sketch. * * @param lgMaxMapSize Log2 of the physical size of the internal hash map managed by this * sketch. The maximum capacity of this internal hash map is 0.75 times 2^lgMaxMapSize. * Both the ultimate accuracy and size of this sketch are a function of lgMaxMapSize. * * @param lgCurMapSize Log2 of the starting (current) physical size of the internal hash * map managed by this sketch. */ LongsSketch(final int lgMaxMapSize, final int lgCurMapSize) { //set initial size of hash map this.lgMaxMapSize = Math.max(lgMaxMapSize, LG_MIN_MAP_SIZE); final int lgCurMapSz = Math.max(lgCurMapSize, LG_MIN_MAP_SIZE); hashMap = new ReversePurgeLongHashMap(1 << lgCurMapSz); this.curMapCap = hashMap.getCapacity(); final int maxMapCap = (int) ((1 << lgMaxMapSize) * ReversePurgeLongHashMap.getLoadFactor()); offset = 0; sampleSize = Math.min(SAMPLE_SIZE, maxMapCap); } /** * Returns a sketch instance of this class from the given srcMem, * which must be a Memory representation of this sketch class. * * @param srcMem a Memory representation of a sketch of this class. * See Memory * @return a sketch instance of this class. */ public static LongsSketch getInstance(final Memory srcMem) { final long pre0 = PreambleUtil.checkPreambleSize(srcMem); //make sure preamble will fit final int maxPreLongs = Family.FREQUENCY.getMaxPreLongs(); final int preLongs = extractPreLongs(pre0); //Byte 0 final int serVer = extractSerVer(pre0); //Byte 1 final int familyID = extractFamilyID(pre0); //Byte 2 final int lgMaxMapSize = extractLgMaxMapSize(pre0); //Byte 3 final int lgCurMapSize = extractLgCurMapSize(pre0); //Byte 4 final boolean empty = (extractFlags(pre0) & EMPTY_FLAG_MASK) != 0; //Byte 5 // Checks final boolean preLongsEq1 = (preLongs == 1); //Byte 0 final boolean preLongsEqMax = (preLongs == maxPreLongs); if (!preLongsEq1 && !preLongsEqMax) { throw new SketchesArgumentException( "Possible Corruption: PreLongs must be 1 or " + maxPreLongs + ": " + preLongs); } if (serVer != SER_VER) { //Byte 1 throw new SketchesArgumentException( "Possible Corruption: Ser Ver must be " + SER_VER + ": " + serVer); } final int actFamID = Family.FREQUENCY.getID(); //Byte 2 if (familyID != actFamID) { throw new SketchesArgumentException( "Possible Corruption: FamilyID must be " + actFamID + ": " + familyID); } if (empty ^ preLongsEq1) { //Byte 5 and Byte 0 throw new SketchesArgumentException( "Possible Corruption: (PreLongs == 1) ^ Empty == True."); } if (empty) { return new LongsSketch(lgMaxMapSize, LG_MIN_MAP_SIZE); } //get full preamble final long[] preArr = new long[preLongs]; srcMem.getLongArray(0, preArr, 0, preLongs); LongsSketch fls = new LongsSketch(lgMaxMapSize, lgCurMapSize); fls.streamLength = 0; //update after fls.offset = preArr[3]; final int preBytes = preLongs << 3; final int activeItems = extractActiveItems(preArr[1]); //Get countArray final long[] countArray = new long[activeItems]; srcMem.getLongArray(preBytes, countArray, 0, activeItems); //Get itemArray final int itemsOffset = preBytes + 8 * activeItems; final long[] itemArray = new long[activeItems]; srcMem.getLongArray(itemsOffset, itemArray, 0, activeItems); //update the sketch for (int i = 0; i < activeItems; i++) { fls.update(itemArray[i], countArray[i]); } fls.streamLength = preArr[2]; //override streamLength due to updating return fls; } /** * Returns a sketch instance of this class from the given String, * which must be a String representation of this sketch class. * * @param string a String representation of a sketch of this class. * @return a sketch instance of this class. */ public static LongsSketch getInstance(final String string) { final String[] tokens = string.split(","); if (tokens.length < STR_PREAMBLE_TOKENS + 2) { throw new SketchesArgumentException( "String not long enough: " + tokens.length); } final int serVer = Integer.parseInt(tokens[0]); final int famID = Integer.parseInt(tokens[1]); final int lgMax = Integer.parseInt(tokens[2]); final int flags = Integer.parseInt(tokens[3]); final long streamLength = Long.parseLong(tokens[4]); final long offset = Long.parseLong(tokens[5]); //should always get at least the next 2 from the map final int numActive = Integer.parseInt(tokens[6]); final int lgCur = Integer.numberOfTrailingZeros(Integer.parseInt(tokens[7])); //checks if (serVer != SER_VER) { throw new SketchesArgumentException("Possible Corruption: Bad SerVer: " + serVer); } Family.FREQUENCY.checkFamilyID(famID); final boolean empty = flags > 0; final boolean zeroStream = (streamLength == 0); if (empty ^ zeroStream) { throw new SketchesArgumentException( "Possible Corruption: (Empty ^ StreamLength=0) = true : Empty: " + empty + ", strLen: " + streamLength); } final int numTokens = tokens.length; if (2 * numActive != (numTokens - STR_PREAMBLE_TOKENS - 2)) { throw new SketchesArgumentException( "Possible Corruption: Incorrect # of tokens: " + numTokens + ", numActive: " + numActive); } final LongsSketch sketch = new LongsSketch(lgMax, lgCur); sketch.streamLength = streamLength; sketch.offset = offset; sketch.hashMap = deserializeFromStringArray(tokens); return sketch; } //Serialization /** * Returns a String representation of this sketch * * @return a String representation of this sketch */ public String serializeToString() { final StringBuilder sb = new StringBuilder(); //start the string with parameters of the sketch final int serVer = SER_VER; //0 final int famID = Family.FREQUENCY.getID(); //1 final int lgMaxMapSz = lgMaxMapSize; //2 final int flags = (hashMap.getNumActive() == 0) ? EMPTY_FLAG_MASK : 0; //3 final String fmt = "%d,%d,%d,%d,%d,%d,"; final String s = String.format(fmt, serVer, famID, lgMaxMapSz, flags, streamLength, offset); sb.append(s); sb.append(hashMap.serializeToString()); //numActive, curMaplen, key[i], value[i], ... // maxMapCap, samplesize are deterministic functions of maxMapSize, // so we don't need them in the serialization return sb.toString(); } /** * Returns a byte array representation of this sketch * @return a byte array representation of this sketch */ public byte[] toByteArray() { final int preLongs, outBytes; final boolean empty = isEmpty(); final int activeItems = getNumActiveItems(); if (empty) { preLongs = 1; outBytes = 8; } else { preLongs = Family.FREQUENCY.getMaxPreLongs(); outBytes = (preLongs + 2 * activeItems) << 3; //2 because both keys and values are longs } final byte[] outArr = new byte[outBytes]; final Memory mem = new NativeMemory(outArr); // build first preLong empty or not long pre0 = 0L; pre0 = insertPreLongs(preLongs, pre0); //Byte 0 pre0 = insertSerVer(SER_VER, pre0); //Byte 1 pre0 = insertFamilyID(Family.FREQUENCY.getID(), pre0); //Byte 2 pre0 = insertLgMaxMapSize(lgMaxMapSize, pre0); //Byte 3 pre0 = insertLgCurMapSize(hashMap.getLgLength(), pre0); //Byte 4 pre0 = (empty) ? insertFlags(EMPTY_FLAG_MASK, pre0) : insertFlags(0, pre0); //Byte 5 if (empty) { mem.putLong(0, pre0); } else { final long pre = 0; final long[] preArr = new long[preLongs]; preArr[0] = pre0; preArr[1] = insertActiveItems(activeItems, pre); preArr[2] = this.streamLength; preArr[3] = this.offset; mem.putLongArray(0, preArr, 0, preLongs); final int preBytes = preLongs << 3; mem.putLongArray(preBytes, hashMap.getActiveValues(), 0, activeItems); mem.putLongArray(preBytes + (activeItems << 3), hashMap.getActiveKeys(), 0, activeItems); } return outArr; } /** * Update this sketch with an item and a frequency count of one. * @param item for which the frequency should be increased. */ public void update(final long item) { update(item, 1); } /** * Update this sketch with a item and a positive frequency count. * @param item for which the frequency should be increased. The item can be any long value * and is only used by the sketch to determine uniqueness. * @param count the amount by which the frequency of the item should be increased. * An count of zero is a no-op, and a negative count will throw an exception. */ public void update(final long item, final long count) { if (count == 0) { return; } if (count < 0) { throw new SketchesArgumentException("Count may not be negative"); } this.streamLength += count; hashMap.adjustOrPutValue(item, count); if (getNumActiveItems() > curMapCap) { //over the threshold, we need to do something if (hashMap.getLgLength() < lgMaxMapSize) { //below tgt size, we can grow hashMap.resize(2 * hashMap.getLength()); curMapCap = hashMap.getCapacity(); } else { //At tgt size, must purge offset += hashMap.purge(sampleSize); if (getNumActiveItems() > getMaximumMapCapacity()) { throw new SketchesStateException("Purge did not reduce active items."); } } } } /** * This function merges the other sketch into this one. * The other sketch may be of a different size. * * @param other sketch of this class * @return a sketch whose estimates are within the guarantees of the * largest error tolerance of the two merged sketches. */ public LongsSketch merge(final LongsSketch other) { if (other == null) { return this; } if (other.isEmpty()) { return this; } final long streamLen = this.streamLength + other.streamLength; //capture before merge final ReversePurgeLongHashMap.Iterator iter = other.hashMap.iterator(); while (iter.next()) { //this may add to offset during rebuilds this.update(iter.getKey(), iter.getValue()); } this.offset += other.offset; this.streamLength = streamLen; //corrected streamLength return this; } /** * Gets the estimate of the frequency of the given item. * Note: The true frequency of a item would be the sum of the counts as a result of the * two update functions. * * @param item the given item * @return the estimate of the frequency of the given item */ public long getEstimate(final long item) { // If item is tracked: // Estimate = itemCount + offset; Otherwise it is 0. final long itemCount = hashMap.get(item); return (itemCount > 0) ? itemCount + offset : 0; } /** * Gets the guaranteed upper bound frequency of the given item. * * @param item the given item * @return the guaranteed upper bound frequency of the given item. That is, a number which * is guaranteed to be no smaller than the real frequency. */ public long getUpperBound(final long item) { // UB = itemCount + offset return hashMap.get(item) + offset; } /** * Gets the guaranteed lower bound frequency of the given item, which can never be * negative. * * @param item the given item. * @return the guaranteed lower bound frequency of the given item. That is, a number which * is guaranteed to be no larger than the real frequency. */ public long getLowerBound(final long item) { //LB = itemCount or 0 return hashMap.get(item); } /** * Returns an array of Rows that include frequent items, estimates, upper and lower bounds * given an ErrorCondition. * *

The method first examines all active items in the sketch (items that have a counter). * *

If ErrorType = NO_FALSE_NEGATIVES, this will include an item in the result * list if getUpperBound(item) > getMaximumError(). * There will be no false negatives, i.e., no Type II error. * There may be items in the set with true frequencies less than the threshold * (false positives).

* *

If ErrorType = NO_FALSE_POSITIVES, this will include an item in the result * list if getLowerBound(item) > getMaximumError(). * There will be no false positives, i.e., no Type I error. * There may be items omitted from the set with true frequencies greater than the * threshold (false negatives). This is a subset of the NO_FALSE_NEGATIVES case.

* * @param errorType determines whether no false positives or no false negatives are * desired. * @return an array of frequent items */ public Row[] getFrequentItems(final ErrorType errorType) { return sortItems(getMaximumError(), errorType); } /** * Row class that defines the return values from a getFrequentItems query. */ public static class Row implements Comparable { final long item; final long est; final long ub; final long lb; private static final String fmt = (" %20d%20d%20d %d"); private static final String hfmt = (" %20s%20s%20s %s"); Row(final long item, final long estimate, final long ub, final long lb) { this.item = item; this.est = estimate; this.ub = ub; this.lb = lb; } /** * @return item of type T */ public long getItem() { return item; } /** * @return the estimate */ public long getEstimate() { return est; } /** * @return the upper bound */ public long getUpperBound() { return ub; } /** * @return return the lower bound */ public long getLowerBound() { return lb; } /** * @return the descriptive row header */ public static String getRowHeader() { return String.format(hfmt,"Est", "UB", "LB", "Item"); } @Override public String toString() { return String.format(fmt, est, ub, lb, item); } /** * This compareTo is strictly limited to the Row.getEstimate() value and does not imply any * ordering whatsoever to the other elements of the row: item and upper and lower bounds. * Defined this way, this compareTo will be consistent with hashCode() and equals(Object). * @param that the other row to compare to. * @return a negative integer, zero, or a positive integer as this.getEstimate() is less than, * equal to, or greater than that.getEstimate(). */ @Override public int compareTo(final Row that) { return (this.est < that.est) ? -1 : (this.est > that.est) ? 1 : 0; } /** * This hashCode is computed only from the Row.getEstimate() value. * Defined this way, this hashCode will be consistent with equals(Object):
* If (x.equals(y)) implies: x.hashCode() == y.hashCode().
* If (!x.equals(y)) does NOT imply: x.hashCode() != y.hashCode(). * @return the hashCode computed from getEstimate(). */ @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + (int) (est ^ (est >>> 32)); return result; } /** * This equals is computed only from the Row.getEstimate() value and does not imply equality * of the other items within the row: item and upper and lower bounds. * Defined this way, this equals will be consistent with compareTo(Row). * @param obj the other row to determine equality with. * @return true if this.getEstimate() equals ((Row)obj).getEstimate(). */ @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null) { return false; } if ( !(obj instanceof Row)) { return false; } Row that = (Row) obj; if (est != that.est) { return false; } return true; } } // End of class Row Row[] sortItems(final long threshold, final ErrorType errorType) { final ArrayList rowList = new ArrayList(); final ReversePurgeLongHashMap.Iterator iter = hashMap.iterator(); if (errorType == ErrorType.NO_FALSE_NEGATIVES) { while (iter.next()) { final long est = getEstimate(iter.getKey()); final long ub = getUpperBound(iter.getKey()); final long lb = getLowerBound(iter.getKey()); if (ub >= threshold) { final Row row = new Row(iter.getKey(), est, ub, lb); rowList.add(row); } } } else { //NO_FALSE_POSITIVES while (iter.next()) { final long est = getEstimate(iter.getKey()); final long ub = getUpperBound(iter.getKey()); final long lb = getLowerBound(iter.getKey()); if (lb >= threshold) { final Row row = new Row(iter.getKey(), est, ub, lb); rowList.add(row); } } } // descending order rowList.sort(new Comparator() { @Override public int compare(final Row r1, final Row r2) { return r2.compareTo(r1); } }); final Row[] rowsArr = rowList.toArray(new Row[rowList.size()]); return rowsArr; } /** * Returns the current number of counters the sketch is configured to support. * * @return the current number of counters the sketch is configured to support. */ public int getCurrentMapCapacity() { return this.curMapCap; } /** * @return An upper bound on the maximum error of getEstimate(item) for any item. * This is equivalent to the maximum distance between the upper bound and the lower bound * for any item. */ public long getMaximumError() { return offset; } /** * Returns true if this sketch is empty * * @return true if this sketch is empty */ public boolean isEmpty() { return getNumActiveItems() == 0; } /** * Returns the sum of the frequencies in the stream seen so far by the sketch * * @return the sum of the frequencies in the stream seen so far by the sketch */ public long getStreamLength() { return this.streamLength; } /** * Returns the maximum number of counters the sketch is configured to support. * * @return the maximum number of counters the sketch is configured to support. */ public int getMaximumMapCapacity() { return (int) ((1 << lgMaxMapSize) * ReversePurgeLongHashMap.getLoadFactor()); } /** * @return the number of active items in the sketch. */ public int getNumActiveItems() { return hashMap.getNumActive(); } /** * Returns the number of bytes required to store this sketch as an array of bytes. * * @return the number of bytes required to store this sketch as an array of bytes. */ public int getStorageBytes() { if (isEmpty()) { return 8; } return 6 * 8 + 16 * getNumActiveItems(); } /** * Resets this sketch to a virgin state. */ public void reset() { hashMap = new ReversePurgeLongHashMap(1 << LG_MIN_MAP_SIZE); this.curMapCap = hashMap.getCapacity(); this.offset = 0; this.streamLength = 0; } /** * Returns a human readable summary of this sketch. * @return a human readable summary of this sketch. */ @Override public String toString() { final StringBuilder sb = new StringBuilder(); sb.append("FrequentLongsSketch:").append(LS); sb.append(" Stream Length : " + streamLength).append(LS); sb.append(" Max Error Offset : " + offset).append(LS); sb.append(hashMap.toString()); return sb.toString(); } /** * Deserializes an array of String tokens into a hash map object of this class. * * @param tokens the given array of Strings tokens. * @return a hash map object of this class */ static ReversePurgeLongHashMap deserializeFromStringArray(String[] tokens) { final int ignore = STR_PREAMBLE_TOKENS; final int numActive = Integer.parseInt(tokens[ignore]); final int length = Integer.parseInt(tokens[ignore + 1]); final ReversePurgeLongHashMap hashMap = new ReversePurgeLongHashMap(length); int j = 2 + ignore; for (int i = 0; i < numActive; i++) { long key = Long.parseLong(tokens[j++]); long value = Long.parseLong(tokens[j++]); hashMap.adjustOrPutValue(key, value); } return hashMap; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy