第三章 spring-bean之AbstractBeanFactory(5)

前言

AbstractBeanFactor實現了ConfigurableBeanFactory接口。一棵樹有不只有主幹,樹幹,還有花花葉葉。其餘接口與實現是BeanFactory的主幹與樹幹的話,那ConfigurableBeanFactory負責管理BeanFactory的花花葉葉。java

void setParentBeanFactory(BeanFactory parentBeanFactory) throws IllegalStateException;

	void setBeanClassLoader(ClassLoader beanClassLoader);

	ClassLoader getBeanClassLoader();

	void setTempClassLoader(ClassLoader tempClassLoader);

	ClassLoader getTempClassLoader();

	void setCacheBeanMetadata(boolean cacheBeanMetadata);

	boolean isCacheBeanMetadata();

	void setBeanExpressionResolver(BeanExpressionResolver resolver);

	BeanExpressionResolver getBeanExpressionResolver();

	void setConversionService(ConversionService conversionService);

	ConversionService getConversionService();

	void addPropertyEditorRegistrar(PropertyEditorRegistrar registrar);

	void registerCustomEditor(Class<?> requiredType, Class<? extends PropertyEditor> propertyEditorClass);

	void copyRegisteredEditorsTo(PropertyEditorRegistry registry);

	void setTypeConverter(TypeConverter typeConverter);

	TypeConverter getTypeConverter();

	void addEmbeddedValueResolver(StringValueResolver valueResolver);

	boolean hasEmbeddedValueResolver();

	String resolveEmbeddedValue(String value);

	void addBeanPostProcessor(BeanPostProcessor beanPostProcessor);

	int getBeanPostProcessorCount();

	void registerScope(String scopeName, Scope scope);

	String[] getRegisteredScopeNames();

	Scope getRegisteredScope(String scopeName);

	AccessControlContext getAccessControlContext();
void copyConfigurationFrom(ConfigurableBeanFactory otherFactory);

	void registerAlias(String beanName, String alias) throws BeanDefinitionStoreException;

	void resolveAliases(StringValueResolver valueResolver);

	BeanDefinition getMergedBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;

	boolean isFactoryBean(String name) throws NoSuchBeanDefinitionException;

	void setCurrentlyInCreation(String beanName, boolean inCreation);

	boolean isCurrentlyInCreation(String beanName);

	void registerDependentBean(String beanName, String dependentBeanName);

	String[] getDependentBeans(String beanName);

	String[] getDependenciesForBean(String beanName);
關於parentBeanFactory 變量相關信息
void setParentBeanFactory(BeanFactory parentBeanFactory) throws IllegalStateException;
private BeanFactory parentBeanFactory;

調用方spring

  • org.springframework.beans.factory.support.AbstractBeanFactory
    • containsBean(String)
    • doGetBean(String, Class<T>, Object[], boolean)
    • getAliases(String)
    • getMergedBeanDefinition(String, BeanDefinition, BeanDefinition)
    • getMergedBeanDefinition(String)
    • getType(String)
    • isFactoryBean(String)
    • isPrototype(String)
    • isSingleton(String)
    • isTypeMatch(String, ResolvableType)
  • org.springframework.beans.factory.support.DefaultListableBeanFactory
    • checkBeanNotOfRequiredType(Class<?>, DependencyDescriptor)
    • getBean(Class<T>, Object...)
    • isAutowireCandidate(String, DependencyDescriptor, AutowireCandidateResolver)
    • isPrimary(String, Object)
    • resolveNamedBean(Class<T>)
    • toString()

從被調用的方法名能夠方法,parentBeanFactory屬性主要是在獲得或建立bean的時候,須要進行一樣的操做。好比getBean方法,在當前beanFactory沒有獲得bean,就會去parentBeanFactory執行getBean(){去操做一樣的方法}。express

public <T> T getBean(Class<T> requiredType, Object... args) throws BeansException {
		NamedBeanHolder<T> namedBean = resolveNamedBean(requiredType, args);
		if (namedBean != null) {
			return namedBean.getBeanInstance();
		}
		BeanFactory parent = getParentBeanFactory();
		if (parent != null) {
			return parent.getBean(requiredType, args);
		}
		throw new NoSuchBeanDefinitionException(requiredType);
	}

解讀bootstrap

parentBeanFactory的設計主要是解決在複雜環境的下,當前容器有parent容器的狀況。可是這種設計在當前互聯網與分佈式應用中沒法應用到,因此是一個。菜鳥啊我的,感受這個功能已經十分過期了。tomcat

關於ClassLoader相關變量

void setBeanClassLoader(ClassLoader beanClassLoader);

private ClassLoader beanClassLoader = ClassUtils.getDefaultClassLoader();

public static ClassLoader getDefaultClassLoader() {
		ClassLoader cl = null;
		try {
			cl = Thread.currentThread().getContextClassLoader();
		}
		catch (Throwable ex) {
			// Cannot access thread context ClassLoader - falling back...
		}
		if (cl == null) {
			// No thread context class loader -> use class loader of this class.
			cl = ClassUtils.class.getClassLoader();
			if (cl == null) {
				// getClassLoader() returning null indicates the bootstrap ClassLoader
				try {
					cl = ClassLoader.getSystemClassLoader();
				}
				catch (Throwable ex) {
					// Cannot access system ClassLoader - oh well, maybe the caller can live with null...
				}
			}
		}
		return cl;
	}

