1.Spring5源码分析之@Configuration注解的源码详解。希望读者能够耐着性子看完
2.《Java面向对象编程》导读-在Java类中使用自定义注解
3.java中注解是中的注解什么意思?
4.注解 是什么意思?
5.从源码层面带你实现一个自动注入注解
6.注释和注解的区别
Spring5源码分析之@Configuration注解的详解。希望读者能够耐着性子看完
Spring5源码中@Configuration注解详解,源码让你理解无需XML的中的注解Bean创建。在Spring 3.0以后,源码@Configuration注解的中的注解窗口绑定源码出现,允许开发者在运行时动态创建和初始化Bean,源码无需依赖XML配置。中的注解它实际上标记了@Component元注解,源码被@ComponentScan扫描并纳入Spring容器管理。中的注解
使用@Configuration时,源码Bean的中的注解默认名称与方法名称相同,可通过name属性指定。源码它不仅自身可以作为受管理的中的注解组件,还能通过@Autowired和@Inject注解注入其他Bean。源码例如,修改Demo,配置类可以作为服务组件被自动扫描。
@Configuration不仅支持@ComponentScan,还能与@Controller、@Service、@Repository等注解配合,幽冥传奇纯源码这些注解本质上都有@Component,适合不同场景的管理。此外,@Configuration可以同@Import和@Profile注解组合,实现更灵活的配置导入和环境条件控制。
在配置类内部嵌套@Configuration,可以利用静态内部类简化@Import的使用。配置类的初始化可以通过@Lazy注解延迟,提供更细致的控制。配置类解析涉及@ConfigurationClassPostProcessor处理器,处理@Configuration类的@Bean、@ComponentScan和环境相关注解。
最后,@Configuration类的Bean定义信息由ConfigurationClassBeanDefinitionReader处理并注册到Spring容器,整个过程包括解析@Configuration类、扫描相关注解和Bean定义的加载。
理解@Configuration的解析流程,能帮助你更高效地利用Spring的动态配置能力。如果你对文章内容有所收获,别忘了分享和关注我们的更多内容。
《Java面向对象编程》导读-在Java类中使用自定义注解
在这篇文章中,redis源码调试event将介绍如何在Java类中使用自定义注解。将定义三个注解:@Programmer、@ConstructorNote和@Common。接下来,这些注解可以在其他类中使用,示例代码如下:
例程1:Person类
在Person类的源代码中,使用注解进行标记。在类前添加@Programmer注解,在构造方法前添加@ConstructorNote注解,在成员变量、成员方法和参数前添加@Common注解。
注解成员的赋值有几种方式:默认值、以"成员名=成员值"形式赋值,或当注解仅有一个成员且名为"value"时,以成员值直接赋值。
编译Person类时,由于自定义注解的有效范围为RetentionPolicy.RUNTIME,编译器会将注解编译至类文件中。
在定义@Programmer注解时引用了JDK类库的内置@Documented注解,因此在使用JDK的javadoc命令生成JavaDoc文档时,包含@Programmer注解信息。ccl副图源码
以上内容参考了孙卫琴的经典Java书籍《Java面向对象编程》。希望这篇介绍能帮助你更好地理解和使用自定义注解。
java中注解是什么意思?
java中注解是什么意思
Java注解是一种元数据,它们为源代码添加了信息,而不会修改实际代码。注解不会影响程序的运行,但可以用来提供关于代码的附加信息,例如执行某些任务或配置某些值。它们可以像Javadoc一样被编译器和其他工具使用,用于生成文档、执行静态分析等操作。注解是Java程序中非常有用的一种特性,它们可以用于许多不同的任务,包括测试、配置和性能优化等。
Java注解可以应用于各种不同的元素,包括类、方法、变量、参数等等。使用注解可以为这些元素提供额外的怎么解压源码包信息,或指定某些行为。例如,注解可以指定方法的可见性、参数的名称或类型,甚至可以标记需要特殊处理的代码块。Java中的许多库和框架都使用了注解来实现某些功能,例如JUnit测试框架就使用注解标记测试方法和断言条件。通过使用注解,开发人员可以使代码更加清晰和易于维护。
Java注解的本质是一种元数据,它们不会修改程序代码,而是提供了一些额外的信息。注解可以在源代码级别访问,可以帮助编写更好的文档和工具,同时还可以增强代码的可读性和可维护性。注解并不是Java中的一个新特性,它们从Java5版本开始引入,并经过多年的发展变得越来越强大和灵活。注解在Java程序中应用广泛,是提高代码质量和开发效率的重要工具之一。
注解 是什么意思?
注解(Annotation)是Java语言中的一种元数据(meta-data)机制,用于描述程序代码中各个元素的信息(如类、方法、变量等),可以给目标代码添加额外信息,起到解析和说明的作用。注解不会改变程序的执行逻辑,但在程序开发和调试中,注解能够对代码做出更详细的描述,提高程序的可读性、可维护性和可扩展性。
a. 使程序更加易读易懂。通过注解,我们可以为代码添加更多的语义信息,让程序更加清晰易懂。
b. 使程序更加灵活可扩展。注解提供了一种在不修改源代码的情况下为程序添加信息的方式,使程序更加灵活、可扩展,可以快速应对功能扩展或版本升级的需求。
c. 减少程序员的工作量。在Java语言中,注解提供了很多标准的元数据机制,可以快速帮助程序员完成一些繁重、重复的工作。
注解在Java中的应用非常广泛,常见的有以下几个方面:
a. 标记和配置元素。例如,可以通过 @Override 注解来标记方法覆写,通过 @Deprecated 注解来标记过时的代码。
b. 提供编译时检查。例如,可以通过 @SuppressWarnings 注解来关闭编译时警告信息。
c. 生成文档。例如,可以通过 @param 、@return 等注解来生成javadoc文档。
d. 做一些额外的处理。例如,可以通过自定义注解来完成一些特殊的处理,比如开发框架中的事务处理、权限校验等。
从源码层面带你实现一个自动注入注解
首先,需要了解到的是。SpringBean的生命周期在生命周期中。注入bean属性的位置是在以下代码:populateBean位置中
那么我们在项目中使用注解产生一个bean的时候必定会经过以下代码进行一个bean的创建流程
/**省略代码**///开始初始化bean实例对象ObjectexposedObject=bean;try{ //<5>对bean进行填充,将各个属性值注入,其中,可能存在依赖于其他bean的属性populateBean(beanName,mbd,instanceWrapper);//<6>调用初始化方法exposedObject=initializeBean(beanName,exposedObject,mbd);}catch(Throwableex){ if(exinstanceofBeanCreationException&&beanName.equals(((BeanCreationException)ex).getBeanName())){ throw(BeanCreationException)ex;}else{ thrownewBeanCreationException(mbd.getResourceDescription(),beanName,"Initializationofbeanfailed",ex);}}/**省略代码**/在生命周期中populateBean进行填充bean数据。把其他依赖引入进来
BeanPostProcessor是一个bean创建时候的一个钩子。
以下代码是循环调用实现了BeanPostProcessor子类InstantiationAwareBeanPostProcessor#postProcessProperties方法
Spring在以下代码中有自动注入的拓展点。关键就是实现InstantiationAwareBeanPostProcessor#postProcessProperties
/**省略代码**/for(BeanPostProcessorbp:getBeanPostProcessors()){ if(bpinstanceofInstantiationAwareBeanPostProcessor){ InstantiationAwareBeanPostProcessoribp=(InstantiationAwareBeanPostProcessor)bp;//对所有需要依赖检查的属性进行后处理PropertyValuespvsToUse=ibp.postProcessProperties(pvs,bw.getWrappedInstance(),beanName);if(pvsToUse==null){ //从bw对象中提取PropertyDescriptor结果集//PropertyDescriptor:可以通过一对存取方法提取一个属性if(filteredPds==null){ filteredPds=filterPropertyDescriptorsForDependencyCheck(bw,mbd.allowCaching);}pvsToUse=ibp.postProcessPropertyValues(pvs,filteredPds,bw.getWrappedInstance(),beanName);if(pvsToUse==null){ return;}}pvs=pvsToUse;}}/**省略代码**/我们展开来讲一下@Autowired的实现是怎么样的吧:
实现类为AutowiredAnnotationBeanPostProcessor.java
从上面可以得知,填充bean的时候。时调用了方法ibp.postProcessPropertyValues()
那么AutowiredAnnotationBeanPostProcessor#postProcessPropertyValues()则会被调用
调用findAutowiringMetadata获取class以及父类带有@Autowired或者@Value的属性或者方法:
/**省略代码**/publicPropertyValuespostProcessProperties(PropertyValuespvs,Objectbean,StringbeanName){ //获取所有可以注入的元数据InjectionMetadatametadata=findAutowiringMetadata(beanName,bean.getClass(),pvs);try{ //注入数据metadata.inject(bean,beanName,pvs);}catch(BeanCreationExceptionex){ throwex;}catch(Throwableex){ thrownewBeanCreationException(beanName,"Injectionofautowireddependenciesfailed",ex);}returnpvs;}privateInjectionMetadatafindAutowiringMetadata(StringbeanName,Class<?>clazz,@NullablePropertyValuespvs){ //缓存名字获取StringcacheKey=(StringUtils.hasLength(beanName)?beanName:clazz.getName());InjectionMetadatametadata=this.injectionMetadataCache.get(cacheKey);//获取是否已经读取过这个class类的InjectionMetadata有的话直接从缓存中获取出去if(InjectionMetadata.needsRefresh(metadata,clazz)){ synchronized(this.injectionMetadataCache){ //双重检查metadata=this.injectionMetadataCache.get(cacheKey);if(InjectionMetadata.needsRefresh(metadata,clazz)){ if(metadata!=null){ metadata.clear(pvs);}//构建自动注入的元数据metadata=buildAutowiringMetadata(clazz);this.injectionMetadataCache.put(cacheKey,metadata);}}}returnmetadata;}privateInjectionMetadatabuildAutowiringMetadata(finalClass<?>clazz){ if(!AnnotationUtils.isCandidateClass(clazz,this.autowiredAnnotationTypes)){ returnInjectionMetadata.EMPTY;}List<InjectionMetadata.InjectedElement>elements=newArrayList<>();Class<?>targetClass=clazz;do{ finalList<InjectionMetadata.InjectedElement>currElements=newArrayList<>();//循环targetClass的所有field并执FieldCallback逻辑(函数式编程接口,传入的是一个执行函数)ReflectionUtils.doWithLocalFields(targetClass,field->{ //获得字段上面的Annotation注解MergedAnnotation<?>ann=findAutowiredAnnotation(field);if(ann!=null){ //判断是否为静态属性如果是,则不进行注入if(Modifier.isStatic(field.getModifiers())){ if(logger.isInfoEnabled()){ logger.info("Autowiredannotationisnotsupportedonstaticfields:"+field);}return;}//注解是否为必须依赖项booleanrequired=determineRequiredStatus(ann);currElements.add(newAutowiredFieldElement(field,required));}});//循环targetClass的所有Method并执MethodCallback逻辑(函数式编程接口,传入的是一个执行函数)ReflectionUtils.doWithLocalMethods(targetClass,method->{ MethodbridgedMethod=BridgeMethodResolver.findBridgedMethod(method);if(!BridgeMethodResolver.isVisibilityBridgeMethodPair(method,bridgedMethod)){ return;}MergedAnnotation<?>ann=findAutowiredAnnotation(bridgedMethod);if(ann!=null&&method.equals(ClassUtils.getMostSpecificMethod(method,clazz))){ //判断是否为静态方法如果是,则不进行注入if(Modifier.isStatic(method.getModifiers())){ if(logger.isInfoEnabled()){ logger.info("Autowiredannotationisnotsupportedonstaticmethods:"+method);}return;}//判断静态方法参数是否为0if(method.getParameterCount()==0){ if(logger.isInfoEnabled()){ logger.info("Autowiredannotationshouldonlybeusedonmethodswithparameters:"+method);}}booleanrequired=determineRequiredStatus(ann);PropertyDescriptorpd=BeanUtils.findPropertyForMethod(bridgedMethod,clazz);currElements.add(newAutowiredMethodElement(method,required,pd));}});//数据加到数组最前方父类的的注解都放在靠前的位置elements.addAll(0,currElements);//如果有父类则设置targetClass为父类。如此循环targetClass=targetClass.getSuperclass();}while(targetClass!=null&&targetClass!=Object.class);returnInjectionMetadata.forElements(elements,clazz);}/**省略代码**/真正注入数据的是metadata.inject(bean,beanName,pvs);
调用的是InjectionMetadata#inject方法
publicvoidinject(Objecttarget,@NullableStringbeanName,@NullablePropertyValuespvs)throwsThrowable{ Collection<InjectedElement>checkedElements=this.checkedElements;//带有注解的方法或者属性列表Collection<InjectedElement>elementsToIterate=(checkedElements!=null?checkedElements:this.injectedElements);if(!elementsToIterate.isEmpty()){ for(InjectedElementelement:elementsToIterate){ element.inject(target,beanName,pvs);}}}循环调用之前加入的带有注解的方法或者属性构建的对象AutowiredFieldElement#inject,AutowiredMethodElement#inject
/***属性上有注解构建的处理对象*/privateclassAutowiredFieldElementextendsInjectionMetadata.InjectedElement{ privatefinalbooleanrequired;privatevolatilebooleancached;@NullableprivatevolatileObjectcachedFieldValue;publicAutowiredFieldElement(Fieldfield,booleanrequired){ super(field,null);this.required=required;}@Overrideprotectedvoidinject(Objectbean,@NullableStringbeanName,@NullablePropertyValuespvs)throwsThrowable{ //获取属性名Fieldfield=(Field)this.member;Objectvalue;//Bean不是单例的话,会重复进入注入的这个操作,if(this.cached){ try{ value=resolvedCachedArgument(beanName,this.cachedFieldValue);}catch(NoSuchBeanDefinitionExceptionex){ //Unexpectedremovaloftargetbeanforcachedargument->re-resolvevalue=resolveFieldValue(field,bean,beanName);}}else{ //首次创建的时候进入该方法value=resolveFieldValue(field,bean,beanName);}if(value!=null){ //属性如果不为public的话,则设置为可访问ReflectionUtils.makeAccessible(field);field.set(bean,value);}}@NullableprivateObjectresolveFieldValue(Fieldfield,Objectbean,@NullableStringbeanName){ //构建DependencyDescriptor对象DependencyDescriptordesc=newDependencyDescriptor(field,this.required);desc.setContainingClass(bean.getClass());//注入bean的数量。有可能字段上是一个ListSet<String>autowiredBeanNames=newLinkedHashSet<>(1);Assert.state(beanFactory!=null,"NoBeanFactoryavailable");//获得beanFactory类型转换类TypeConvertertypeConverter=beanFactory.getTypeConverter();Objectvalue;try{ //查找依赖关系value=beanFactory.resolveDependency(desc,beanName,autowiredBeanNames,typeConverter);}catch(BeansExceptionex){ thrownewUnsatisfiedDependencyException(null,beanName,newInjectionPoint(field),ex);}synchronized(this){ if(!this.cached){ ObjectcachedFieldValue=null;if(value!=null||this.required){ cachedFieldValue=desc;//填入依赖关系registerDependentBeans(beanName,autowiredBeanNames);//判断如果注入依赖是只有一个if(autowiredBeanNames.size()==1){ StringautowiredBeanName=autowiredBeanNames.iterator().next();if(beanFactory.containsBean(autowiredBeanName)&&beanFactory.isTypeMatch(autowiredBeanName,field.getType())){ cachedFieldValue=newShortcutDependencyDescriptor(desc,autowiredBeanName,field.getType());}}}this.cachedFieldValue=cachedFieldValue;this.cached=true;}}returnvalue;}}/***方法上有注解构建的处理对象*/privateclassAutowiredMethodElementextendsInjectionMetadata.InjectedElement{ privatefinalbooleanrequired;privatevolatilebooleancached;@NullableprivatevolatileObject[]cachedMethodArguments;publicAutowiredMethodElement(Methodmethod,booleanrequired,@NullablePropertyDescriptorpd){ super(method,pd);this.required=required;}@Overrideprotectedvoidinject(Objectbean,@NullableStringbeanName,@NullablePropertyValuespvs)throwsThrowable{ //检查属性是不会在之前就已经注入过了。如果主如果则不进行二次覆盖if(checkPropertySkipping(pvs)){ return;}Methodmethod=(Method)this.member;Object[]arguments;if(this.cached){ try{ arguments=resolveCachedArguments(beanName);}catch(NoSuchBeanDefinitionExceptionex){ //Unexpectedremovaloftargetbeanforcachedargument->re-resolvearguments=resolveMethodArguments(method,bean,beanName);}}else{ //首次创建的时候进入该方法arguments=resolveMethodArguments(method,bean,beanName);}if(arguments!=null){ try{ //属性如果不为public的话,则设置为可访问ReflectionUtils.makeAccessible(method);//调用方法并传入参数method.invoke(bean,arguments);}catch(InvocationTargetExceptionex){ throwex.getTargetException();}}}@NullableprivateObject[]resolveCachedArguments(@NullableStringbeanName){ Object[]cachedMethodArguments=this.cachedMethodArguments;if(cachedMethodArguments==null){ returnnull;}Object[]arguments=newObject[cachedMethodArguments.length];for(inti=0;i<arguments.length;i++){ arguments[i]=resolvedCachedArgument(beanName,cachedMethodArguments[i]);}returnarguments;}@NullableprivateObject[]resolveMethodArguments(Methodmethod,Objectbean,@NullableStringbeanName){ //获取方法上有几个参数intargumentCount=method.getParameterCount();Object[]arguments=newObject[argumentCount];DependencyDescriptor[]descriptors=newDependencyDescriptor[argumentCount];Set<String>autowiredBeans=newLinkedHashSet<>(argumentCount);Assert.state(beanFactory!=null,"NoBeanFactoryavailable");TypeConvertertypeConverter=beanFactory.getTypeConverter();for(inti=0;i<arguments.length;i++){ //方法参数,从方法参数中取出i构造MethodParameter对象MethodParametermethodParam=newMethodParameter(method,i);DependencyDescriptorcurrDesc=newDependencyDescriptor(methodParam,this.required);currDesc.setContainingClass(bean.getClass());descriptors[i]=currDesc;try{ //获取方法中i参数的内容Objectarg=beanFactory.resolveDependency(currDesc,beanName,autowiredBeans,typeConverter);if(arg==null&注释和注解的区别
注释和注解的区别介绍如下:(1)注解:用于描述代码,说明程序,主要目的是为了给计算机看,且能够影响程序的运行。
(2)注释:用于描述代码的作用和一些关键性的知识点,使用文字描述程序,是为了给程序员观看,以此来使程序员能够以快的时间了解被注释的代码。
解释:
注解
叫元数据,一种代码级别的说明,它是JDK1.5及以后版本引入的一个特性,与类、接口、枚举在同一个层次,它可以声明在包、类、字段、局部变量、方法参数等的前面,用来对这些元素进行说明、注释。
注解的作用分类
编写文档:通过代码里表示的元数据生成文档生成doc文档。
代码分析:通过代码里表示的元数据进行分析使用反射。
编译检查:通过代码里表示的元数据让编译器能够实现基本的编译检查Override。
注解按照运行机制分类
源码注解:注解只在源码中存在,编译成.class文件之后就不存在了。
编译时注解:注解在源码存在的基础上,也会在.class文件中存在,但是在运行阶段中就不存在了,例如:@Override。
运行时注解:注解在运行阶段依然存在,且能够影响程序的运行过程,例如:@Autowired。