- 使用 Spring Initializr 创建 Spring Boot 应用程序
- 在Spring Boot中配置Cassandra
- 在 Spring Boot 上配置 Tomcat 连接池
- 将Camel消息路由到嵌入WildFly的Artemis上
一. Spring注解发展过程
SpringBoot的自动装配依赖于注解,所以我们先来看一下注解的发展过程。
以下主要对核心注解进行说明
Spring1.0:刚刚出现注解。
@Transaction:简化了事务的操作
Spring2.0:一些配置开始被xml代替,但是还不能完全摆脱xml,主要是component-scan标签。
@Required:用在set方法上,如果加上该注解,表示在xml中必须设置属性的值,不然就会报错。
@Aspect :AOP相关的一个注解,用来标识配置类。
@Autowired,@Qualifier:依赖注入
@Component,@Service,@Controller,@Repository:主要是声明一些bean对象放入IOC中。
@RequestMapping: 声明请求对应的处理方法
Spring3.0:已经完全可以用注解代替xml文件了
@Configuration:配置类,代理xml配置文件
@ComponentScan:扫描其他注解,代理xml中的component-scan标签。
@Import:只能用在类上,主要是用来加载第三方的类。
@import(value = {XXX.class}):加载一个普通的类
@Import(MyImportSelector.class):这种主要是根据业务选择性加载一些类。
public class MyImportSelector implements ImportSelector {//继承该接口
@Override //重写selectImports方法
publicString[] selectImports(AnnotationMetadata importingClassMetadata) {
//返回对象对应的类型的全类路径的字符串数组
return new String[]{XXX1.class.getName(), XXX2.class.getName()};
}
}
public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {//继承该接口
@Override //重写registerBeanDefinitions方法
public voidregisterBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
//将需要注册的对象封装为 RootBeanDefinition 对象
RootBeanDefinition xxx1 = new RootBeanDefinition(XXX1.class);
registry.registerBeanDefinition("xxx1", xxx1);
//再注册一个
RootBeanDefinition xxx2 = new RootBeanDefinition(XXX2.class);
registry.registerBeanDefinition("xxx2", xxx2);
}
}
Spring4.0:
@Conditional:按照一定的条件进行判断,满足条件就给容器注册Bean实例。
/*** 定义一个 Condition 接口的是实现
*/
public class MyCondition implementsCondition {
@Override
public booleanmatches(ConditionContext context, AnnotatedTypeMetadata metadata) {
//业务逻辑...
return false; //默认返回false
}
}
//使用
@Configuration
public classJavaConfig {
@Bean
//条件注解,添加的类型必须是 实现了 Condition 接口的类型
//MyCondition的 matches 方法返回true 则注入,返回false 则不注入
@Conditional(MyCondition.class)
publicStudentService studentService() {
return newStudentService();
}
}
Spring5.0:
@Indexed:在Spring Boot应用场景中,大量使用@ComponentScan扫描,导致Spring模式的注解解析时间耗时增大,因此,5.0时代引入@Indexed,为Spring模式注解添加索引。
当我们在项目中使用了 @Indexed 之后,编译打包的时候会在项目中自动生成METAINT/spring.components文件。根据该文件进行扫描注入,可以提高效率。
** 二. SpringBoot自动装配原理**
自动装配还是利用了SpringFactoriesLoader来加载META-INF/spring.factoires文件里所有配置的EnableAutoConfgruation,它会经过exclude和filter等操作,最终确定要装配的类
1.一切的开始都源于@SpringBootApplication,它是一个组合注解
除了元注解之外,关注这三个注解:
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan
2.看来这个@EnableAutoConfiguration不简单
@Import(AutoConfigurationImportSelector.class)
它的内部主要是使用@import注解导入一个选择器。
3.那么我们看看这个AutoConfigurationImportSelector类
上文提到继承ImportSelector接口的类,需要重写 selectImports( ),那我们就看看这个方法
@Override
publicString[] selectImports(AnnotationMetadata annotationMetadata) {
if (!isEnabled(annotationMetadata)) {
returnNO_IMPORTS;
}
AutoConfigurationEntry autoConfigurationEntry =getAutoConfigurationEntry(annotationMetadata);
returnStringUtils.toStringArray(autoConfigurationEntry.getConfigurations());
}
该方法其实也没说啥,现在的重心就放在getAutoConfigurationEntry()中
4.getAutoConfigurationEntry()
protectedAutoConfigurationEntry getAutoConfigurationEntry(AnnotationMetadata annotationMetadata) {
if (!isEnabled(annotationMetadata)) {
returnEMPTY_ENTRY;
}
AnnotationAttributes attributes =getAttributes(annotationMetadata);
//获取候选配置信息,加载的是当前项目的classpath目录下的所有的 spring.factories 文件中的 key 为
//org.springframework.boot.autoconfigure.EnableAutoConfiguration 的信息。
//点进去通过"SpringFactoriesLoader"进行加载List<String> configurations =getCandidateConfigurations(annotationMetadata, attributes);
//removeDuplicates方法的作用是 移除同名的
configurations =removeDuplicates(configurations);
//获取我们配置的 exclude 信息
//比如:@SpringBootApplication(exclude = {RabbitAutoConfiguration.class}) ,显示的指定不要加载那个配置类
Set<String> exclusions =getExclusions(annotationMetadata, attributes);
checkExcludedClasses(configurations, exclusions);
configurations.removeAll(exclusions);
//filter的作用是 过滤掉咱们不需要使用的配置类。
configurations =getConfigurationClassFilter().filter(configurations);
fireAutoConfigurationImportEvents(configurations, exclusions);
return newAutoConfigurationEntry(configurations, exclusions);
}
5.前面几个都好理解,现在我们主要看看filter(),是怎么移除不需要的类
我们可以看到有具体的匹配方法 match。里面有个关键的属性是 autoConfigurationMetadata , 的本质是 加载的 META-INF/spring-autoconfigure-metadata.properties 的文件中的内容。
其实原理很简单,如果没有对应的实现类,就不进行加载。
到这里自动装配的原理就完事了~ 三. 何时进行自动装配
四. run 方法
@SpringBootApplication
public classSpringBootVipDemoApplication {
public static voidmain(String[] args) {
//基于配置文件的方式
ApplicationContext ac1 = new ClassPathXmlApplicationContext("");
//基于Java配置类的方式
ApplicationContext ac2 = new AnnotationConfigApplicationContext(SpringBootVipDemoApplication.class);
//run 方法的返回对象是 ConfigurableApplicationContext 对象,
//ConfigurableApplicationContext就是ApplicationContext的一个子接口
ConfigurableApplicationContext ac3 = SpringApplication.run(SpringBootVipDemoApplication.class, args);
}
}
根据返回结果,我们猜测SpringBoot项目的启动其实就是Spring的初始化操作【IOC】。
所以我们发现SpringBoot项目的启动,本质上就是Spring的初始化操作。
想亲身感受自动装配,可以参考手写一个SpringBoot starter
寄语:努力的意义就是随时有能力跳出自己厌恶的圈子
我想做的是,如果鼠标位于“下一个”按钮上,它会以慢速向右滚动,如果鼠标没有位于“下一个”按钮上,它会停止滚动? 这是我的尝试http://jsfiddle.net/mdanz/nCCRy/14/ $(
StyleCop 是一个很棒的视觉工作室小插件。但它不会向您显示实时提示或提供任何自动修复。 随之而来的是 reSharper 和 StyleCop for reSharper,这是理想的解决方案,但
我为我的MatchQuery使用了模糊性选项,但是我想将模糊性值设置为auto。有什么办法吗? 另外,对于完成建议程序,您可以将其设置为支持unicode,对于我的MatchQuery,有什么方法可以
我想从表中获取一行[字符串名称,字符串密码,int 某些内容]并将其映射到一个 User 对象,该对象具有 3 个属性,如上面的 getter 和 setter有什么方法可以自动完成吗?我考虑过反射,
我有一个像这样的方法:void m1(string str) 并且有一个像这样的类: public class MyClass { public bool b1 { set; get; }
我正在尝试使用 $rootScope 从一个 Controller 向另一个 Controller $broadcast 一些数据。 如果我使用像 ng-click 这样的触发器来运行将广播的功能,它
我考虑了很多关于是要使用完全自动化的缓存还是手动缓存。 我们的自动方法是一种解决方案,它可以挖掘数据库、查询和格式化每个潜在和 future 的数据请求,并将其保存到适当的缓存存储(内存缓存或基于磁盘
我的 CSS 必须使用过渡来更改,直到现在我都使用 div:hover 来实现。 当您单击另一个 div 时需要激活过渡,而不是当您将鼠标悬停在必须移动/更改的 div 上时。 我该怎么做? 谢谢 永
在我的应用程序中,我需要一些动画,但如果它已经设置了动画,则不需要持续时间。但我的问题是它会自动添加持续时间。 在这里你可以看到 2 个函数,第二个没有持续时间但它确实有持续时间(可能从 1 秒开始)
两年前,我需要制作一个工具,通过 POST 自动将 txt/csv 文件上传到我的 Web 服务器,然后使用 cronjob 通过 PHP 对其进行解析。 这有两次在每天午夜自动发生。尽管这行得通,但
请阅读下面程序中的评论: #include void test(char c[]) { c=c+2; //why does this work ? c--; printf("%
也许是个幼稚的问题,但是...... 确认或拒绝: 自动和静态存储持续时间的对象/变量的内存的存在是在编译时确定的,程序运行时失败的可能性绝对为零,因为没有足够的内存用于自动对象。 自然地,当自动对象
有没有什么方法可以自动获得类中属性更改的通知,而不必在每个 setter 中都编写 OnPropertyChanged? (我有数百个属性,我想知道它们是否已更改)。 安东建议 dynamic pro
我们在使用 Azure DevOps 的项目中采用了 gitflow 流程。我有以下场景: 当功能分支合并到 Develop 时,我想在完成拉取请求的同时执行压缩合并策略 当 Release 分支定期
我的网站上有一个评论部分,我将 html 编码的评论保存在我的数据库中。所以我添加了这条评论- "testing" `quotes` \and backslashes\ and html 并将其保存在
是否存在“ checkin 前 TFS 自动 checkout ”这样的功能,以便在我说“ checkin ”之前我不会 checkout 任何文件,例如以防我只是临时更改文件 - 这一直发生。 换句
我有一个运行在 Linux/Apache/Tomcat 堆栈上的网站,它需要每隔几个月自动脱机以进行服务器维护,这将持续任意时间。有哪些选项可以让 Apache 建立和取消“服务器维护”页面? 我需要
我经常在工作中创建文档,在公司内部,由于我们使用的首字母缩写词和缩写词的数量,我们几乎拥有自己的语言。因此,我厌倦了在发布文档之前手动创建首字母缩写词和缩写表,并且快速的谷歌搜索发现了一个可以有效地为
我希望在用户或宏将计算模式从自动更改为手动或手动更改为自动时运行代码。是否有为此触发的事件? (属性是 Application.Calculation 在 Excel 互操作中。) 使用 Excel
这个问题在这里已经有了答案: Repeat command automatically in Linux (13 个回答) 6年前关闭。 我想创建一个脚本来获取另一个文件夹中的所有文件夹名称。并为这些
我是一名优秀的程序员,十分优秀!