SpringFactoriesLoader as the simplest possible mechanism for BeanInfoFactory loading
This commit is contained in:
parent
0a42c80c11
commit
787d8f5916
|
|
@ -62,6 +62,10 @@ public class CachedIntrospectionResults {
|
||||||
|
|
||||||
private static final Log logger = LogFactory.getLog(CachedIntrospectionResults.class);
|
private static final Log logger = LogFactory.getLog(CachedIntrospectionResults.class);
|
||||||
|
|
||||||
|
/** Stores the BeanInfoFactory instances */
|
||||||
|
private static List<BeanInfoFactory> beanInfoFactories =
|
||||||
|
SpringFactoriesLoader.loadFactories(BeanInfoFactory.class, CachedIntrospectionResults.class.getClassLoader());
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set of ClassLoaders that this CachedIntrospectionResults class will always
|
* Set of ClassLoaders that this CachedIntrospectionResults class will always
|
||||||
* accept classes from, even if the classes do not qualify as cache-safe.
|
* accept classes from, even if the classes do not qualify as cache-safe.
|
||||||
|
|
@ -75,11 +79,6 @@ public class CachedIntrospectionResults {
|
||||||
*/
|
*/
|
||||||
static final Map<Class, Object> classCache = Collections.synchronizedMap(new WeakHashMap<Class, Object>());
|
static final Map<Class, Object> classCache = Collections.synchronizedMap(new WeakHashMap<Class, Object>());
|
||||||
|
|
||||||
/** Stores the BeanInfoFactory instances */
|
|
||||||
private static List<BeanInfoFactory> beanInfoFactories;
|
|
||||||
|
|
||||||
private static final Object beanInfoFactoriesMutex = new Object();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Accept the given ClassLoader as cache-safe, even if its classes would
|
* Accept the given ClassLoader as cache-safe, even if its classes would
|
||||||
|
|
@ -230,7 +229,6 @@ public class CachedIntrospectionResults {
|
||||||
}
|
}
|
||||||
|
|
||||||
BeanInfo beanInfo = null;
|
BeanInfo beanInfo = null;
|
||||||
List<BeanInfoFactory> beanInfoFactories = getBeanInfoFactories(beanClass.getClassLoader());
|
|
||||||
for (BeanInfoFactory beanInfoFactory : beanInfoFactories) {
|
for (BeanInfoFactory beanInfoFactory : beanInfoFactories) {
|
||||||
beanInfo = beanInfoFactory.getBeanInfo(beanClass);
|
beanInfo = beanInfoFactory.getBeanInfo(beanClass);
|
||||||
if (beanInfo != null) {
|
if (beanInfo != null) {
|
||||||
|
|
@ -325,16 +323,4 @@ public class CachedIntrospectionResults {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static List<BeanInfoFactory> getBeanInfoFactories(ClassLoader classLoader) {
|
|
||||||
if (beanInfoFactories == null) {
|
|
||||||
synchronized (beanInfoFactoriesMutex) {
|
|
||||||
if (beanInfoFactories == null) {
|
|
||||||
beanInfoFactories = Collections.synchronizedList(SpringFactoriesLoader
|
|
||||||
.loadFactories(BeanInfoFactory.class, classLoader));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return beanInfoFactories;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -20,7 +20,6 @@ import java.io.IOException;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.Enumeration;
|
import java.util.Enumeration;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
@ -28,22 +27,19 @@ import java.util.Properties;
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
|
|
||||||
import org.springframework.core.annotation.AnnotationAwareOrderComparator;
|
import org.springframework.core.OrderComparator;
|
||||||
import org.springframework.core.io.UrlResource;
|
import org.springframework.core.io.UrlResource;
|
||||||
import org.springframework.util.Assert;
|
import org.springframework.util.Assert;
|
||||||
import org.springframework.util.ClassUtils;
|
import org.springframework.util.ClassUtils;
|
||||||
import org.springframework.util.StringUtils;
|
import org.springframework.util.StringUtils;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* General purpose factory loading mechanism.
|
* General purpose factory loading mechanism for internal use within the framework.
|
||||||
*
|
*
|
||||||
* <p>The {@code SpringFactoriesLoader} loads and instantiates factories of a given type
|
* <p>The {@code SpringFactoriesLoader} loads and instantiates factories of a given type
|
||||||
* from a given file location. If a location is not given, the {@linkplain
|
* from "META-INF/spring.factories" files. The file should be in {@link Properties} format,
|
||||||
* #DEFAULT_FACTORIES_LOCATION default location} is used.
|
* where the key is the fully qualified interface or abstract class name, and the value
|
||||||
*
|
* is a comma-separated list of implementation class names. For instance:
|
||||||
* <p>The file should be in {@link Properties} format, where the key is the fully
|
|
||||||
* qualified interface or abstract class name, and the value is a comma separated list of
|
|
||||||
* implementations. For instance:
|
|
||||||
*
|
*
|
||||||
* <pre class="code">example.MyService=example.MyServiceImpl1,example.MyServiceImpl2</pre>
|
* <pre class="code">example.MyService=example.MyServiceImpl1,example.MyServiceImpl2</pre>
|
||||||
*
|
*
|
||||||
|
|
@ -51,147 +47,72 @@ import org.springframework.util.StringUtils;
|
||||||
* {@code MyServiceImpl2} are the two implementations.
|
* {@code MyServiceImpl2} are the two implementations.
|
||||||
*
|
*
|
||||||
* @author Arjen Poutsma
|
* @author Arjen Poutsma
|
||||||
|
* @author Juergen Hoeller
|
||||||
* @since 3.2
|
* @since 3.2
|
||||||
*/
|
*/
|
||||||
public abstract class SpringFactoriesLoader {
|
public abstract class SpringFactoriesLoader {
|
||||||
|
|
||||||
|
/** The location to look for the factories. Can be present in multiple JAR files. */
|
||||||
|
private static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";
|
||||||
|
|
||||||
private static final Log logger = LogFactory.getLog(SpringFactoriesLoader.class);
|
private static final Log logger = LogFactory.getLog(SpringFactoriesLoader.class);
|
||||||
|
|
||||||
/** The location to look for the factories. Can be present in multiple JAR files. */
|
|
||||||
public static final String DEFAULT_FACTORIES_LOCATION = "META-INF/spring.factories";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Loads the factory implementations of the given type from the default location.
|
|
||||||
*
|
|
||||||
* <p>The returned factories are ordered in accordance with the {@link
|
|
||||||
* AnnotationAwareOrderComparator}.
|
|
||||||
*
|
|
||||||
* @param factoryClass the interface or abstract class representing the factory
|
|
||||||
* @throws IllegalArgumentException in case of errors
|
|
||||||
*/
|
|
||||||
public static <T> List<T> loadFactories(Class<T> factoryClass) {
|
|
||||||
return loadFactories(factoryClass, null, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Loads the factory implementations of the given type from the default location, using
|
* Loads the factory implementations of the given type from the default location, using
|
||||||
* the given class loader.
|
* the given class loader.
|
||||||
*
|
* <p>The returned factories are ordered in accordance with the {@link OrderComparator}.
|
||||||
* <p>The returned factories are ordered in accordance with the {@link
|
|
||||||
* AnnotationAwareOrderComparator}.
|
|
||||||
*
|
|
||||||
* @param factoryClass the interface or abstract class representing the factory
|
* @param factoryClass the interface or abstract class representing the factory
|
||||||
* @param classLoader the ClassLoader to use for loading, can be {@code null} to use the
|
* @param classLoader the ClassLoader to use for loading (can be {@code null} to use the default)
|
||||||
* default
|
|
||||||
* @throws IllegalArgumentException in case of errors
|
|
||||||
*/
|
*/
|
||||||
public static <T> List<T> loadFactories(Class<T> factoryClass,
|
public static <T> List<T> loadFactories(Class<T> factoryClass, ClassLoader classLoader) {
|
||||||
ClassLoader classLoader) {
|
|
||||||
return loadFactories(factoryClass, classLoader, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Loads the factory implementations of the given type from the given location, using the
|
|
||||||
* given class loader.
|
|
||||||
*
|
|
||||||
* <p>The returned factories are ordered in accordance with the {@link
|
|
||||||
* AnnotationAwareOrderComparator}.
|
|
||||||
*
|
|
||||||
* @param factoryClass the interface or abstract class representing the factory
|
|
||||||
* @param classLoader the ClassLoader to use for loading, can be {@code null} to use the
|
|
||||||
* default
|
|
||||||
* @param factoriesLocation the factories file location, can be {@code null} to use the
|
|
||||||
* {@linkplain #DEFAULT_FACTORIES_LOCATION default}
|
|
||||||
* @throws IllegalArgumentException in case of errors
|
|
||||||
*/
|
|
||||||
public static <T> List<T> loadFactories(Class<T> factoryClass,
|
|
||||||
ClassLoader classLoader, String factoriesLocation) {
|
|
||||||
Assert.notNull(factoryClass, "'factoryClass' must not be null");
|
Assert.notNull(factoryClass, "'factoryClass' must not be null");
|
||||||
|
|
||||||
if (classLoader == null) {
|
if (classLoader == null) {
|
||||||
classLoader = ClassUtils.getDefaultClassLoader();
|
classLoader = SpringFactoriesLoader.class.getClassLoader();
|
||||||
}
|
}
|
||||||
if (factoriesLocation == null) {
|
List<String> factoryNames = loadFactoryNames(factoryClass, classLoader);
|
||||||
factoriesLocation = DEFAULT_FACTORIES_LOCATION;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<String> factoryNames =
|
|
||||||
loadFactoryNames(factoryClass, classLoader, factoriesLocation);
|
|
||||||
|
|
||||||
if (logger.isTraceEnabled()) {
|
if (logger.isTraceEnabled()) {
|
||||||
logger.trace(
|
logger.trace("Loaded [" + factoryClass.getName() + "] names: " + factoryNames);
|
||||||
"Loaded [" + factoryClass.getName() + "] names: " + factoryNames);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
List<T> result = new ArrayList<T>(factoryNames.size());
|
List<T> result = new ArrayList<T>(factoryNames.size());
|
||||||
for (String factoryName : factoryNames) {
|
for (String factoryName : factoryNames) {
|
||||||
result.add(instantiateFactory(factoryName, factoryClass, classLoader));
|
result.add(instantiateFactory(factoryName, factoryClass, classLoader));
|
||||||
}
|
}
|
||||||
|
OrderComparator.sort(result);
|
||||||
Collections.sort(result, new AnnotationAwareOrderComparator());
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static List<String> loadFactoryNames(Class<?> factoryClass,
|
private static List<String> loadFactoryNames(Class<?> factoryClass, ClassLoader classLoader) {
|
||||||
ClassLoader classLoader, String factoriesLocation) {
|
|
||||||
|
|
||||||
String factoryClassName = factoryClass.getName();
|
String factoryClassName = factoryClass.getName();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
List<String> result = new ArrayList<String>();
|
List<String> result = new ArrayList<String>();
|
||||||
Enumeration<URL> urls = classLoader.getResources(factoriesLocation);
|
Enumeration<URL> urls = classLoader.getResources(FACTORIES_RESOURCE_LOCATION);
|
||||||
while (urls.hasMoreElements()) {
|
while (urls.hasMoreElements()) {
|
||||||
URL url = urls.nextElement();
|
URL url = urls.nextElement();
|
||||||
Properties properties =
|
Properties properties = PropertiesLoaderUtils.loadProperties(new UrlResource(url));
|
||||||
PropertiesLoaderUtils.loadProperties(new UrlResource(url));
|
|
||||||
String factoryClassNames = properties.getProperty(factoryClassName);
|
String factoryClassNames = properties.getProperty(factoryClassName);
|
||||||
result.addAll(Arrays.asList(
|
result.addAll(Arrays.asList(StringUtils.commaDelimitedListToStringArray(factoryClassNames)));
|
||||||
StringUtils.commaDelimitedListToStringArray(factoryClassNames)));
|
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
catch (IOException ex) {
|
catch (IOException ex) {
|
||||||
throw new IllegalArgumentException(
|
throw new IllegalArgumentException("Unable to load [" + factoryClass.getName() +
|
||||||
"Unable to load [" + factoryClass.getName() +
|
"] factories from location [" + FACTORIES_RESOURCE_LOCATION + "]", ex);
|
||||||
"] factories from location [" +
|
|
||||||
factoriesLocation + "]", ex);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
private static <T> T instantiateFactory(String instanceClassName,
|
private static <T> T instantiateFactory(String instanceClassName, Class<T> factoryClass, ClassLoader classLoader) {
|
||||||
Class<T> factoryClass, ClassLoader classLoader) {
|
|
||||||
try {
|
try {
|
||||||
Class<?> instanceClass = ClassUtils.forName(instanceClassName, classLoader);
|
Class<?> instanceClass = ClassUtils.forName(instanceClassName, classLoader);
|
||||||
if (!factoryClass.isAssignableFrom(instanceClass)) {
|
if (!factoryClass.isAssignableFrom(instanceClass)) {
|
||||||
throw new IllegalArgumentException(
|
throw new IllegalArgumentException(
|
||||||
"Class [" + instanceClassName + "] is not assignable to [" +
|
"Class [" + instanceClassName + "] is not assignable to [" + factoryClass.getName() + "]");
|
||||||
factoryClass.getName() + "]");
|
|
||||||
}
|
}
|
||||||
return (T) instanceClass.newInstance();
|
return (T) instanceClass.newInstance();
|
||||||
}
|
}
|
||||||
catch (ClassNotFoundException ex) {
|
catch (Throwable ex) {
|
||||||
throw new IllegalArgumentException(
|
throw new IllegalArgumentException("Cannot instantiate factory class: " + factoryClass.getName(), ex);
|
||||||
factoryClass.getName() + " class [" + instanceClassName +
|
|
||||||
"] not found", ex);
|
|
||||||
}
|
|
||||||
catch (LinkageError err) {
|
|
||||||
throw new IllegalArgumentException(
|
|
||||||
"Invalid " + factoryClass.getName() + " class [" + instanceClassName +
|
|
||||||
"]: problem with handler class file or dependent class", err);
|
|
||||||
}
|
|
||||||
catch (InstantiationException ex) {
|
|
||||||
throw new IllegalArgumentException(
|
|
||||||
"Could not instantiate bean class [" + instanceClassName +
|
|
||||||
"]: Is it an abstract class?", ex);
|
|
||||||
}
|
|
||||||
catch (IllegalAccessException ex) {
|
|
||||||
throw new IllegalArgumentException(
|
|
||||||
"Could not instantiate bean class [" + instanceClassName +
|
|
||||||
"Is the constructor accessible?", ex);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,47 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright 2002-2012 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.core.io.support;
|
|
||||||
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
|
||||||
import static org.junit.Assert.assertTrue;
|
|
||||||
import org.junit.Test;
|
|
||||||
|
|
||||||
/** @author Arjen Poutsma */
|
|
||||||
public class SpringFactoriesLoaderTests {
|
|
||||||
|
|
||||||
private static final String FACTORIES_LOCATION =
|
|
||||||
"org/springframework/core/io/support/springFactoriesLoaderTests.properties";
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void loadFactories() {
|
|
||||||
List<DummyFactory> factories = SpringFactoriesLoader
|
|
||||||
.loadFactories(DummyFactory.class, null, FACTORIES_LOCATION);
|
|
||||||
|
|
||||||
assertEquals(2, factories.size());
|
|
||||||
assertTrue(factories.get(0) instanceof MyDummyFactory1);
|
|
||||||
assertTrue(factories.get(1) instanceof MyDummyFactory2);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test(expected = IllegalArgumentException.class)
|
|
||||||
public void loadInvalid() {
|
|
||||||
SpringFactoriesLoader.loadFactories(String.class, null,
|
|
||||||
FACTORIES_LOCATION);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
Loading…
Reference in New Issue