org.springframework.aop.aspectj org.springframework.aop.aspectj.annotation org.springframework.aop.config org.springframework.aop.scope org.springframework.beans.factory.support org.springframework.context.annotation org.springframework.context.event org.springframework.context.expression org.springframework.context.support org.springframework.remoting.jaxws併發

解讀分佈式

  1. 可能會得到三種不一樣的類加載器
  2. 在不一樣應用中,spring的類加載器,可能不其餘的類加載器 不同。好比在tomcat容器中,得到的類加載器與spring默認類加載器不同
  3. 不建議修改beanClassLoader類加載器的值

conversionService 變量

private ConversionService conversionService;

public void setConversionService(ConversionService conversionService) {
	this.conversionService = conversionService;
}

@Override
public ConversionService getConversionService() {
	return this.conversionService;
}
}
String CONVERSION_SERVICE_BEAN_NAME = "conversionService";
protected void finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory) {
	if (beanFactory.containsBean(CONVERSION_SERVICE_BEAN_NAME) && beanFactory.isTypeMatch(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class)) {
		beanFactory.setConversionService( beanFactory.getBean(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class));
	}
	.....
}

解讀ide

  1. 若是沒有配置 ConversionService 子類,默認在容器裏面是沒有ConversionService實現類的getConversionService返回null
  2. 關於 ConversionService 的詳解請求第二章 二節spring-core之converter深刻解讀

關於BeanPostProcessor相關的變量

private final List<BeanPostProcessor> beanPostProcessors = new ArrayList<BeanPostProcessor>();

/** Indicates whether any InstantiationAwareBeanPostProcessors have been registered */
private boolean hasInstantiationAwareBeanPostProcessors;

/** Indicates whether any DestructionAwareBeanPostProcessors have been registered */
private boolean hasDestructionAwareBeanPostProcessors;

public void addBeanPostProcessor(BeanPostProcessor beanPostProcessor) {
		Assert.notNull(beanPostProcessor, "BeanPostProcessor must not be null");
		this.beanPostProcessors.remove(beanPostProcessor);
		this.beanPostProcessors.add(beanPostProcessor);
		if (beanPostProcessor instanceof InstantiationAwareBeanPostProcessor) {
			this.hasInstantiationAwareBeanPostProcessors = true;
		}
		if (beanPostProcessor instanceof DestructionAwareBeanPostProcessor) {
			this.hasDestructionAwareBeanPostProcessors = true;
		}
}

public int getBeanPostProcessorCount() {
	return this.beanPostProcessors.size();
}

public List<BeanPostProcessor> getBeanPostProcessors() {
	return this.beanPostProcessors;
}

protected boolean hasInstantiationAwareBeanPostProcessors() {
	return this.hasInstantiationAwareBeanPostProcessors;
}

protected boolean hasDestructionAwareBeanPostProcessors() {
	return this.hasDestructionAwareBeanPostProcessors;
}

protected Object getEarlyBeanReference(String beanName, RootBeanDefinition mbd, Object bean) {
	Object exposedObject = bean;
	if (bean != null && !mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {
		for (BeanPostProcessor bp : getBeanPostProcessors()) {
			if (bp instanceof SmartInstantiationAwareBeanPostProcessor) {
				SmartInstantiationAwareBeanPostProcessor ibp = (SmartInstantiationAwareBeanPostProcessor) bp;
				exposedObject = ibp.getEarlyBeanReference(exposedObject, beanName);
				if (exposedObject == null) {
					return null;
				}
			}
		}
	}
	return exposedObject;
}

解讀ui

  1. beanPostProcessors變量保存了全部BeanPostProcessor實現類,
  2. hasDestructionAwareBeanPostProcessors與hasDestructionAwareBeanPostProcessors的做用是避免進行無效的循環操做

RootBeanDefinition相關變量

private boolean cacheBeanMetadata = true;

	protected RootBeanDefinition getMergedBeanDefinition( String beanName, BeanDefinition bd, BeanDefinition containingBd)throws BeanDefinitionStoreException {
		if (containingBd == null && isCacheBeanMetadata()) {
			this.mergedBeanDefinitions.put(beanName, mbd);
		}
	}


	private final Map<String, RootBeanDefinition> mergedBeanDefinitions = new ConcurrentHashMap<String, RootBeanDefinition>(256);

	private final Set<String> alreadyCreated = Collections.newSetFromMap(new ConcurrentHashMap<String, Boolean>(256));
	protected void markBeanAsCreated(String beanName) {
		if (!this.alreadyCreated.contains(beanName)) {
			synchronized (this.mergedBeanDefinitions) {
				if (!this.alreadyCreated.contains(beanName)) {
					clearMergedBeanDefinition(beanName);
					this.alreadyCreated.add(beanName);
				}
			}
		}
	} 
	protected <T> T doGetBean(final String name, final Class<T> requiredType, final Object[] args, boolean typeCheckOnly) throws BeansException {
		if (!typeCheckOnly) {
				markBeanAsCreated(beanName);
			}
		try {
			final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
			checkMergedBeanDefinition(mbd, beanName, args);
		}
	}

解讀this

  1. 不建議修改cacheBeanMetadata的值
  2. mergedBeanDefinitions保存了RootBeanDefinition對象,RootBeanDefinition對象纔會生成一個對象。其餘的beanDefinition實現不會。奇怪的設計。
  3. alreadyCreated控制RootBeanDefinition併發建立問題

總結

本節的內容感受比較慘白無力,有些細節點仍是比較重要的。好比RootBeanDefinition,ConversionService,ClassLoad等。其餘方面鳥菜啊,感受就是無用就不解讀了。

相關文章
相關標籤/搜索