LruCacheClassPathRepository.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  *  Unless required by applicable law or agreed to in writing, software
  12.  *  distributed under the License is distributed on an "AS IS" BASIS,
  13.  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  *  See the License for the specific language governing permissions and
  15.  *  limitations under the License.
  16.  */
  17. package org.apache.bcel.util;

  18. import java.util.LinkedHashMap;
  19. import java.util.Map;

  20. import org.apache.bcel.classfile.JavaClass;

  21. /**
  22.  * Maintains a least-recently-used (LRU) cache of {@link JavaClass} with maximum size {@code cacheSize}.
  23.  *
  24.  * <p>
  25.  * This repository supports a class path consisting of too many JAR files to handle in {@link ClassPathRepository} or
  26.  * {@link MemorySensitiveClassPathRepository} without causing {@code OutOfMemoryError}.
  27.  * </p>
  28.  *
  29.  * @since 6.4.0
  30.  */
  31. public class LruCacheClassPathRepository extends AbstractClassPathRepository {

  32.     private final LinkedHashMap<String, JavaClass> loadedClasses;

  33.     public LruCacheClassPathRepository(final ClassPath path, final int cacheSize) {
  34.         super(path);

  35.         if (cacheSize < 1) {
  36.             throw new IllegalArgumentException("cacheSize must be a positive number.");
  37.         }
  38.         final int initialCapacity = (int) (0.75 * cacheSize);
  39.         final boolean accessOrder = true; // Evicts least-recently-accessed
  40.         loadedClasses = new LinkedHashMap<String, JavaClass>(initialCapacity, cacheSize, accessOrder) {

  41.             private static final long serialVersionUID = 1L;

  42.             @Override
  43.             protected boolean removeEldestEntry(final Map.Entry<String, JavaClass> eldest) {
  44.                 return size() > cacheSize;
  45.             }
  46.         };
  47.     }

  48.     @Override
  49.     public void clear() {
  50.         loadedClasses.clear();
  51.     }

  52.     @Override
  53.     public JavaClass findClass(final String className) {
  54.         return loadedClasses.get(className);
  55.     }

  56.     @Override
  57.     public void removeClass(final JavaClass javaClass) {
  58.         loadedClasses.remove(javaClass.getClassName());
  59.     }

  60.     @Override
  61.     public void storeClass(final JavaClass javaClass) {
  62.         // Not storing parent's _loadedClass
  63.         loadedClasses.put(javaClass.getClassName(), javaClass);
  64.         javaClass.setRepository(this);
  65.     }
  66. }