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

org.neo4j.kernel.impl.api.state.StateDefaults Maven / Gradle / Ivy

Go to download

Neo4j kernel is a lightweight, embedded Java database designed to store data structured as graphs rather than tables. For more information, see http://neo4j.org.

There is a newer version: 5.26.0
Show newest version
/*
 * Copyright (c) 2002-2015 "Neo Technology,"
 * Network Engine for Objects in Lund AB [http://neotechnology.com]
 *
 * This file is part of Neo4j.
 *
 * Neo4j is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * 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.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
package org.neo4j.kernel.impl.api.state;

import java.util.Collection;
import java.util.HashMap;
import java.util.Map;

import org.neo4j.helpers.collection.Iterables;

/**
 * Utility for {@linkplain #get(TxState, Object) retrieving} and
 * {@linkplain #getOrCreate(TxState, Object) initializing} lazy state held in maps in {@link TxState}.
 * 

* {@linkplain #get(TxState, Object) Retrieving} state only guarantees that a readable object is returned, it does not * guarantee a writable version. This allows us to return a read-only default value if the state has not been * initialized. Only when invoking {@link #getOrCreate(TxState, Object)} do we need to return a writable version, and * at this point the state is initialized, if it has not been before, by creating a new instance and putting it in the * map. *

* There are two categories of methods in this class, one category concerns the value type, and the other concerns the * {@linkplain TxState value holder}. Implementations for methods of these two categories are preferably provided in * two stages, as to have each of those participating types contribute their part to the final implementation. * See this illustrative example: *

 * interface ValueType
 * {
 *     class Mutable extends ValueType {}
 *
 *     // stage one - implement methods concerning the value type
 *     abstract class Defaults extends StateDefaults<String, ValueType, Mutable>
 *     {
 *         private static final ValueType DEFAULT = new ValueType() { ... };
 *         ValueType defaultValue() { return DEFAULT; }
 *         Mutable createValue( String key ) { return new Mutable(); }
 *     }
 * }
 *
 * class ValueHolder
 * {
 *     // stage two - implement methods concerning the reference to the state
 *     private Map<String, ValueType.Mutable> state;
 *     private static final StateDefaults<String, ValueType, ValueType.Mutable> STATE = new ValueType.Defaults()
 *     {
 *         Map<String, ValueType.Mutable> getMap( ValueHolder holder ) { return holder.state; }
 *         void setMap( ValueHolder holder, Map<String, ValueType.Mutable> map ) { holder.state = map; }
 *     };
 * }
 * 
* * @param The type of the key in the map for the state accessed by an instance of this class * @param The read-only version of the value type stored in the state * @param The read/write version of the value type stored in the state */ abstract class StateDefaults { final RO get( TxState state, KEY key ) { Map map = getMap( state ); if ( map == null ) { return defaultValue(); } RO value = map.get( key ); return value == null ? defaultValue() : value; } final RW getOrCreate( TxState state, KEY key ) { Map map = getMap( state ); if ( map == null ) { setMap( state, map = new HashMap<>() ); } RW value = map.get( key ); if ( value == null ) { map.put( key, value = createValue( key ) ); } return value; } final Iterable values( TxState state ) { Map map = getMap( state ); if ( map == null ) { return Iterables.empty(); } @SuppressWarnings("unchecked") Collection values = map.values(); return values; } /** Implemented for the value holder - get the map from the state field. */ abstract Map getMap( TxState state ); /** Implemented for the value holder - set the map to the state field. */ abstract void setMap( TxState state, Map map ); /** Implemented for the value type - initializes state by creating a new instance. */ abstract RW createValue( KEY key ); /** Implemented for the value type - returns a default read-only version of the value type. */ abstract RO defaultValue(); }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy