Spring Configuration:@Import的用法和源码解析
Spring 3.0之后的@Configuration注解和注解配置体系替代了XML配置,本文主要讲解@Import的处理用法和源码解析。@Import的注解乐聊 源码下载用法
配置类(带有@Configuration注解)不仅可通过@Bean声明bean,还可通过@Import导入其他类。源码例如,处理WebMvcConfig类通过@Import导入其他配置类,注解同时启用@EnableWebMvc。源码直接导入
配置类上使用@Import可以导入一个或多个类,处理甚至可以出现在父类注解中。注解如WebMvcConfig导入DelegatingWebMvcConfiguration等。源码ImportBeanDefinitionRegistrar和ImportSelector
@Import除了导入配置类,处理还可以导入实现了ImportBeanDefinitionRegistrar(如@EnableAspectJAutoProxy)和ImportSelector(如@EnableTransactionManagement)的注解类。源码解析
ConfigurationClassPostProcessor负责处理@Configuration类,通过ConfigurationClassParser解析配置和导入,由ConfigurationClassBeanDefinitionReader注册BeanDefinition。在解析过程中,处理@Import避免循环导入,通过导入链和ImportStack进行判断。处理直接导入时,通过导入链判断循环。
处理注册器和选择器时,提前触发Aware接口方法,然后在适当时机注册导入的类。
总结来说,@Import提供了多种导入方式的灵活性,Spring的源码设计考虑了循环导入和重复解析的处理,展示了其强大的自定义配置能力。Netty源码-Sharable注解,我猜你理解错了!
学习目标
理解Sharable注解在Netty中的作用。个人臆想,公告系统源码只要在ChannelHandler上添加了@ChannelHandler.Sharable注解,所有的channel都共享这一个ChannelHandler实例。真是这样的吗?如果你认为所有channel共享一个pipeline,只能说明你对pipeline的初始化不理解。你臆想完全错误!!!
Sharable注解介绍
在深入探讨之前,建议阅读《Netty源码-ChannelPipeline的剖析》。阅读后,问题迎刃而解。同时,你可能还会疑惑Sharable注解的用途。它的主要目的是在pipeline之间共享信息,例如统计连接数、限流、白名单等。
如何使用Sharable注解
源码中有明确说明,你可以在ChannelHandler的成员变量上使用@ChannelHandler.Sharable注解,表示这个变量的实例可以被所有pipeline共享。如果不想共享,就每次创建新实例。通过代码示例可以直观理解共享实例和每次创建新实例的区别。
使用场景
Sharable注解适用于在pipeline之间需要共享数据或状态的场景。例如统计连接数、限流、白名单管理等。
注意事项
使用Sharable注解时,务必考虑线程安全。在多线程环境中,确保共享数据的正确性和一致性。
实战应用-统计当前连接数
通过实践,我们实现了一个简单的java linkedmap源码统计当前连接数的示例。执行结果如下:
Connected to the target VM, address: '.0.0.1:', transport: 'socket' count:1 InBoundHandler1-channelRead:abc InBoundHandler1-channelRead:asd count:2 InBoundHandler1-channelRead:abc InBoundHandler1-channelRead:asd
注解处理器(APT)了解一下
APT全称为Annotation Processing Tool,即注解处理器,是一种命令行程序,用于处理源代码中添加的注解对象,并通过注解处理器对其进行处理。通常,APT用于生成源码,然后将其加入编译目录以简化开发周期。
在实际开发中,自定义注解的使用相对较少。注解是Java1.5中引入的概念,用于标记代码的元数据。定义一个注解使用@interface关键字,如定义名为Test的注解。
元注解是注解的注解,用于标记注解的元数据,允许控制注解的属性和行为。Java中存在四种元注解:Retention、Inherited、Documented和Target。
Retention表示注解的保留范围,取值为枚举类:RetentionPolicy,包括三个类型。
Target用于控制注解的使用范围,取值为枚举类:ElementType。
Documented用于描述注解是否需要加入到如javadoc工具生成的公共API文档中,带有Documented注解的注解将保留在生成的文档中。
Inherited元注解用于标记注解,当子类继承使用该注解的类时,子类将继承该注解。例如,定义注解Test并使用@Inherited元注解标记,然后定义Father类使用@Test注解,selenium源码下载再定义Son类继承Father类,Son类将拥有注解Test。
APT的基本原理是在需要使用的元素上添加自定义注解,然后在编译时收集使用了该注解的元素并进行统一处理,如根据元素生成对应的工具类,以提高开发效率。
创建一个注解处理器涉及以下步骤:定义注解、创建注解处理器类、实现核心方法。
以ButterKnife框架为例,它通过APT技术实现,功能是通过注解为对象赋值。创建控件后,需要使用控件ID获取控件对象,多个重复代码,此框架通过注解直接初始化控件。
创建注解处理器的步骤包括定义注解、创建处理器类、实现关键方法。在处理器类中,通过getSupportedAnnotationTypes()方法返回注解类型,process方法用于解析注解元素并生成Java代码。
使用注解处理器时,通过创建配置文件声明处理器全限定名称,配置完成后,按照ButterKnife的使用方式即可使用APT生成的代码。
APT技术提供了高效的注解处理机制,简化了开发流程。通过学习APT,开发者可以更高效地处理注解,提高代码的可维护性和可读性。了解更多APT技术内容,请关注公众号:zhangke_blog。spring 3.0源码
SpringBoot源码学习——SpringBoot自动装配源码解析+Spring如何处理配置类的
SpringBoot通过SPI机制,借助外部引用jar包中的META-INF/spring.factories文件,实现引入starter即可激活功能,简化手动配置bean,实现即开即用。
启动SpringBoot服务,通常使用Main方法启动,其中@SpringBootApplication注解包含@SpringBootConfiguration、@EnableAutoConfiguration、@ComponentScan,自动装配的核心。
深入分析@SpringBootApplication,其实质是执行了@SpringBootConfiguration、@EnableAutoConfiguration、@ComponentScan三个注解的功能,简化了配置过程,强调了约定大于配置的思想。
SpringBoot的自动装配原理着重于研究如何初始化ApplicationContext,Spring依赖于ApplicationContext实现其功能,SpringApplication#run方法为初始化ApplicationContext的入口。
分析SpringApplication构造方法,SpringApplication.run(启动类.class, args) 实际调用的是该方法,其关键在于根据项目类型反射生成合适的ApplicationContext。
选择AnnotationConfigServletWebServerApplicationContext,此上下文具备启动Servlet服务器和注册Servlet或过滤器类型bean的能力。
准备刷新ApplicationContext,SpringBoot将主类注册到Spring容器中,以便@ConfigurationClassPostProcessor解析主类注解,发挥@Import、@ComponentScan的作用。
刷新ApplicationContext过程包括一系列前置准备,如将主类信息封装成AnnotatedGenericBeanDefinition,解析注解并调用BeanDefinitionCustomizer自定义处理。
解析配置类中的注解,通过BeanDefinitionRegistryPostProcessor和ConfigurationClassParser实现,筛选、排序候选者,并解析@Import注解实现自动装配。
增强配置类,ConfigurationClassPostProcessor对full模式的配置进行增强,确保@Import正确处理,CGLIB用于增强原配置类,确保生命周期完整,避免真正执行@Bean方法逻辑。
深入解析AutoConfigurationImportSelector实现自动装配,通过spring.boot.enableautoconfiguration设置开启状态,读取spring-autoconfigure-metadata.properties和META-INF/spring.factories文件,筛选并加载自动配置类。
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中注解是什么意思
Java注解是一种元数据,它们为源代码添加了信息,而不会修改实际代码。注解不会影响程序的运行,但可以用来提供关于代码的附加信息,例如执行某些任务或配置某些值。它们可以像Javadoc一样被编译器和其他工具使用,用于生成文档、执行静态分析等操作。注解是Java程序中非常有用的一种特性,它们可以用于许多不同的任务,包括测试、配置和性能优化等。
Java注解可以应用于各种不同的元素,包括类、方法、变量、参数等等。使用注解可以为这些元素提供额外的信息,或指定某些行为。例如,注解可以指定方法的可见性、参数的名称或类型,甚至可以标记需要特殊处理的代码块。Java中的许多库和框架都使用了注解来实现某些功能,例如JUnit测试框架就使用注解标记测试方法和断言条件。通过使用注解,开发人员可以使代码更加清晰和易于维护。
Java注解的本质是一种元数据,它们不会修改程序代码,而是提供了一些额外的信息。注解可以在源代码级别访问,可以帮助编写更好的文档和工具,同时还可以增强代码的可读性和可维护性。注解并不是Java中的一个新特性,它们从Java5版本开始引入,并经过多年的发展变得越来越强大和灵活。注解在Java程序中应用广泛,是提高代码质量和开发效率的重要工具之一。
@Lazy注解源码分析
@Lazy注解是Spring框架3.0版本后引入的,用于控制bean的懒加载行为,主要用途是延迟依赖注入的初始化。默认情况下,当ApplicationContext启动和刷新时,所有的单例bean会被立即初始化。然而,有时可能希望某些bean在首次使用时才被初始化。实现这一目标的方法是将@Lazy注解应用到bean或注入点,如@Autowired,以创建懒解析代理,从而实现延迟注入。
@Lazy注解对@Bean、@Component或@Bean定义的bean的延迟初始化特别有用。当用在@Configuration类上时,它会影响该配置中的所有@Bean定义。通过在启动类入口使用AnnotationConfigApplicationContext并提供MyConfiguration组件类,从MyService bean获取并调用其show方法,可以观察到MyBean在首次被请求时才被初始化,而MyService的初始化则立即进行。MyBean类的构造函数在被调用时打印"MyBean的构造函数被调用了!",show方法则打印"hello world!"。MyService类通过@Autowired注入MyBean,由于在注入点上添加了@Lazy注解,myBean的实际注入被延迟,直到首次尝试访问它时。
源码分析表明,在启动类构造函数中,执行了三个步骤以初始化实例。在refresh方法中,重点关注了finishBeanFactoryInitialization方法,该方法会对所有剩余非懒加载的单例bean对象进行初始化,除非它们显式标记为懒加载。在preInstantiateSingletons方法中,确保所有非懒加载的单例bean在容器启动时被初始化,除非它们被标记为懒加载。这使得@Lazy注解对于希望推迟bean初始化的场景非常有用。
在getBean()方法中,通过doGetBean方法执行了创建bean的过程。在doCreateBean方法中,对bean的属性进行注入。在populateBean方法中,如果一个属性被标记为@Autowired,并且与@Lazy结合使用,那么实际的懒加载逻辑会在其他部分处理,特别是通过AutowiredAnnotationBeanPostProcessor。在resolveFieldValue方法中,解析@Autowired字段的值,并确定应为目标字段注入哪个bean。在resolveDependency方法中,如果依赖关系标记为懒加载,它将返回一个懒加载代理,只有在应用程序真正访问该依赖时,实际的bean才会被初始化。
总结而言,@Lazy注解提供了在Spring容器中控制bean初始化的灵活性,允许开发者根据需要延迟依赖注入的初始化,从而优化应用性能和资源管理。在实践过程中,注意合理使用@Lazy注解,确保代码的清晰性和可维护性。同时,理解Spring容器在bean初始化过程中的工作原理,可以帮助开发者更有效地利用该框架的特性,实现更高效的应用开发。
2025-01-04 09:37
2025-01-04 08:43
2025-01-04 08:23
2025-01-04 07:40
2025-01-04 07:26