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的实例化以及注入相关推荐

  1. Spring Boot 2.0系列文章(四):Spring Boot 2.0 源码阅读环境搭建

    前提 前几天面试的时候,被问过 Spring Boot 的自动配置源码怎么实现的,没看过源码的我只能投降��了. 这不,赶紧来补补了,所以才有了这篇文章的出现,Spring Boot 2. 0 源码阅 ...

  2. mybatis源码阅读(四):mapper(dao)实例化

    转载自   mybatis源码阅读(四):mapper(dao)实例化 在开始分析之前,先来了解一下这个模块中的核心组件之间的关系,如图: 1.MapperRegistry&MapperPro ...

  3. spring依赖注入_Spring源码阅读:Spring依赖注入容器

    依赖注入 依赖注入是Spring框架最核心的能力,Spring框架提供的AOP,WebMVC等其它功能都是以依赖注入容器作为基础构建的,Spring依赖注入容器类似于一个用于组装对象的框架内核,任何应 ...

  4. batch spring 重复执行_Spring源码高级笔记之——Spring AOP应用

    Spring AOP应用 AOP本质:在不改变原有业务逻辑的情况下增强横切逻辑,横切逻辑代码往往是权限校验代码.日志代码.事务控制代码.性能监控代码. 第1节AOP相关术语 1.1业务主线 在讲解AO ...

  5. java常用类介绍及源码阅读(ArrayList)

    java.util  类 ArrayList<E> 继承关系: java.lang.Objectjava.util.AbstractCollection<E>java.util ...

  6. Spring Boot Transactional注解源码阅读笔记(二)

      在源码笔记(一)中,我们留下了几个问题: Spring Boot是怎么扫描到我们的bean里面有 Transactional 这个注解,并且把 InfrastructureAdvisorAutoP ...

  7. vc+ mfc 方法怎么被调用_Spring源码阅读(二)我的方法是怎么被自动调用的

    曾几何时,我们在写Java代码时,写好一个类,然后去创建这个类的对象,再调用它的方法,项目可能变得有些冗余,但却其乐融融,因为我知道程序每一步干了什么, 但现在,时代变了,你发现我往往只需要实现一个接 ...

  8. java jdk 类加载机制_JDK源码阅读之类加载

    java类加载 类的生命周期(类加载过程) LLIUU+VPR 加载(Loading) 链接(Linking) 验证(Verification) 准备(Preparation) 解析(Resoluti ...

  9. 【java】java 定时任务线程池 ScheduledThreadPoolExecutor 源码阅读

    文章目录 1.概述 1.1 问题 1.2 简介 1.2 数据结构 2.源码解析 2.1 内部类ScheduledFutureTask 2.1.1 compareTo 2.1.2 核心方法run() 2 ...

最新文章

  1. 关于在64位系统上,使用APlayer,运行报错“找不到指定的模块”
  2. 蒙特卡洛树搜索的主要流程有哪些_海运拼箱操作流程主要有哪些
  3. Thymelaf中使用select进行消息的回显
  4. C++中了类继承和调用父类的构造函数方法
  5. 数据下载工作笔记三:脚本
  6. c++ 线程什么时候run_阿里后端Java面试题:启动线程是start()还是run()?为什么?...
  7. matlab求微分数值,用MATLAB语言求微积分方程的数值解.(xd^2y)/dx^2-5dy/dx+y=0y(0)=0y'(0)=0...
  8. 微信H5支付(基于Java实现微信H5支付)
  9. 以太坊使用puppeth工具
  10. ES6-18:class类及其继承
  11. linux 64位module内联汇编,@yuanbor: Linux内联汇编总结
  12. mes系统和plc通讯案例_「MES系统 | 应用案例」奥松电子云MES系统项目启动大会顺利召开...
  13. MATLAB——imhist函数
  14. 酷狗.kgtemp文件加密算法逆向
  15. 单片机课程设计题目及要求——电风扇模拟控制系统(仿真图加代码加原理图都有)
  16. 干细胞科学家李陶博士入股机器人365
  17. MySQL数据库插入中文时出现Incorrect string value: '\xE6\x97\xB7\xE5\x85\xA8' for column 'sz_name' at row 1...
  18. 海底捞成功的全套培训体系(收藏)
  19. excel分列---多页批量操作--vba宏实现
  20. java入门基础掌握单词汇总

热门文章

  1. 数据结构C++实现(顺序栈) 青岛大学王卓老师
  2. 金融计算器 npv_CFA考试用什么计算器?怎么操作?
  3. 阿piu传-文档批量上传客户端豆丁版使用帮助
  4. Fama-Macbeth回归:EAP.fama_macbeth
  5. 如何快速查询极兔快递并设置参数?
  6. 重新认识java(十一)---- java中的数组
  7. facetune2 android,facetune2
  8. DEDEcms终极SEO优化教程
  9. Java 报错 restartedMain] o.s.boot.SpringApplication : Application run failed
  10. 杉岩数据:解读软件定义存储(SDS)及应用