1 /* 2 * ArrayCache 3 * 4 * Author: Lasse Collin <lasse.collin@tukaani.org> 5 * 6 * This file has been put into the public domain. 7 * You can do whatever you want with this file. 8 */ 9 10 package org.tukaani.xz; 11 12 /** 13 * Caches large arrays for reuse (base class and a dummy cache implementation). 14 * <p> 15 * When compressing or decompressing many (very) small files in a row, the 16 * time spent in construction of new compressor or decompressor objects 17 * can be longer than the time spent in actual compression or decompression. 18 * A large part of this initialization overhead comes from allocation and 19 * garbage collection of large arrays. 20 * <p> 21 * The {@code ArrayCache} API provides a way to cache large array allocations 22 * for reuse. It can give a major performance improvement when compressing or 23 * decompressing many tiny files. If you are only (de)compressing one or two 24 * files or the files a very big, array caching won't improve anything, 25 * although it won't make anything slower either. 26 * <p> 27 * <b>Important: The users of ArrayCache don't return the allocated arrays 28 * back to the cache in all situations.</b> 29 * This a reason why it's called a cache instead of a pool. 30 * If it is important to be able to return every array back to a cache, 31 * {@link ResettableArrayCache} can be useful. 32 * <p> 33 * In compressors (OutputStreams) the arrays are returned to the cache 34 * when a call to {@code finish()} or {@code close()} returns 35 * successfully (no exceptions are thrown). 36 * <p> 37 * In decompressors (InputStreams) the arrays are returned to the cache when 38 * the decompression is successfully finished ({@code read} returns {@code -1}) 39 * or {@code close()} or {@code close(boolean)} is called. This is true even 40 * if closing throws an exception. 41 * <p> 42 * Raw decompressors don't support {@code close(boolean)}. With raw 43 * decompressors, if one wants to put the arrays back to the cache without 44 * closing the underlying {@code InputStream}, one can wrap the 45 * {@code InputStream} into {@link CloseIgnoringInputStream} when creating 46 * the decompressor instance. Then one can use {@code close()}. 47 * <p> 48 * Different cache implementations can be extended from this base class. 49 * All cache implementations must be thread safe. 50 * <p> 51 * This class also works as a dummy cache that simply calls {@code new} 52 * to allocate new arrays and doesn't try to cache anything. A statically 53 * allocated dummy cache is available via {@link #getDummyCache()}. 54 * <p> 55 * If no {@code ArrayCache} is specified when constructing a compressor or 56 * decompressor, the default {@code ArrayCache} implementation is used. 57 * See {@link #getDefaultCache()} and {@link #setDefaultCache(ArrayCache)}. 58 * <p> 59 * This is a class instead of an interface because it's possible that in the 60 * future we may want to cache other array types too. New methods can be 61 * added to this class without breaking existing cache implementations. 62 * 63 * @since 1.7 64 * 65 * @see BasicArrayCache 66 */ 67 public class ArrayCache { 68 /** 69 * Global dummy cache instance that is returned by {@code getDummyCache()}. 70 */ 71 private static final ArrayCache dummyCache = new ArrayCache(); 72 73 /** 74 * Global default {@code ArrayCache} that is used when no other cache has 75 * been specified. 76 */ 77 private static volatile ArrayCache defaultCache = dummyCache; 78 79 /** 80 * Returns a statically-allocated {@code ArrayCache} instance. 81 * It can be shared by all code that needs a dummy cache. 82 */ getDummyCache()83 public static ArrayCache getDummyCache() { 84 return dummyCache; 85 } 86 87 /** 88 * Gets the default {@code ArrayCache} instance. 89 * This is a global cache that is used when the application 90 * specifies nothing else. The default is a dummy cache 91 * (see {@link #getDummyCache()}). 92 */ getDefaultCache()93 public static ArrayCache getDefaultCache() { 94 // It's volatile so no need for synchronization. 95 return defaultCache; 96 } 97 98 /** 99 * Sets the default {@code ArrayCache} instance. 100 * Use with care. Other libraries using this package probably shouldn't 101 * call this function as libraries cannot know if there are other users 102 * of the xz package in the same application. 103 */ setDefaultCache(ArrayCache arrayCache)104 public static void setDefaultCache(ArrayCache arrayCache) { 105 if (arrayCache == null) 106 throw new NullPointerException(); 107 108 // It's volatile so no need for synchronization. 109 defaultCache = arrayCache; 110 } 111 112 /** 113 * Creates a new {@code ArrayCache} that does no caching 114 * (a dummy cache). If you need a dummy cache, you may want to call 115 * {@link #getDummyCache()} instead. 116 */ ArrayCache()117 public ArrayCache() {} 118 119 /** 120 * Allocates a new byte array. 121 * <p> 122 * This implementation simply returns {@code new byte[size]}. 123 * 124 * @param size the minimum size of the array to allocate; 125 * an implementation may return an array that 126 * is larger than the given {@code size} 127 * 128 * @param fillWithZeros if true, the caller expects that the first 129 * {@code size} elements in the array are zero; 130 * if false, the array contents can be anything, 131 * which speeds things up when reusing a cached 132 * array 133 */ getByteArray(int size, boolean fillWithZeros)134 public byte[] getByteArray(int size, boolean fillWithZeros) { 135 return new byte[size]; 136 } 137 138 /** 139 * Puts the given byte array to the cache. The caller must no longer 140 * use the array. 141 * <p> 142 * This implementation does nothing. 143 */ putArray(byte[] array)144 public void putArray(byte[] array) {} 145 146 /** 147 * Allocates a new int array. 148 * <p> 149 * This implementation simply returns {@code new int[size]}. 150 * 151 * @param size the minimum size of the array to allocate; 152 * an implementation may return an array that 153 * is larger than the given {@code size} 154 * 155 * @param fillWithZeros if true, the caller expects that the first 156 * {@code size} elements in the array are zero; 157 * if false, the array contents can be anything, 158 * which speeds things up when reusing a cached 159 * array 160 */ getIntArray(int size, boolean fillWithZeros)161 public int[] getIntArray(int size, boolean fillWithZeros) { 162 return new int[size]; 163 } 164 165 /** 166 * Puts the given int array to the cache. The caller must no longer 167 * use the array. 168 * <p> 169 * This implementation does nothing. 170 */ putArray(int[] array)171 public void putArray(int[] array) {} 172 } 173