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

com.dyuproject.protostuff.GraphProtostuffOutput Maven / Gradle / Ivy

//========================================================================
//Copyright 2007-2011 David Yu [email protected]
//------------------------------------------------------------------------
//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.dyuproject.protostuff;

import static com.dyuproject.protostuff.WireFormat.WIRETYPE_END_GROUP;
import static com.dyuproject.protostuff.WireFormat.WIRETYPE_REFERENCE;
import static com.dyuproject.protostuff.WireFormat.WIRETYPE_START_GROUP;
import static com.dyuproject.protostuff.WireFormat.makeTag;

import java.io.IOException;
import java.util.Map;

/**
 * A ProtostuffOutput w/c can handle cyclic dependencies when serializing 
 * objects with graph transformations.
 *
 * @author David Yu
 * @created Dec 10, 2010
 */
public final class GraphProtostuffOutput extends FilterOutput
{
    
    private final IdentityMap references;
    private int refCount = 0;
    
    public GraphProtostuffOutput(ProtostuffOutput output)
    {
        super(output);
        references = new IdentityMap();
    }
    
    public GraphProtostuffOutput(ProtostuffOutput output, int initialCapacity)
    {
        super(output);
        references = new IdentityMap(initialCapacity);
    }

    public  void writeObject(int fieldNumber, T value, Schema schema, 
            boolean repeated) throws IOException
    {
        final ProtostuffOutput output = this.output;
        
        if(references.shouldIncrement(refCount, value, output, fieldNumber))
        {
            refCount++;
            
            output.tail = output.sink.writeVarInt32(
                    makeTag(fieldNumber, WIRETYPE_START_GROUP), 
                    output, 
                    output.tail);
            
            schema.writeTo(this, value);
            
            output.tail = output.sink.writeVarInt32(
                    makeTag(fieldNumber, WIRETYPE_END_GROUP), 
                    output, 
                    output.tail);
        }
    }
    
    
    /**
     * A trimed-down version of IdentityHashMap w/c caters to the 
     * specific needs of {@link GraphOutput}.
     */
    private static final class IdentityMap
    {

        /**
         * The initial capacity used by the no-args constructor. MUST be a power of
         * two. The value 32 corresponds to the (specified) expected maximum size of
         * 21, given a load factor of 2/3.
         */
        private static final int DEFAULT_CAPACITY = 32;

        /**
         * The minimum capacity, used if a lower value is implicitly specified by
         * either of the constructors with arguments. The value 4 corresponds to an
         * expected maximum size of 2, given a load factor of 2/3. MUST be a power
         * of two.
         */
        private static final int MINIMUM_CAPACITY = 4;

        /**
         * The maximum capacity, used if a higher value is implicitly specified by
         * either of the constructors with arguments. MUST be a power of two <=
         * 1<<29.
         */
        private static final int MAXIMUM_CAPACITY = 1 << 29;

        /**
         * The table, resized as necessary. Length MUST always be a power of two.
         */
        private transient Object[] table;

        /**
         * The number of key-value mappings contained in this identity hash map.
         * 
         * @serial
         */
        private int size;

        /*
         * The number of modifications, to support fast-fail iterators
         */
        // private transient volatile int modCount;

        /**
         * The next size value at which to resize (capacity * load factor).
         */
        private transient int threshold;

        /**
         * Constructs a new, empty identity hash map with a default expected maximum
         * size (21).
         */
        public IdentityMap()
        {
            init(DEFAULT_CAPACITY);
        }

        /**
         * Constructs a new, empty map with the specified expected maximum size.
         * Putting more than the expected number of key-value mappings into the map
         * may cause the internal data structure to grow, which may be somewhat
         * time-consuming.
         * 
         * @param expectedMaxSize
         *            the expected maximum size of the map
         * @throws IllegalArgumentException
         *             if expectedMaxSize is negative
         */
        public IdentityMap(int expectedMaxSize)
        {
            if (expectedMaxSize < 0)
                throw new IllegalArgumentException("expectedMaxSize is negative: " + expectedMaxSize);
            init(capacity(expectedMaxSize));
        }

        /**
         * Returns the appropriate capacity for the specified expected maximum size.
         * Returns the smallest power of two between MINIMUM_CAPACITY and
         * MAXIMUM_CAPACITY, inclusive, that is greater than (3 *
         * expectedMaxSize)/2, if such a number exists. Otherwise returns
         * MAXIMUM_CAPACITY. If (3 * expectedMaxSize)/2 is negative, it is assumed
         * that overflow has occurred, and MAXIMUM_CAPACITY is returned.
         */
        private int capacity(int expectedMaxSize)
        {
            // Compute min capacity for expectedMaxSize given a load factor of 2/3
            int minCapacity = (3 * expectedMaxSize) / 2;

            // Compute the appropriate capacity
            int result;
            if (minCapacity > MAXIMUM_CAPACITY || minCapacity < 0)
            {
                result = MAXIMUM_CAPACITY;
            }
            else
            {
                result = MINIMUM_CAPACITY;
                while (result < minCapacity)
                    result <<= 1;
            }
            return result;
        }

        /**
         * Initializes object to be an empty map with the specified initial
         * capacity, which is assumed to be a power of two between MINIMUM_CAPACITY
         * and MAXIMUM_CAPACITY inclusive.
         */
        private void init(int initCapacity)
        {
            // assert (initCapacity & -initCapacity) == initCapacity; // power of 2
            // assert initCapacity >= MINIMUM_CAPACITY;
            // assert initCapacity <= MAXIMUM_CAPACITY;

            threshold = (initCapacity * 2) / 3;
            table = new Object[2 * initCapacity];
        }

        /**
         * Returns index for Object x.
         */
        private static int hash(Object x, int length)
        {
            int h = System.identityHashCode(x);
            // Multiply by -127, and left-shift to use least bit as part of hash
            return ((h << 1) - (h << 8)) & (length - 1);
        }

        /**
         * Circularly traverses table of size len.
         */
        private static int nextKeyIndex(int i, int len)
        {
            return (i + 2 < len?i + 2:0);
        }

        /*
         * Returns the value to which the specified key is mapped, or {@code null}
         * if this map contains no mapping for the key.
         * 
         * 

* More formally, if this map contains a mapping from a key {@code k} to a * value {@code v} such that {@code (key == k)}, then this method returns * {@code v}; otherwise it returns {@code null}. (There can be at most one * such mapping.) * *

* A return value of {@code null} does not necessarily indicate that * the map contains no mapping for the key; it's also possible that the map * explicitly maps the key to {@code null}. The {@link #containsKey * containsKey} operation may be used to distinguish these two cases. * * @see #put(Object, Object) * public Integer get(Object k) { Object[] tab = table; int len = tab.length; int i = hash(k, len); while (true) { Object item = tab[i]; if (item == k) return (Integer)tab[i + 1]; if (item == null) return null; i = nextKeyIndex(i, len); } } /* * Associates the specified value with the specified key in this identity * hash map. If the map previously contained a mapping for the key, the old * value is replaced. * * @param key * the key with which the specified value is to be associated * @param value * the value to be associated with the specified key * @return the previous value associated with key, or null * if there was no mapping for key. (A null return * can also indicate that the map previously associated * null with key.) * @see Object#equals(Object) * @see #get(Object) * @see #containsKey(Object) * public Integer put(Object k, Integer value) { Object[] tab = table; int len = tab.length; int i = hash(k, len); Object item; while ((item = tab[i]) != null) { if (item == k) { Integer oldValue = (Integer)tab[i + 1]; tab[i + 1] = value; return oldValue; } i = nextKeyIndex(i, len); } // modCount++; tab[i] = k; tab[i + 1] = value; if (++size >= threshold) resize(len); // len == 2 * current capacity. return null; }*/ /** * Returns true if the provided int should increment(unique index id). */ public boolean shouldIncrement(int value, Object k, WriteSession output, int fieldNumber) throws IOException { Object[] tab = table; int len = tab.length; int i = hash(k, len); Object item; while ((item = tab[i]) != null) { if (item == k) { if(k instanceof Map.Entry // filter on standard java map impls only && k.getClass().getName().startsWith("java.util")) { // IdentityHashMap and EnumMap re-uses the same Map.Entry. // It simply holds references to the actual data (key/value). return true; } output.tail = output.sink.writeVarInt32( ((Integer)tab[i + 1]).intValue(), output, output.sink.writeVarInt32( makeTag(fieldNumber, WIRETYPE_REFERENCE), output, output.tail)); return false; } i = nextKeyIndex(i, len); } // modCount++; tab[i] = k; tab[i + 1] = Integer.valueOf(value); if (++size >= threshold) resize(len); // len == 2 * current capacity. return true; } /** * Resize the table to hold given capacity. * * @param newCapacity * the new capacity, must be a power of two. */ private void resize(int newCapacity) { // assert (newCapacity & -newCapacity) == newCapacity; // power of 2 int newLength = newCapacity * 2; Object[] oldTable = table; int oldLength = oldTable.length; if (oldLength == 2 * MAXIMUM_CAPACITY) { // can't expand any further if (threshold == MAXIMUM_CAPACITY - 1) throw new IllegalStateException("Capacity exhausted."); threshold = MAXIMUM_CAPACITY - 1; // Gigantic map! return; } if (oldLength >= newLength) return; Object[] newTable = new Object[newLength]; threshold = newLength / 3; for (int j = 0; j < oldLength; j += 2) { Object key = oldTable[j]; if (key != null) { Object value = oldTable[j + 1]; oldTable[j] = null; oldTable[j + 1] = null; int i = hash(key, newLength); while (newTable[i] != null) i = nextKeyIndex(i, newLength); newTable[i] = key; newTable[i + 1] = value; } } table = newTable; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy