java spring ioc 实例_Spring 源码阅读(IOC容器)-bean的实例化以及注入
3.Bean的实例化以及注入过程分析
Bean的实例以及注入是在getBean时触发的,由于外部容器是与外部调用交互的桥梁,我们首先从外部容器入手,AbstractApplicationContext类实现了getBean方法,代码如下:
AbstractApplicationContext下的代码:
publicObject getBean(String name)throwsBeansException{
//Bean的获取外部容器交给了内部容器returngetBeanFactory().getBean(name);
}
内部容器由DefaultListableBeanFactory承当,但真实的getBean方法实现是由其父类AbstractBeanFactory实现的sssssssssssssssssssssss。
AbstractBeanFactory下的代码:
publicObject getBean(String name, Class requiredType, Object[] args)throwsBeansException{
returndoGetBean(name, requiredType, args,false);
}
3.1 doGetBean方法分析
protectedObject doGetBean(
finalString name,finalClass requiredType,finalObject[] args,booleantypeCheckOnly)throwsBeansException{
//把当前传入的beanName转化为标准beanName,例如把去除&引用(如果带&表示要获取FactoryBean)、alias转化为标准的beanName等finalString beanName=transformedBeanName(name);
Object bean=null;
//Eagerly check singleton cache for manually registered singletons.
//先从singleton对象缓存中查找,如果未找到接着到早期的singleton对象缓存查找,如果还是没有找到,接下来先获得beanName的singletonFactory,然后调用getObject获得对象。Object sharedInstance=getSingleton(beanName);
if(sharedInstance!=null){
/**//*获得给定bean实例的对象。这里的处理过程是这样的:
(1)如果sharedInstance不是FactoryBean实例,而是个普通bean,那么直接返回;
(2)如果sharedInstance是FactoryBean实例,且name要求返回FactoryBean实例,那么直接返回;
(3)否则从本地缓存获得合并BeanDefinition,如果本地缓存不存在合并BeanDefinition,则从本地缓存取得BeanDefinition,构造合并BeanDefinition,然后通过factoryBean获得bean*/ bean=getObjectForBeanInstance(sharedInstance, name, beanName,null);
}else{
//检查beanName所代表的bean是否处于正在创建阶段,主要是防止循环引用
if(isPrototypeCurrentlyInCreation(beanName)){
thrownewBeanCurrentlyInCreationException(beanName);
}
//首先检查父工厂是否存在,如果父工厂存在,且当前工厂不存在该beanName的BeanDefinition,那么当前工厂将委托父工厂完成getBean的任务. BeanFactory parentBeanFactory = getParentBeanFactory();
if(parentBeanFactory!=null&&!containsBeanDefinition(beanName)){
//转化为标准的beanNameString nameToLookup=originalBeanName(name);
if(args!=null){
//Delegation to parent with explicit args.returnparentBeanFactory.getBean(nameToLookup, args);
}
else{
//从父工厂获得bean,过程与本方法类似,相当于递归returnparentBeanFactory.getBean(nameToLookup, requiredType);
} }
if(!typeCheckOnly){
//是否要类型检查,如果不检查,直接把beanName标记为已创建markBeanAsCreated(beanName);
}
/**//*通过beanName,从mergedBeanDefinitions(针对mergeBeanDefinition的缓存)中取出合并的BeanDefinition,mergedBeanDefinition主要指两个BeanDefinition的合并,比如父子两个的合并*/finalRootBeanDefinition mbd=getMergedLocalBeanDefinition(beanName);
checkMergedBeanDefinition(mbd, beanName, args);
//Guarantee initialization of beans that the current bean depends on.String[] dependsOn=mbd.getDependsOn();
if(dependsOn!=null){//如果bean存在依赖bean,先构建依赖bean
for(inti=0; i{
String dependsOnBean=dependsOn[i];
getBean(dependsOnBean);
registerDependentBean(dependsOnBean, beanName);
} }
//Create bean instance.
if(mbd.isSingleton()){//
sharedInstance=getSingleton(beanName,newObjectFactory(){
publicObject getObject()throwsBeansException{
try{
/**//*所有生成Bean都有BeanWrapper封装,bean的生成采用策略模式,CglibSubclassingInstantiationStrategy实现是默认的策略
1.如果配置为工厂方法创建(配置了factory-mother的),由java反射Mehthod的invoke完成bean的创建。
2.如果采用构造子配置创建(配置了constructor-arg的),如果bean对应的类包含多个构造子,采用cglib动态字节码构造;如果只有唯一的构造子,那么采用java反射Constructor的newInstance方法
3.如果是普通bean配置,直接通过反射Class默认的Constructor,然后调用newInstance获得bean.*/returncreateBean(beanName, mbd, args);
}
catch(BeansException ex){
//Explicitly remove instance from singleton cache: It might have been put there.
//eagerly by the creation process, to allow for circular reference resolution.
//Also remove any beans that received a temporary reference to the bean.destroySingleton(beanName);
throwex;
} } });
/**//*由于当前构造的bean可能是FactoryBean,所以要通过以下方法获得真实需要的bean
(1)如果sharedInstance不是FactoryBean实例,而是个普通bean,那么直接返回;
(2)如果sharedInstance是FactoryBean实例,且name要求返回FactoryBean实例,那么直接返回;
(3)否则从本地缓存获得合并BeanDefinition,如果本地缓存不存在合并BeanDefinition,则从本地缓存取得BeanDefinition,构造合并BeanDefinition,然后通过factoryBean获得bean*/ bean=getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
}elseif(mbd.isPrototype()){
//It's a prototype -> create a new instance.Object prototypeInstance=null;
try{
beforePrototypeCreation(beanName);//维护prototypesCurrentlyInCreation,主要把beanName加入到其中prototypeInstance=createBean(beanName, mbd, args);
}
finally{
afterPrototypeCreation(beanName);
} bean=getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
}else{
String scopeName=mbd.getScope();
finalScope scope=(Scope)this.scopes.get(scopeName);
if(scope==null){
thrownewIllegalStateException("No Scope registered for scope '"+scopeName+"'");
}
try{
Object scopedInstance=scope.get(beanName,newObjectFactory(){
publicObject getObject()throwsBeansException{
beforePrototypeCreation(beanName);
try{
returncreateBean(beanName, mbd, args);
}
finally{
afterPrototypeCreation(beanName);
} } });
bean=getObjectForBeanInstance(scopedInstance, name, beanName, mbd);
}
catch(IllegalStateException ex){
thrownewBeanCreationException(beanName,
"Scope '"+scopeName+"' is not active for the current thread;"+"consider defining a scoped proxy for this bean if you intend to refer to it from a singleton",
ex);
} } }
//Check if required type matches the type of the actual bean instance.
if(requiredType!=null&&bean!=null&&!requiredType.isAssignableFrom(bean.getClass())){
thrownewBeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
}returnbean;
}
3.1.1 getObjectForBeanInstance方法分析
该方法从名称上就能看出,其实现的功能是从一个给定的bean实例,获取客户端调用所真实想要的bean。由于Spring能够管理各式的bean,当bean是FactoryBean实例的时候,客户端调用可以通过&+beanName的方式获得FactoryBean实例,而如果bean实例是普通的bean,那么该方法直接返回。
protectedObject getObjectForBeanInstance(
Object beanInstance, String name, String beanName, RootBeanDefinition mbd){
/**//*如果bean是工厂解除参照(即name带有&),且不是FactoryBean实例,那么抛出异常。这里的意思就是说如果name带有&,而传进来的又不是FactoryBean实例*/
if(BeanFactoryUtils.isFactoryDereference(name)&&!(beanInstanceinstanceofFactoryBean)){
thrownewBeanIsNotAFactoryException(transformedBeanName(name), beanInstance.getClass());
}
/**//*现在我们拥有了bean实例,它可能是一个普通bean或者一个FactoryBean.如果是一个工厂bean,我们使用它创建一个bean实例,除非调用者确实想要工厂的引用。
如果bean实例未实现FactoryBean,或者要求返回BeanFactory自身,那么直接返回。*/
if(!(beanInstanceinstanceofFactoryBean)||BeanFactoryUtils.isFactoryDereference(name)){
returnbeanInstance;
}
/**//*以下的代码意味着处理非普通bean,而是FactoryBean之类的*/ Object object=null;
if(mbd==null){//
/**//*这里主要是从一个FactoryBean到object映射的缓存中,取出由FactoryBean创建的object.这种映射关系是由factoryBeanObjectCache(映射关系:FactoryBean name --> object ,例如’myJndi’--->myJndiObject)字段维护,该字段位于FactoryBeanRegistrySupport中*/ object=getCachedObjectForFactoryBean(beanName);
}
if(object==null){
FactoryBean factory=(FactoryBean) beanInstance;
/**//*containsBeanDefinition方法是由默认内部工厂DefaultListableBeanFactory实现,因为BeanDefinition是缓存在它之下的*/
if(mbd==null&&containsBeanDefinition(beanName)){
/**//*从本地获得指定beanName的合并BeanDefinition,所谓的合并BeanDefinition就是指子类的BeanDefinition与父类BeanDefinition的合并。*/ mbd=getMergedLocalBeanDefinition(beanName);
}
/**//*判断beanDefinition是否是合成的,即是否与其他beanDefinition进行了合并*/booleansynthetic=(mbd!=null&&mbd.isSynthetic());
//从factoryBean获得beanobject=getObjectFromFactoryBean(factory, beanName,!synthetic);
}returnobject;
}
3.1.1.1 合并BeanDefinition获取的分析
在容器初始化阶段,对解释资源获得的BeanDefinition是由GenericBeanDefinition来定义,其只能刻画bean自身的一些特性。但bean与bean之间还存在着继承、方法覆盖、包含等一些复杂的关系。而这些关系的刻画是由RootBeanDefinition来负责的。RootBeanDefinition也就是我们下面所说的mergedBeanDefinition,mergedBeanDefinition是指对具有继承关系的bean所对应的BeanDefinition的合成,以达到完成表达一个bean的目的。
mergedBeanDefinition在获得后也将被缓存,其被缓存在mergedBeanDefinitions字段中。
protectedRootBeanDefinition getMergedLocalBeanDefinition(String beanName)throwsBeansException{
//Quick check on the concurrent map first, with minimal locking.RootBeanDefinition mbd=(RootBeanDefinition)this.mergedBeanDefinitions.get(beanName);
if(mbd!=null){
returnmbd;
}returngetMergedBeanDefinition(beanName, getBeanDefinition(beanName));
}
protectedRootBeanDefinition getMergedBeanDefinition(String beanName, BeanDefinition bd)
throwsBeanDefinitionStoreException{
returngetMergedBeanDefinition(beanName, bd,null);
}
/**//*以下是真正的合并BeanDefinition处理*/protectedRootBeanDefinition getMergedBeanDefinition(
String beanName, BeanDefinition bd, BeanDefinition containingBd)
throwsBeanDefinitionStoreException{
synchronized(this.mergedBeanDefinitions){//同步mergedBeanDefinitionsRootBeanDefinition mbd=null;
/**//*如果containingBd为空,先从缓存中查找,containingBd代表的包含bean的beanDefinition,那么什么是包含bean呢?比如B是A的内部类,那么A的实例就是B的实例的包含bean,这里的containingBd所代表的就是那个A。*/
if(containingBd==null){
//这里说明了合并BeanDefinition的处理顺序是先里后外。mbd=(RootBeanDefinition)this.mergedBeanDefinitions.get(beanName);
}
if(mbd==null){//如果缓存中不存在
if(bd.getParentName()==null){//且无父类
//如果当前BeanDefinition不存在parentName,说明该类不存在父类,直接把该类对应的BeanDefinition拷贝到合并BeanDefinitionmbd=newRootBeanDefinition(bd);
}
else{
BeanDefinition pbd=null;
try{
String parentBeanName=transformedBeanName(bd.getParentName());//对beanName进行标准化
if(!beanName.equals(parentBeanName)){//如果父类的名称与当前类的名称不相等
//获得父类的合并BeanDefinition
//这里说明了合并BeanDefinition的处理顺序是先父后子pbd=getMergedBeanDefinition(parentBeanName);
}
else{
if(getParentBeanFactory()instanceofConfigurableBeanFactory){//如果相等,则从父工厂中获取pbd=((ConfigurableBeanFactory) getParentBeanFactory()).getMergedBeanDefinition(parentBeanName);
}
else{
thrownewNoSuchBeanDefinitionException(bd.getParentName(),
"Parent name '"+bd.getParentName()+"' is equal to bean name '"+beanName+"': cannot be resolved without an AbstractBeanFactory parent");
} } }
catch(NoSuchBeanDefinitionException ex){
thrownewBeanDefinitionStoreException(bd.getResourceDescription(), beanName,
"Could not resolve parent bean definition '"+bd.getParentName()+"'", ex);
}//进行深度复制mbd=newRootBeanDefinition(pbd);
//用子的BeanDefinition覆盖父亲的mbd.overrideFrom(bd);
}
if(containingBd!=null&&!containingBd.isSingleton()&&mbd.isSingleton()){
//如果包含bean存在,且包含Bean不是单例的(也就是意味着是原型的),而当前bean却是单例的,那么当前bean的scope应与包含bean的一致。mbd.setScope(containingBd.getScope());
}
if(containingBd==null&&isCacheBeanMetadata()&&isBeanEligibleForMetadataCaching(beanName)){
this.mergedBeanDefinitions.put(beanName, mbd);//置入缓存} }
returnmbd;
} }
3.1.1.2 getObjectFromFactoryBean方法分析
//主要是从FactoryBean获得object
protectedObject getObjectFromFactoryBean(FactoryBean factory, String beanName,booleanshouldPostProcess){
if(factory.isSingleton()&&containsSingleton(beanName)){
//如果facoryBean是单例的话,需要维护factoryBean与object的关系。我们在这里整理一下几个缓存的关系
//<1>factoryBeanObjectCache里存在的一定是由FactoryBean所创建的bean,其对应关系为beanName-->bean 实例;
synchronized(getSingletonMutex()){
Object object=this.factoryBeanObjectCache.get(beanName);
if(object==null){
//这里通过调用factory.getObject()获取对象object=doGetObjectFromFactoryBean(factory, beanName, shouldPostProcess);
this.factoryBeanObjectCache.put(beanName, (object!=null?object : NULL_OBJECT));
}return(object!=NULL_OBJECT?object :null);
} }
else{
returndoGetObjectFromFactoryBean(factory, beanName, shouldPostProcess);
} }
3.1.2 getSingleton方法分析
publicObject getSingleton(String beanName, ObjectFactory singletonFactory){
synchronized(this.singletonObjects){
Object singletonObject=this.singletonObjects.get(beanName);
if(singletonObject==null){
if(this.singletonsCurrentlyInDestruction){
//如果当前工厂的单例缓存处于销毁状态,抛出异常thrownewBeanCreationNotAllowedException(beanName,
"Singleton bean creation not allowed while the singletons of this factory are in destruction"+"(Do not request a bean from a BeanFactory in a destroy method implementation!)");
}//把beanName加入到正在创建状态队列中,用以标识beanName正在创建中beforeSingletonCreation(beanName);
booleanrecordSuppressedExceptions=(this.suppressedExceptions==null);
if(recordSuppressedExceptions){
this.suppressedExceptions=newLinkedHashSet();
}
try{
//这里将要调用creatBean方法singletonObject=singletonFactory.getObject();
}
catch(BeanCreationException ex){
if(recordSuppressedExceptions){
for(Iterator it=this.suppressedExceptions.iterator(); it.hasNext();){
ex.addRelatedCause((Exception) it.next());
} }throwex;
}
finally{
if(recordSuppressedExceptions){
this.suppressedExceptions=null;
}//把beanName从正在创建队列中删除afterSingletonCreation(beanName); } addSingleton(beanName, singletonObject);
}return(singletonObject!=NULL_OBJECT?singletonObject :null);
} }
3.1.3 createBean方法分析
protectedObject createBean(finalString beanName,finalRootBeanDefinition mbd,finalObject[] args)
throwsBeanCreationException{
AccessControlContext acc=AccessController.getContext();
returnAccessController.doPrivileged(newPrivilegedAction(){
publicObject run(){
//检查BeanDefinition中的beanClass是否存在,如果不存在则通过工具类对className处理获得resolveBeanClass(mbd, beanName);
//Prepare method overrides.
//预处理方法的overload情况(对应lookup-method、replaced-method元素),如果方法在当前beanClass上实现的数量、在beanClass实现的接口声明的数量、在beanClass超类上声明或实现的数量相加的和等于1,那么将overload设为false。如果对lookup-method不熟悉请参看后面的lookup-method使用示例。
//
try{
mbd.prepareMethodOverrides();
}
catch(BeanDefinitionValidationException ex){
……
}
try{
//Give BeanPostProcessors a chance to return a proxy instead of the target bean instance.
//如果bean配置了PostProcessor,那么这里返回的是一个proxyObject bean=resolveBeforeInstantiation(beanName, mbd);
if(bean!=null){
returnbean;
} }
catch(Throwable ex){
thrownewBeanCreationException(mbd.getResourceDescription(), beanName,
"BeanPostProcessor before instantiation of bean failed", ex);
}//所有生成Bean都有BeanWrapper封装,bean的生成采用策略模式,CglibSubclassingInstantiationStrategy实现是默认的策略
//1.如果配置为工厂方法创建(配置了factory-mother的),由java反射Mehthod的invoke完成bean的创建。
//2.如果采用构造子配置创建(配置了constructor-arg的),如果bean对应的类包含多个构造子,采用cglib动态字节码构造;如果只有唯一的构造子,那么采用java反射Constructor的newInstance方法
//3.如果是普通bean配置,直接通过反射Class默认的Constructor,然后调用newInstance获得bean.Object beanInstance=doCreateBean(beanName, mbd, args);
if(logger.isDebugEnabled()){
logger.debug("Finished creating instance of bean '"+beanName+"'");
}returnbeanInstance;
} }, acc);
}
3.1.3.1 doCreateBean方法分析
protectedObject doCreateBean(finalString beanName,finalRootBeanDefinition mbd,finalObject[] args){
//Instantiate the bean.BeanWrapper instanceWrapper=null;
if(mbd.isSingleton()){//如果是singleton类型的,先删除factoryBeaninstanceWrapper=(BeanWrapper)this.factoryBeanInstanceCache.remove(beanName);
}
if(instanceWrapper==null){
//所有生成Bean都有BeanWrapper封装,bean的生成采用策略模式,CglibSubclassingInstantiationStrategy实现是默认的策略
//1.如果配置为工厂方法创建(配置了factory-mother的),由java反射Mehthod的invoke完成bean的创建。
//2.如果采用构造子配置创建(配置了constructor-arg的),如果bean对应的类包含多个构造子,采用cglib动态字节码构造;如果只有唯一的构造子,那么采用java反射Constructor的newInstance方法
//3.如果是普通bean配置,直接通过反射Class默认的Constructor,然后调用newInstance获得bean.instanceWrapper=createBeanInstance(beanName, mbd, args);
}finalObject bean=(instanceWrapper!=null?instanceWrapper.getWrappedInstance() :null);
Class beanType=(instanceWrapper!=null?instanceWrapper.getWrappedClass() :null);
//Allow post-processors to modify the merged bean definition.
synchronized(mbd.postProcessingLock){
if(!mbd.postProcessed){
applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);
mbd.postProcessed=true;
} }
booleanearlySingletonExposure=(mbd.isSingleton()&&this.allowCircularReferences&& isSingletonCurrentlyInCreation(beanName));
if(earlySingletonExposure){
if(logger.isDebugEnabled()){
logger.debug("Eagerly caching bean '"+beanName+"' to allow for resolving potential circular references");
}
addSingletonFactory(beanName,newObjectFactory(){
publicObject getObject()throwsBeansException{
returngetEarlyBeanReference(beanName, mbd, bean);
} });
}
//Initialize the bean instance.Object exposedObject=bean;
try{
populateBean(beanName, mbd, instanceWrapper);
exposedObject=initializeBean(beanName, exposedObject, mbd);
}
catch(Throwable ex){
if(exinstanceofBeanCreationException&&beanName.equals(((BeanCreationException) ex).getBeanName())){
throw(BeanCreationException) ex;
}
else{
thrownewBeanCreationException(mbd.getResourceDescription(), beanName,"Initialization of bean failed", ex);
} }
if(earlySingletonExposure){
Object earlySingletonReference=getSingleton(beanName,false);
if(earlySingletonReference!=null){
if(exposedObject==bean){
exposedObject=earlySingletonReference;
}
elseif(!this.allowRawInjectionDespiteWrapping&&hasDependentBean(beanName)){
String[] dependentBeans=getDependentBeans(beanName);
Set actualDependentBeans=newLinkedHashSet(dependentBeans.length);
for(inti=0; i{
String dependentBean=dependentBeans[i];
if(!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)){
actualDependentBeans.add(dependentBean);
} }
if(!actualDependentBeans.isEmpty()){
thrownewBeanCurrentlyInCreationException(beanName,
"Bean with name '"+beanName+"' has been injected into other beans ["+ StringUtils.collectionToCommaDelimitedString(actualDependentBeans)+"] in its raw version as part of a circular reference, but has eventually been"+"wrapped. This means that said other beans do not use the final version of the"+"bean. This is often the result of over-eager type matching - consider using"+"'getBeanNamesOfType' with the 'allowEagerInit' flag turned off, for example.");
} } } }
//Register bean as disposable.registerDisposableBeanIfNecessary(beanName, bean, mbd);
returnexposedObject;
}
posted on 2012-01-30 16:04 zhangxl 阅读(3197) 评论(0) 编辑 收藏 所属分类: Spring
java spring ioc 实例_Spring 源码阅读(IOC容器)-bean的实例化以及注入相关推荐
- Spring Boot 2.0系列文章(四):Spring Boot 2.0 源码阅读环境搭建
前提 前几天面试的时候,被问过 Spring Boot 的自动配置源码怎么实现的,没看过源码的我只能投降��了. 这不,赶紧来补补了,所以才有了这篇文章的出现,Spring Boot 2. 0 源码阅 ...
- mybatis源码阅读(四):mapper(dao)实例化
转载自 mybatis源码阅读(四):mapper(dao)实例化 在开始分析之前,先来了解一下这个模块中的核心组件之间的关系,如图: 1.MapperRegistry&MapperPro ...
- spring依赖注入_Spring源码阅读:Spring依赖注入容器
依赖注入 依赖注入是Spring框架最核心的能力,Spring框架提供的AOP,WebMVC等其它功能都是以依赖注入容器作为基础构建的,Spring依赖注入容器类似于一个用于组装对象的框架内核,任何应 ...
- batch spring 重复执行_Spring源码高级笔记之——Spring AOP应用
Spring AOP应用 AOP本质:在不改变原有业务逻辑的情况下增强横切逻辑,横切逻辑代码往往是权限校验代码.日志代码.事务控制代码.性能监控代码. 第1节AOP相关术语 1.1业务主线 在讲解AO ...
- java常用类介绍及源码阅读(ArrayList)
java.util 类 ArrayList<E> 继承关系: java.lang.Objectjava.util.AbstractCollection<E>java.util ...
- Spring Boot Transactional注解源码阅读笔记(二)
在源码笔记(一)中,我们留下了几个问题: Spring Boot是怎么扫描到我们的bean里面有 Transactional 这个注解,并且把 InfrastructureAdvisorAutoP ...
- vc+ mfc 方法怎么被调用_Spring源码阅读(二)我的方法是怎么被自动调用的
曾几何时,我们在写Java代码时,写好一个类,然后去创建这个类的对象,再调用它的方法,项目可能变得有些冗余,但却其乐融融,因为我知道程序每一步干了什么, 但现在,时代变了,你发现我往往只需要实现一个接 ...
- java jdk 类加载机制_JDK源码阅读之类加载
java类加载 类的生命周期(类加载过程) LLIUU+VPR 加载(Loading) 链接(Linking) 验证(Verification) 准备(Preparation) 解析(Resoluti ...
- 【java】java 定时任务线程池 ScheduledThreadPoolExecutor 源码阅读
文章目录 1.概述 1.1 问题 1.2 简介 1.2 数据结构 2.源码解析 2.1 内部类ScheduledFutureTask 2.1.1 compareTo 2.1.2 核心方法run() 2 ...
最新文章
- 关于在64位系统上,使用APlayer,运行报错“找不到指定的模块”
- 蒙特卡洛树搜索的主要流程有哪些_海运拼箱操作流程主要有哪些
- Thymelaf中使用select进行消息的回显
- C++中了类继承和调用父类的构造函数方法
- 数据下载工作笔记三:脚本
- c++ 线程什么时候run_阿里后端Java面试题:启动线程是start()还是run()?为什么?...
- matlab求微分数值,用MATLAB语言求微积分方程的数值解.(xd^2y)/dx^2-5dy/dx+y=0y(0)=0y'(0)=0...
- 微信H5支付(基于Java实现微信H5支付)
- 以太坊使用puppeth工具
- ES6-18:class类及其继承
- linux 64位module内联汇编,@yuanbor: Linux内联汇编总结
- mes系统和plc通讯案例_「MES系统 | 应用案例」奥松电子云MES系统项目启动大会顺利召开...
- MATLAB——imhist函数
- 酷狗.kgtemp文件加密算法逆向
- 单片机课程设计题目及要求——电风扇模拟控制系统(仿真图加代码加原理图都有)
- 干细胞科学家李陶博士入股机器人365
- MySQL数据库插入中文时出现Incorrect string value: '\xE6\x97\xB7\xE5\x85\xA8' for column 'sz_name' at row 1...
- 海底捞成功的全套培训体系(收藏)
- excel分列---多页批量操作--vba宏实现
- java入门基础掌握单词汇总
热门文章
- 数据结构C++实现(顺序栈) 青岛大学王卓老师
- 金融计算器 npv_CFA考试用什么计算器?怎么操作?
- 阿piu传-文档批量上传客户端豆丁版使用帮助
- Fama-Macbeth回归:EAP.fama_macbeth
- 如何快速查询极兔快递并设置参数?
- 重新认识java(十一)---- java中的数组
- facetune2 android,facetune2
- DEDEcms终极SEO优化教程
- Java 报错 restartedMain] o.s.boot.SpringApplication : Application run failed
- 杉岩数据:解读软件定义存储(SDS)及应用