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

org.apache.solr.util.CharArrayMap Maven / Gradle / Ivy

The newest version!
/**
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 org.apache.solr.util;

import java.util.*;
import java.io.Serializable;

/**
 * A simple class that stores key Strings as char[]'s in a
 * hash table. Note that this is not a general purpose
 * class.  For example, it cannot remove items from the
 * map, nor does it resize its hash table to be smaller,
 * etc.  It is designed to be quick to retrieve items
 * by char[] keys without the necessity of converting
 * to a String first.
 */

public class CharArrayMap extends AbstractMap
                               implements Map, Cloneable, Serializable
{
  private final static int INIT_SIZE = 2;
  private char[][] keys;
  private Object[] values;
  private int count;
  private final boolean ignoreCase;

  /** Create map with enough capacity to hold startSize
   *  terms */
  public CharArrayMap(int initialCapacity, boolean ignoreCase) {
    this.ignoreCase = ignoreCase;
    int size = INIT_SIZE;
    // load factor of .75, inverse is 1.25, or x+x/4
    initialCapacity = initialCapacity + (initialCapacity >>2);
    while(size <= initialCapacity)
      size <<= 1;
    keys = new char[size][];
    values = new Object[size];
  }

  public boolean ignoreCase() {
    return ignoreCase;
  }

  public V get(char[] key) {
    return get(key, 0, key.length);
  }

  public V get(char[] key, int off, int len) {
    return (V)values[getSlot(key, off, len)];
  }

  public V get(CharSequence key) {
    return (V)values[getSlot(key)];
  }

  @Override
  public V get(Object key) {
    return (V)values[getSlot(key)];
  }

  @Override
  public boolean containsKey(Object s) {
    return keys[getSlot(s)] != null; 
  }

  @Override
  public boolean containsValue(Object value) {
    if (value == null) {
      // search for key with a null value
      for (int i=0; i>8)+code)|1;
      do {
        code += inc;
        pos = code & (keys.length-1);
        key2 = keys[pos];
      } while (key2 != null && !equals(key, off, len, key2));
    }
    return pos;
  }

  /** Returns true if the String is in the set */
  private int getSlot(CharSequence key) {
    int code = getHashCode(key);
    int pos = code & (keys.length-1);
    char[] key2 = keys[pos];
    if (key2 != null && !equals(key, key2)) {
      final int inc = ((code>>8)+code)|1;
      do {
        code += inc;
        pos = code & (keys.length-1);
        key2 = keys[pos];
      } while (key2 != null && !equals(key, key2));
    }
    return pos;
  }

  public V put(CharSequence key, V val) {
    return put(key.toString(), val); // could be more efficient
  }

  @Override
  public V put(String key, V val) {
    return put(key.toCharArray(), val);
  }

  /** Add this key,val pair to the map.
   * The char[] key is directly used, no copy is made.
   * If ignoreCase is true for this Map, the key array will be directly modified.
   * The user should never modify the key after calling this method.
   */
  public V put(char[] key, Object val) {
    if (ignoreCase)
      for(int i=0;i< key.length;i++)
        key[i] = Character.toLowerCase(key[i]);
    int slot = getSlot(key, 0, key.length);
    if (keys[slot] == null) count++;
    Object prev = values[slot];
    keys[slot] = key;
    values[slot] = val;

    if (count + (count>>2) >= keys.length) {
      rehash();
    }

    return (V)prev;
  }
  

  private boolean equals(char[] text1, int off, int len, char[] text2) {
    if (len != text2.length)
      return false;
    if (ignoreCase) {
      for(int i=0;i> entrySet() {
    return new EntrySet();
  }

  /** Returns an EntryIterator over this Map. */
  public EntryIterator iterator() {
    return new EntryIterator();
  }

  /** public iterator class so efficient methods are exposed to users */
  public class EntryIterator implements Iterator> {
    int pos=-1;
    int lastPos;

    EntryIterator() {
      goNext();
    }

    private void goNext() {
      lastPos = pos;
      pos++;
      while (pos < keys.length && keys[pos] == null) pos++;
    }

    public boolean hasNext() {
      return pos < keys.length;
    }

    /** gets the next key... do not modify the returned char[] */
    public char[] nextKey() {
      goNext();
      return keys[lastPos];
    }

    /** gets the next key as a newly created String object */
    public String nextKeyString() {
      return new String(nextKey());
    }

    /** returns the value associated with the last key returned */
    public V currentValue() {
      return (V)values[lastPos];
    }

    /** sets the value associated with the last key returned */    
    public V setValue(V value) {
      V old = (V)values[lastPos];
      values[lastPos] = value;
      return old;      
    }

    /** Returns an Entry object created on the fly...
     * use nextCharArray() + currentValie() for better efficiency. */
    public Map.Entry next() {
      goNext();
      return new MapEntry(lastPos);
    }

    public void remove() {
      throw new UnsupportedOperationException();
    }
  }


  private class MapEntry implements Map.Entry {
    final int pos;

    MapEntry(int pos) {
      this.pos = pos;
    }

    public char[] getCharArr() {
      return keys[pos];
    }

    public String getKey() {
      return new String(getCharArr());
    }

    public V getValue() {
      return (V)values[pos];
    }

    public V setValue(V value) {
      V old = (V)values[pos];
      values[pos] = value;
      return old;
    }

    public String toString() {
      return getKey() + '=' + getValue();
    }
  }



  private class EntrySet extends AbstractSet> {
    public EntryIterator iterator() {
      return new EntryIterator();
    }
    public boolean contains(Object o) {
      if (!(o instanceof Map.Entry))
        return false;
      Map.Entry e = (Map.Entry)o;
      Object key = e.getKey();
      if (key==null) return false;  // we don't support null keys
      Object val = e.getValue();
      Object v = get(key);
      return v==null ? val==null : v.equals(val);
    }
    public boolean remove(Object o) {
      throw new UnsupportedOperationException();
    }
    public int size() {
      return count;
    }
    public void clear() {
      CharArrayMap.this.clear();
    }
  }

  @Override
  public Object clone() {
    CharArrayMap map = null;
    try {
      map = (CharArrayMap)super.clone();
      map.keys = keys.clone();
      map.values = values.clone();
    } catch (CloneNotSupportedException e) {
      // impossible
    }
    return map;
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy