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

com.codename1.io.CacheMap Maven / Gradle / Ivy

/*
 * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code 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
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores
 * CA 94065 USA or visit www.oracle.com if you need additional information or
 * have any questions.
 */

package com.codename1.io;

import com.codename1.ui.Display;
import java.io.IOException;
import java.util.Enumeration;
import java.util.Hashtable;
import java.util.Vector;

/**
 * A cache map is essentially a hashtable that indexes entries based on age and is
 * limited to a fixed size. Hence when an entry is placed into the cache map and the
 * cache size needs to increase, the least referenced entry is removed.
 * A cache hit is made both on fetching and putting, hence frequently fetched elements
 * will never be removed from a sufficiently large cache.
 * Cache can work purely in memory or swap data into storage based on user definitions.
 * Notice that this class isn't threadsafe.
 *
 * @author Shai Almog
 */
public class CacheMap {
    private int cacheSize = 10;
    private Hashtable memoryCache = new Hashtable();
    private Hashtable weakCache = new Hashtable();

    private int storageCacheSize = 0;
    private Vector storageCacheContentVec;
    private String cachePrefix = "";
    private boolean alwaysStore;
    private int storageKey = -1;
    
    private Vector getStorageCacheContent() {
        if(storageCacheContentVec == null) {
            storageCacheContentVec = (Vector)Storage.getInstance().readObject("$CACHE$Idx" + cachePrefix);
            if(storageCacheContentVec == null) {
                storageCacheContentVec = new Vector();
            }
        }
        return storageCacheContentVec;
    }
    
    /**
     * Default constructor
     */
    public CacheMap() {
    }
    

    /**
     * Creates a cache map with a prefix string
     * @param prefix string to prepend to the cache entries in storage
     */
    public CacheMap(String prefix) {
        this.cachePrefix = prefix;
    }
    
    
    
    /**
     * Indicates the size of the memory cache after which the cache won't grow further
     * Size is indicated by number of elements stored and not by KB or similar benchmark!
     *
     * @return the cacheSize
     */
    public int getCacheSize() {
        return cacheSize;
    }

    /**
     * Indicates the size of the memory cache after which the cache won't grow further
     * Size is indicated by number of elements stored and not by KB or similar benchmark!
     * 
     * @param cacheSize the cacheSize to set
     */
    public void setCacheSize(int cacheSize) {
        this.cacheSize = cacheSize;
    }

    /**
     * Puts the given key/value pair in the cache
     *
     * @param key the key
     * @param value the value
     */
    public void put(Object key, Object value) {
        if(cacheSize <= memoryCache.size()) {
            // we need to find the oldest entry
            Enumeration e = memoryCache.keys();
            long oldest = System.currentTimeMillis();
            Object oldestKey = null;
            Object[] oldestValue = null;
            while(e.hasMoreElements()) {
                Object currentKey = e.nextElement();
                Object[] currentValue = (Object[])memoryCache.get(currentKey);
                long currentAge = ((Long)currentValue[0]).longValue();
                if(currentAge <= oldest || oldestValue == null) {
                    oldest = currentAge;
                    oldestKey = currentKey;
                    oldestValue = currentValue;
                }
            }
            placeInStorageCache(oldestKey, oldest, oldestValue[1]);
            weakCache.put(oldestKey, Display.getInstance().createSoftWeakRef(oldestValue[1]));
            memoryCache.remove(oldestKey);
        }
        long lastAccess = System.currentTimeMillis();
        memoryCache.put(key, new Object[]{new Long(lastAccess), value});
        if(alwaysStore) {
            placeInStorageCache(key, lastAccess, value);
        }
    }


    /**
     * Deletes a cached entry
     * 
     * @param key entry to remove from the cache
     */
    public void delete(Object key) {
        memoryCache.remove(key);
        weakCache.remove(key);
        Vector storageCacheContent = getStorageCacheContent();
        int s = storageCacheContent.size();
        for(int iter = 0 ; iter < s ; iter++) {
            Object[] obj = (Object[])storageCacheContent.elementAt(iter);
            if(obj[1].equals(key)) {
                Storage.getInstance().deleteStorageFile("$CACHE$" + cachePrefix + key.toString());
                obj[0] = new Long(Long.MIN_VALUE);
                obj[1] = obj[0];                
                Storage.getInstance().writeObject("$CACHE$Idx" + cachePrefix, storageCacheContent);
                return;
            }
        }
    }
    
    /**
     * Returns the object matching the given key
     *
     * @param key key object
     * @return value from a previous put or null
     */
    public Object get(Object key) {
        Object[] o = (Object[])memoryCache.get(key);
        if(o != null) {
            return o[1];
        }
        Object ref = weakCache.get(key);
        if(ref != null) {
            ref = Display.getInstance().extractHardRef(ref);
            if(ref != null) {
                // cache hit! Promote it to the hard cache again
                put(key, ref);
                return ref;
            }
        }
        if(storageCacheSize > 0) {
            Vector storageCacheContent = getStorageCacheContent();
            for(int iter = 0 ; iter < storageCacheContent.size() ; iter++) {
                Object[] obj = (Object[])storageCacheContent.elementAt(iter);
                if(obj[1].equals(key)) {
                    // place the object back into the memory cache and return the value
                    Vector v = (Vector)Storage.getInstance().readObject("$CACHE$" + cachePrefix + key.toString());
                    if(v != null) {
                        Object val = v.elementAt(0);
                        put(key, val);
                        return val;
                    }
                    return null;
                }
            }
        }
        return null;
    }

