Subversion Repositories mkgmap

Rev

Rev 3803 | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
 * Copyright (C) 2008 - 2012.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 3 or
 * version 2 as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 */


package uk.me.parabola.mkgmap.reader.osm;

import java.util.AbstractMap;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.TreeSet;

/**
 * Store the tags that belong to an Element.
 *
 * Used to use a HashMap for this.  We used to have a requirement to be able
 * to add to the map during iteration over it but now the main reason
 * to keep this class is that it is more memory efficient than a regular
 * HashMap (hash maps are the main use of memory in the
 * application), as it doesn't allocate a Map.Entry object for every tag.
 * Performance of the whole application is unchanged compared with when
 * a regular HashMap was used.
 *
 * It doesn't fully behave the same way that a map would.
 *
 * @author Steve Ratcliffe
 */

public class Tags {
        private static final int INIT_SIZE = 8;
        private static final TagDict tagDict = TagDict.getInstance();  

        private short keySize;
        private short capacity;
       
        private short size;

        private short[] keys;
        private String[] values;

        public Tags() {
                keys = new short[INIT_SIZE];
                values = new String[INIT_SIZE];
                capacity = INIT_SIZE;
        }

        public String get(String key) {
                short k = tagDict.xlate(key);
                int ind = keyPos(k);
                if (ind < 0)
                        return null;

                return values[ind];
        }
       
        public String get(short key) {
                int ind = keyPos(key);
               
                if (ind < 0)
                        return null;

                return values[ind];
        }
       
        /**
         * Retrieves the number of tags.
         * @return number of tags
         */

        public int size() {
                return size;
        }

        public String put(String key, String value) {
                assert key != null : "key is null";
                short dictIdx = tagDict.xlate(key);
                return put(dictIdx,value);
        }

        public String put(short key, String value) {
                assert value != null : "value is null";
                ensureSpace();
                int ind = keyPos(key);
                if (ind < 0)
                        assert false : "keyPos(" + key + ") returns null - size = " + keySize + ", capacity = " + capacity;
                keys[ind] = key;

                String old = values[ind];
                if (old == null) {
                        keySize++;
                        size++;
                }
                values[ind] = value;

                return old;
        }

        public String remove(short key) {
                int k = keyPos(key);

                if (k >= 0 && values[k] != null) {
                        // because of the way this works, you can never remove keys
                        // except when resizing.
                        String old = values[k];
                        values[k] = null;
                        size--;
                        return old;
                }
                return null;
        }
       
        public String remove(String key) {
                short kd = tagDict.xlate(key);
                return remove(kd);
        }
       
        /**
         * Make a deep copy of this object.
         * @return A copy of this object.
         */

        public Tags copy() {
                Tags cp = new Tags();
                cp.keySize = keySize;
                cp.size = size;
                cp.capacity = capacity;

                cp.keys = Arrays.copyOf(keys, keys.length);
                cp.values = Arrays.copyOf(values, values.length);
                return cp;
        }

        private void ensureSpace() {
                while (keySize + 1 >= capacity) {
                        short ncap = (short) (capacity*2);
                        short[] okey = keys;
                        String[] oval = values;
                        keys = new short[ncap];
                        values = new String[ncap];
                        capacity = ncap;
                        keySize = 0;
                        size = 0;
                        for (int i = 0; i < okey.length; i++) {
                                short k = okey[i];
                                String v = oval[i]; // null if tag has been removed
                                if (k != TagDict.INVALID_TAG_VALUE && v != null){
                                        //put(keyDict.get(k), v);
                                        int ind = keyPos(k);
                                        keys[ind] = k;
                                        values[ind] = v;
                                        ++keySize;
                                        ++size;
                                }
                        }
                }
                assert keySize < capacity;
        }

        private int keyPos(short key) {
                int k = key & (capacity - 1);

                int i = k;
                do {
                        if (keys[i] == TagDict.INVALID_TAG_VALUE || keys[i] == key)
                                return i;
                        i++;
                        if (i >= capacity)
                                i = 0;
                } while (i != k);
                return -1;
        }

        public Iterator<Map.Entry<String, String>> entryIterator() {
                return new Iterator<Map.Entry<String, String>>() {
                        private int pos;
                        private int done;

                        public boolean hasNext() {
                                return done < size;
                        }

                        public Map.Entry<String, String> next() {
                                if (done >= size)
                                        throw new NoSuchElementException();

                                for (; pos < capacity; pos++) {
                                        if (values[pos] != null) {
                                                break;
                                        }
                                }
                                Map.Entry<String, String> entry = new AbstractMap.SimpleEntry<>(tagDict.get(keys[pos]), values[pos]);

                                pos++;
                                done++;
                                return entry;
                        }

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

        public Iterator<Map.Entry<Short, String>> entryShortIterator() {
                return new Iterator<Map.Entry<Short, String>>() {
                        private int pos;
                        private int done;

                        public boolean hasNext() {
                                return done < size;
                        }

                        public Map.Entry<Short, String> next() {
                                if (done >= size)
                                        throw new NoSuchElementException();

                                for (; pos < capacity; pos++) {
                                        if (values[pos] != null) {
                                                break;
                                        }
                                }

                                Map.Entry<Short, String> entry = new AbstractMap.SimpleEntry<>(keys[pos], values[pos]);

                                pos++;
                                done++;
                                return entry;
                        }

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

        public Map<String, String> getTagsWithPrefix(String prefix, boolean removePrefix) {
                Map<String, String> map = new HashMap<>();

                int prefixLen = prefix.length();
                for(int i = 0; i < capacity; ++i) {
                        if (keys[i] != 0){
                                String k = tagDict.get(keys[i]);
                                if(k != null && k.startsWith(prefix)) {
                                if(removePrefix)
                                                map.put(k.substring(prefixLen), values[i]);
                                else
                                                map.put(k, values[i]);
                                }
                        }
                }

                return map;
        }
       

        public String toString() {
                // sort the tags by key to make the result predictable and easier to read
                TreeSet<String> sorted = new TreeSet<>();
                for (int i = 0; i < capacity; i++) {
                        if (values[i] != null) {
                                sorted.add(tagDict.get(keys[i]) + "=" + values[i]);
                        }
                }
                return sorted.toString();
        }
}