/* * Copyright 2002-2008 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 org.springframework.beans.MutablePropertyValues; import org.springframework.beans.factory.config.BeanDefinition; import org.springframework.beans.factory.config.ConstructorArgumentValues; import org.springframework.util.ObjectUtils; /** * Bean definition for beans which inherit settings from their parent. * Child bean definitions have a fixed dependency on a parent bean definition. * * <p>A child bean definition will inherit constructor argument values, * property values and method overrides from the parent, with the option * to add new values. If init method, destroy method and/or static factory * method are specified, they will override the corresponding parent settings. * The remaining settings will <i>always</i> be taken from the child definition: * depends on, autowire mode, dependency check, singleton, lazy init. * * <p><b>NOTE:</b> Since Spring 2.5, the preferred way to register bean * definitions programmatically is the {@link GenericBeanDefinition} class, * which allows to dynamically define parent dependencies through the * {@link GenericBeanDefinition#setParentName} method. This effectively * supersedes the ChildBeanDefinition class for most use cases. * * @author Rod Johnson * @author Juergen Hoeller * @see GenericBeanDefinition * @see RootBeanDefinition */ public class ChildBeanDefinition extends AbstractBeanDefinition { private String parentName; /** * Create a new ChildBeanDefinition for the given parent, to be * configured through its bean properties and configuration methods. * @param parentName the name of the parent bean * @see #setBeanClass * @see #setBeanClassName * @see #setScope * @see #setAutowireMode * @see #setDependencyCheck * @see #setConstructorArgumentValues * @see #setPropertyValues */ public ChildBeanDefinition(String parentName) { super(); this.parentName = parentName; } /** * Create a new ChildBeanDefinition for the given parent. * @param parentName the name of the parent bean * @param pvs the additional property values of the child */ public ChildBeanDefinition(String parentName, MutablePropertyValues pvs) { super(null, pvs); this.parentName = parentName; } /** * Create a new ChildBeanDefinition for the given parent. * @param parentName the name of the parent bean * @param cargs the constructor argument values to apply * @param pvs the additional property values of the child */ public ChildBeanDefinition( String parentName, ConstructorArgumentValues cargs, MutablePropertyValues pvs) { super(cargs, pvs); this.parentName = parentName; } /** * Create a new ChildBeanDefinition for the given parent, * providing constructor arguments and property values. * @param parentName the name of the parent bean * @param beanClass the class of the bean to instantiate * @param cargs the constructor argument values to apply * @param pvs the property values to apply */ public ChildBeanDefinition( String parentName, Class beanClass, ConstructorArgumentValues cargs, MutablePropertyValues pvs) { super(cargs, pvs); this.parentName = parentName; setBeanClass(beanClass); } /** * Create a new ChildBeanDefinition for the given parent, * providing constructor arguments and property values. * Takes a bean class name to avoid eager loading of the bean class. * @param parentName the name of the parent bean * @param beanClassName the name of the class to instantiate * @param cargs the constructor argument values to apply * @param pvs the property values to apply */ public ChildBeanDefinition( String parentName, String beanClassName, ConstructorArgumentValues cargs, MutablePropertyValues pvs) { super(cargs, pvs); this.parentName = parentName; setBeanClassName(beanClassName); } /** * Create a new ChildBeanDefinition as deep copy of the given * bean definition. * @param original the original bean definition to copy from */ public ChildBeanDefinition(ChildBeanDefinition original) { super((BeanDefinition) original); } public void setParentName(String parentName) { this.parentName = parentName; } public String getParentName() { return this.parentName; } public void validate() throws BeanDefinitionValidationException { super.validate(); if (this.parentName == null) { throw new BeanDefinitionValidationException("'parentName' must be set in ChildBeanDefinition"); } } public AbstractBeanDefinition cloneBeanDefinition() { return new ChildBeanDefinition(this); } public boolean equals(Object other) { if (this == other) { return true; } if (!(other instanceof ChildBeanDefinition)) { return false; } ChildBeanDefinition that = (ChildBeanDefinition) other; return (ObjectUtils.nullSafeEquals(this.parentName, that.parentName) && super.equals(other)); } public int hashCode() { return ObjectUtils.nullSafeHashCode(this.parentName) * 29 + super.hashCode(); } public String toString() { StringBuffer sb = new StringBuffer("Child bean with parent '"); sb.append(this.parentName).append("': ").append(super.toString()); return sb.toString(); } }