001/* 002 * Licensed to the Apache Software Foundation (ASF) under one 003 * or more contributor license agreements. See the NOTICE file 004 * distributed with this work for additional information 005 * regarding copyright ownership. The ASF licenses this file 006 * to you under the Apache License, Version 2.0 (the 007 * "License"); you may not use this file except in compliance 008 * with the License. You may obtain a copy of the License at 009 * 010 * https://www.apache.org/licenses/LICENSE-2.0 011 * 012 * Unless required by applicable law or agreed to in writing, 013 * software distributed under the License is distributed on an 014 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 015 * KIND, either express or implied. See the License for the 016 * specific language governing permissions and limitations 017 * under the License. 018 */ 019package org.apache.commons.compress.harmony.unpack200; 020 021import java.util.ArrayList; 022import java.util.Collections; 023import java.util.HashMap; 024import java.util.IdentityHashMap; 025import java.util.List; 026 027/** 028 * The SegmentConstantPool spends a lot of time searching through large arrays of Strings looking for matches. This can be sped up by caching the arrays in 029 * HashMaps so the String keys are looked up and resolve to positions in the array rather than iterating through the arrays each time. 030 * 031 * Because the arrays only grow (never shrink or change) we can use the last known size as a way to determine if the array has changed. 032 * 033 * Note that this cache must be synchronized externally if it is shared. 034 */ 035public class SegmentConstantPoolArrayCache { 036 037 /** 038 * Keeps track of the last known size of an array as well as a HashMap that knows the mapping from element values to the indices of the array 039 * which contain that value. 040 */ 041 protected class CachedArray { 042 String[] primaryArray; 043 int lastKnownSize; 044 HashMap<String, List<Integer>> primaryTable; 045 046 public CachedArray(final String[] array) { 047 this.primaryArray = array; 048 this.lastKnownSize = array.length; 049 this.primaryTable = new HashMap<>(lastKnownSize); 050 cacheIndexes(); 051 } 052 053 /** 054 * Given a primaryArray, cache its values in a HashMap to provide a backwards mapping from element values to element indexes. For instance, a 055 * primaryArray of: {"Zero", "Foo", "Two", "Foo"} would yield a HashMap of: "Zero" -> 0 "Foo" -> 1, 3 "Two" -> 2 which is then cached. 056 */ 057 protected void cacheIndexes() { 058 for (int index = 0; index < primaryArray.length; index++) { 059 final String key = primaryArray[index]; 060 primaryTable.computeIfAbsent(key, k -> new ArrayList<>()).add(Integer.valueOf(index)); 061 } 062 } 063 064 /** 065 * Given a particular key, answer a List of index locations in the array which contain that key. 066 * 067 * If no elements are found, answer an empty list. 068 * 069 * @param key String element of the array 070 * @return List of indexes containing that key in the array. 071 */ 072 public List<Integer> indexesForKey(final String key) { 073 final List<Integer> list = primaryTable.get(key); 074 return list != null ? list : Collections.emptyList(); 075 } 076 077 /** 078 * Answer the last known size of the array cached. If the last known size is not the same as the current size, the array must have changed. 079 * 080 * @return int last known size of the cached array 081 */ 082 public int lastKnownSize() { 083 return lastKnownSize; 084 } 085 } 086 087 protected IdentityHashMap<String[], CachedArray> knownArrays = new IdentityHashMap<>(1000); 088 protected List<Integer> lastIndexes; 089 protected String[] lastArray; 090 091 protected String lastKey; 092 093 /** 094 * Tests whether a String array is correctly cached. Return false if the array is not cached, or if the array cache is outdated. 095 * 096 * @param array of String 097 * @return boolean true if up-to-date cache, otherwise false. 098 */ 099 protected boolean arrayIsCached(final String[] array) { 100 final CachedArray cachedArray = knownArrays.get(array); 101 return !(cachedArray == null || cachedArray.lastKnownSize() != array.length); 102 } 103 104 /** 105 * Caches the array passed in as the argument 106 * 107 * @param array String[] to cache 108 */ 109 protected void cacheArray(final String[] array) { 110 if (arrayIsCached(array)) { 111 throw new IllegalArgumentException("Trying to cache an array that already exists"); 112 } 113 knownArrays.put(array, new CachedArray(array)); 114 // Invalidate the cache-within-a-cache 115 lastArray = null; 116 } 117 118 /** 119 * Gets the indices for the given key in the given array. If no such key exists in the cached array, answer -1. 120 * 121 * @param array String[] array to search for the value 122 * @param key String value for which to search 123 * @return List collection of index positions in the array 124 */ 125 public List<Integer> indexesForArrayKey(final String[] array, final String key) { 126 if (!arrayIsCached(array)) { 127 cacheArray(array); 128 } 129 130 // If the search is one we've just done, don't even 131 // bother looking and return the last indices. This 132 // is a second cache within the cache. This is 133 // efficient because we are usually looking for 134 // several secondary elements with the same primary 135 // key. 136 if (lastArray == array && lastKey == key) { 137 return lastIndexes; 138 } 139 140 // Remember the last thing we found. 141 lastArray = array; 142 lastKey = key; 143 lastIndexes = knownArrays.get(array).indexesForKey(key); 144 145 return lastIndexes; 146 } 147}