Open Source Repository

Home /spring/spring-beans-3.0.5 | Repository Home



org/springframework/beans/factory/support/StaticListableBeanFactory.java
/*
 * Copyright 2002-2009 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.beans.factory.support;

import java.lang.annotation.Annotation;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanCreationException;
import org.springframework.beans.factory.BeanFactoryUtils;
import org.springframework.beans.factory.BeanIsNotAFactoryException;
import org.springframework.beans.factory.BeanNotOfRequiredTypeException;
import org.springframework.beans.factory.FactoryBean;
import org.springframework.beans.factory.ListableBeanFactory;
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
import org.springframework.beans.factory.SmartFactoryBean;
import org.springframework.core.annotation.AnnotationUtils;
import org.springframework.util.StringUtils;

/**
 * Static {@link org.springframework.beans.factory.BeanFactory} implementation
 * which allows to register existing singleton instances programmatically.
 * Does not have support for prototype beans or aliases.
 *
 <p>Serves as example for a simple implementation of the
 {@link org.springframework.beans.factory.ListableBeanFactory} interface,
 * managing existing bean instances rather than creating new ones based on bean
 * definitions, and not implementing any extended SPI interfaces (such as
 {@link org.springframework.beans.factory.config.ConfigurableBeanFactory}).
 *
 <p>For a full-fledged factory based on bean definitions, have a look
 * at {@link DefaultListableBeanFactory}.
 *
 @author Rod Johnson
 @author Juergen Hoeller
 @since 06.01.2003
 @see DefaultListableBeanFactory
 */
public class StaticListableBeanFactory implements ListableBeanFactory {

  /** Map from bean name to bean instance */
  private final Map<String, Object> beans = new HashMap<String, Object>();


  /**
   * Add a new singleton bean.
   * Will overwrite any existing instance for the given name.
   @param name the name of the bean
   @param bean the bean instance
   */
  public void addBean(String name, Object bean) {
    this.beans.put(name, bean);
  }


  //---------------------------------------------------------------------
  // Implementation of BeanFactory interface
  //---------------------------------------------------------------------

  public Object getBean(String namethrows BeansException {
    String beanName = BeanFactoryUtils.transformedBeanName(name);
    Object bean = this.beans.get(beanName);

    if (bean == null) {
      throw new NoSuchBeanDefinitionException(beanName,
          "Defined beans are [" + StringUtils.collectionToCommaDelimitedString(this.beans.keySet()) "]");
    }

    // Don't let calling code try to dereference the
    // bean factory if the bean isn't a factory
    if (BeanFactoryUtils.isFactoryDereference(name&& !(bean instanceof FactoryBean)) {
      throw new BeanIsNotAFactoryException(beanName, bean.getClass());
    }

    if (bean instanceof FactoryBean && !BeanFactoryUtils.isFactoryDereference(name)) {
      try {
        return ((FactoryBeanbean).getObject();
      }
      catch (Exception ex) {
        throw new BeanCreationException(beanName, "FactoryBean threw exception on object creation", ex);
      }
    }
    else {
      return bean;
    }
  }

  @SuppressWarnings("unchecked")
  public <T> T getBean(String name, Class<T> requiredTypethrows BeansException {
    Object bean = getBean(name);
    if (requiredType != null && !requiredType.isAssignableFrom(bean.getClass())) {
      throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
    }
    return (Tbean;
  }

  public <T> T getBean(Class<T> requiredTypethrows BeansException {
    String[] beanNames = getBeanNamesForType(requiredType);
    if (beanNames.length == 1) {
      return getBean(beanNames[0], requiredType);
    }
    else {
      throw new NoSuchBeanDefinitionException(requiredType, "expected single bean but found " + beanNames.length);
    }
  }

  public Object getBean(String name, Object... argsthrows BeansException {
    if (args != null) {
      throw new UnsupportedOperationException(
          "StaticListableBeanFactory does not support explicit bean creation arguments)");
    }
    return getBean(name);
  }

  public boolean containsBean(String name) {
    return this.beans.containsKey(name);
  }

  public boolean isSingleton(String namethrows NoSuchBeanDefinitionException {
    Object bean = getBean(name);
    // In case of FactoryBean, return singleton status of created object.
    return (bean instanceof FactoryBean && ((FactoryBeanbean).isSingleton());
  }

  public boolean isPrototype(String namethrows NoSuchBeanDefinitionException {
    Object bean = getBean(name);
    // In case of FactoryBean, return prototype status of created object.
    return ((bean instanceof SmartFactoryBean && ((SmartFactoryBeanbean).isPrototype()) ||
        (bean instanceof FactoryBean && !((FactoryBeanbean).isSingleton()));
  }

  public boolean isTypeMatch(String name, Class targetTypethrows NoSuchBeanDefinitionException {
    Class type = getType(name);
    return (targetType == null || (type != null && targetType.isAssignableFrom(type)));
  }

  public Class<?> getType(String namethrows NoSuchBeanDefinitionException {
    String beanName = BeanFactoryUtils.transformedBeanName(name);

    Object bean = this.beans.get(beanName);
    if (bean == null) {
      throw new NoSuchBeanDefinitionException(beanName,
          "Defined beans are [" + StringUtils.collectionToCommaDelimitedString(this.beans.keySet()) "]");
    }

    if (bean instanceof FactoryBean && !BeanFactoryUtils.isFactoryDereference(name)) {
      // If it's a FactoryBean, we want to look at what it creates, not the factory class.
      return ((FactoryBeanbean).getObjectType();
    }
    return bean.getClass();
  }

  public String[] getAliases(String name) {
    return new String[0];
  }


  //---------------------------------------------------------------------
  // Implementation of ListableBeanFactory interface
  //---------------------------------------------------------------------

  public boolean containsBeanDefinition(String name) {
    return this.beans.containsKey(name);
  }

  public int getBeanDefinitionCount() {
    return this.beans.size();
  }

  public String[] getBeanDefinitionNames() {
    return StringUtils.toStringArray(this.beans.keySet());
  }

  public String[] getBeanNamesForType(Class type) {
    return getBeanNamesForType(type, true, true);
  }

  public String[] getBeanNamesForType(Class type, boolean includeNonSingletons, boolean includeFactoryBeans) {
    boolean isFactoryType = (type != null && FactoryBean.class.isAssignableFrom(type));
    List<String> matches = new ArrayList<String>();
    for (String name : this.beans.keySet()) {
      Object beanInstance = this.beans.get(name);
      if (beanInstance instanceof FactoryBean && !isFactoryType) {
        if (includeFactoryBeans) {
          Class objectType = ((FactoryBeanbeanInstance).getObjectType();
          if (objectType != null && (type == null || type.isAssignableFrom(objectType))) {
            matches.add(name);
          }
        }
      }
      else {
        if (type == null || type.isInstance(beanInstance)) {
          matches.add(name);
        }
      }
    }
    return StringUtils.toStringArray(matches);
  }

  public <T> Map<String, T> getBeansOfType(Class<T> typethrows BeansException {
    return getBeansOfType(type, true, true);
  }

  @SuppressWarnings("unchecked")
  public <T> Map<String, T> getBeansOfType(Class<T> type, boolean includeNonSingletons, boolean includeFactoryBeans)
      throws BeansException {

    boolean isFactoryType = (type != null && FactoryBean.class.isAssignableFrom(type));
    Map<String, T> matches = new HashMap<String, T>();

    for (Map.Entry<String, Object> entry : beans.entrySet()) {
      String beanName = entry.getKey();
      Object beanInstance = entry.getValue();
      // Is bean a FactoryBean?
      if (beanInstance instanceof FactoryBean && !isFactoryType) {
        if (includeFactoryBeans) {
          // Match object created by FactoryBean.
          FactoryBean factory = (FactoryBeanbeanInstance;
          Class objectType = factory.getObjectType();
          if ((includeNonSingletons || factory.isSingleton()) &&
              objectType != null && (type == null || type.isAssignableFrom(objectType))) {
            matches.put(beanName, getBean(beanName, type));
          }
        }
      }
      else {
        if (type == null || type.isInstance(beanInstance)) {
          // If type to match is FactoryBean, return FactoryBean itself.
          // Else, return bean instance.
          if (isFactoryType) {
            beanName = FACTORY_BEAN_PREFIX + beanName;
          }
          matches.put(beanName, (TbeanInstance);
        }
      }
    }
    return matches;
  }

  public Map<String, Object> getBeansWithAnnotation(Class<? extends Annotation> annotationType)
      throws BeansException {

    Map<String, Object> results = new LinkedHashMap<String, Object>();
    for (String beanName : this.beans.keySet()) {
      if (findAnnotationOnBean(beanName, annotationType!= null) {
        results.put(beanName, getBean(beanName));
      }
    }
    return results;
  }

  public <A extends Annotation> A findAnnotationOnBean(String beanName, Class<A> annotationType) {
    return AnnotationUtils.findAnnotation(getType(beanName), annotationType);
  }

}