    /**
     * Clears the caches for this cache object
     */
    public void clearAllCache() {
        clearMemoryCache();
        clearStorageCache();
    }

    /**
     * Clears the memory cache
     */
    public void clearMemoryCache() {
        memoryCache.clear();
        weakCache.clear();
    }

    private void placeInStorageCache(Object key, long lastAccessed, Object value) {
        if(storageCacheSize < 1) {
            return;
        }

        Vector storageCacheContent = getStorageCacheContent();
        int vecSize = storageCacheContent.size();

        // find the best place
        for(int iter = 0 ; iter < vecSize ; iter++) {
            Object[] index = (Object[])storageCacheContent.elementAt(iter);
            Object indexKey = index[1];
            if(indexKey.equals(key)) {
                // already in storage just update the data
                placeInStorageCache(iter, key, lastAccessed, value);
                return;
            }
        }
        
        if(storageCacheContent.size() < storageCacheSize) {
            placeInStorageCache(storageCacheContent.size(), key, lastAccessed, value);
        } else {
            long smallest = Long.MAX_VALUE;
            int offset = 0;

            // find the best place
            for(int iter = 0 ; iter < vecSize ; iter++) {
                Object[] index = (Object[])storageCacheContent.elementAt(iter);
                long current = ((Long)index[0]).longValue();
                if(smallest > current) {
                    smallest = current;
                    offset = iter;
                }
            }
            placeInStorageCache(offset, key, lastAccessed, value);
        }
    }

    private void placeInStorageCache(int offset, Object key, long lastAccessed, Object value) {
        Vector v = new Vector();
        v.addElement(value);
        Long l = new Long(lastAccessed);
        v.addElement(l);
        v.addElement(key);
        Storage.getInstance().writeObject("$CACHE$" + cachePrefix + key.toString(), v);
        Vector storageCacheContent = getStorageCacheContent();
        if(storageCacheContent.size() > offset) {
            storageCacheContent.setElementAt(new Object[] {l, key}, offset);
        } else {
            storageCacheContent.insertElementAt(new Object[] {l, key}, offset);
        }
        Storage.getInstance().writeObject("$CACHE$Idx" + cachePrefix, storageCacheContent);
    }
    
    /**
     * Returns the keys for all the objects currently in cache, this is useful
     * to traverse all the objects and refresh them without actually deleting
     * the cache and fetching them from scratch.
* Important this vector is a copy of a current state, keys might * not exist anymore or might change, others might be added in the interim. * * @return a vector containing a snapshot of the current elements within the * cache. */ public Vector getKeysInCache() { Vector r = new Vector(); Enumeration en = memoryCache.keys(); while(en.hasMoreElements()) { r.addElement(en.nextElement()); } Vector storageCacheContent = getStorageCacheContent(); for(int iter = 0 ; iter < storageCacheContent.size() ; iter++) { Object[] o = (Object[])storageCacheContent.elementAt(iter); if(!r.contains(o[1])) { r.addElement(o[1]); } } return r; } private Vector fetchFromStorageCache(int offset) { Vector v = getStorageCacheContent(); Object[] arr = (Object[])v.elementAt(offset); return (Vector)Storage.getInstance().readObject("$CACHE$" + cachePrefix + arr[1].toString()); } /** * Clears the storage cache */ public void clearStorageCache() { if(storageCacheSize > 0) { Vector v = getStorageCacheContent(); int s = v.size(); for(int iter = 0 ; iter < s ; iter++) { Object[] arr = (Object[])v.elementAt(iter); Storage.getInstance().deleteStorageFile("$CACHE$" + cachePrefix + arr[iter].toString()); } Storage.getInstance().deleteStorageFile("$CACHE$Idx" + cachePrefix); storageCacheContentVec = new Vector(); } } /** * Indicates the size of the storage cache after which the cache won't grow further * Size is indicated by number of elements stored and not by KB or similar benchmark! * * @return the storageCacheSize */ public int getStorageCacheSize() { return storageCacheSize; } /** * Indicates the size of the storage cache after which the cache won't grow further * Size is indicated by number of elements stored and not by KB or similar benchmark! * * @param storageCacheSize the storageCacheSize to set */ public void setStorageCacheSize(int storageCacheSize) { this.storageCacheSize = storageCacheSize; /*for(int iter = 0 ; iter < storageCacheSize ; iter++) { Vector v = fetchFromStorageCache(iter); if(v != null) { Object o = new Object[] {v.elementAt(1), v.elementAt(2)}; Vector storageCacheContent = getStorageCacheContent(); if(iter >= storageCacheContent.size()) { storageCacheContent.addElement(o); } else { storageCacheContent.insertElementAt(o, iter); } } }*/ if(storageCacheSize == 0) { alwaysStore = false; } } /** * A prefix prepended to storage entries to differentiate them * @return the cachePrefix */ public String getCachePrefix() { return cachePrefix; } /** * A prefix prepended to storage entries to differentiate them * @param cachePrefix the cachePrefix to set */ public void setCachePrefix(String cachePrefix) { this.cachePrefix = cachePrefix; } /** * When set to true indicates that all entries should be persisted to storage * for a constantly persisting cache * @return the alwaysStore */ public boolean isAlwaysStore() { return alwaysStore; } /** * When set to true indicates that all entries should be persisted to storage * for a constantly persisting cache * @param alwaysStore the alwaysStore to set */ public void setAlwaysStore(boolean alwaysStore) { this.alwaysStore = alwaysStore; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy