/* * Copyright 2012-2017 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.springframework.boot.type.classreading; import java.io.IOException; import java.util.Map; import org.springframework.core.io.Resource; import org.springframework.core.io.ResourceLoader; import org.springframework.core.type.classreading.CachingMetadataReaderFactory; import org.springframework.core.type.classreading.MetadataReader; import org.springframework.core.type.classreading.MetadataReaderFactory; import org.springframework.core.type.classreading.SimpleMetadataReaderFactory; import org.springframework.util.ConcurrentReferenceHashMap; /** * Caching implementation of the {@link MetadataReaderFactory} interface backed by a * {@link ConcurrentReferenceHashMap} , caching {@link MetadataReader} per Spring * {@link Resource} handle (i.e. per ".class" file). * * @author Phillip Webb * @since 1.4.0 * @see CachingMetadataReaderFactory */ public class ConcurrentReferenceCachingMetadataReaderFactory extends SimpleMetadataReaderFactory { private final Map<Resource, MetadataReader> cache = new ConcurrentReferenceHashMap<>(); /** * Create a new {@link ConcurrentReferenceCachingMetadataReaderFactory} instance for * the default class loader. */ public ConcurrentReferenceCachingMetadataReaderFactory() { super(); } /** * Create a new {@link ConcurrentReferenceCachingMetadataReaderFactory} instance for * the given resource loader. * @param resourceLoader the Spring ResourceLoader to use (also determines the * ClassLoader to use) */ public ConcurrentReferenceCachingMetadataReaderFactory( ResourceLoader resourceLoader) { super(resourceLoader); } /** * Create a new {@link ConcurrentReferenceCachingMetadataReaderFactory} instance for * the given class loader. * @param classLoader the ClassLoader to use */ public ConcurrentReferenceCachingMetadataReaderFactory(ClassLoader classLoader) { super(classLoader); } @Override public MetadataReader getMetadataReader(Resource resource) throws IOException { MetadataReader metadataReader = this.cache.get(resource); if (metadataReader == null) { metadataReader = createMetadataReader(resource); this.cache.put(resource, metadataReader); } return metadataReader; } /** * Create the meta-data reader. * @param resource the source resource. * @return the meta-data reader * @throws IOException on error */ protected MetadataReader createMetadataReader(Resource resource) throws IOException { return super.getMetadataReader(resource); } /** * Clear the entire MetadataReader cache, removing all cached class metadata. */ public void clearCache() { this.cache.clear(); } }