什么样的经历,才能领悟成为架构师? >>>
  
  本文主要分析 SpringBoot 的启动过程。
  
  SpringBoot的版本为:2.1.0 release,最新版本。
  
  一.时序图
  
  还是老套路,先把分析过程的时序图摆出来:时序图-SpringBoot2.10启动分析
  
  二.源码分析
  
  首先从我们的一个SpringBoot Demo开始,这里使用 SPRING INITIALIZR 网站生成的starter开始的:
  
  @SpringBootApplication
  
  public class SpringBootDemoApplication {
  
  public static void main(String[] args) {
  
  // 分析的入口,从 run 方法开始
  
  SpringApplication.run(SpringBootDemoApplication.class, args);
  
  }
  
  }
  
  经过SpringApplication多个重载的构造方法,最后到达:
  
  public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
  
  // 从 run() 传入的 resourceLoader 此处为 null
  
  this.resourceLoader = resourceLoader;
  
  // 使用断言判断 resourceLoader 不为空
  
  Assert.notNull(primarySources, "PrimarySources must not be null");
  
  // 把 primarySources 数组转为List,最后放入 primarySources 的一个LinkedHashSet中
  
  this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources));
  
  // 判断应用的类型:REACTIVE NONE SERVLET
  
  this.webApplicationType = WebApplicationType.deduceFromClasspath();
  
  // 实现 SpringBoot 自动装配的基础,此处Spring自己实现的SPI(从META-INF/spring.factories加载class)
  
  // 加载并实例化以 ApplicationContextInitializer 为key的类
  
  setInitializers((Collection) getSpringFactoriesInstances(
  
  ApplicationContextInitializer.class));
  
  // 加载并实例化以 ApplicationListener 为key的类
  
  setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));
  
  // 获取程序当前运行堆栈,看是运行的是哪个类的 main 方法,保存到上下文中
  
  this.mainApplicationClass = deduceMainApplicationClass();
  
  }
  
  看一眼,WebApplicationType#deduceFromClasspath ,deduce意为推断,即根据classpath下的内容推断出应用的类型。实现是通过ClassUtils#isPresent来尝试加载代表不同应用类型特征的Class文件:
  
  static WebApplicationType deduceFromClasspath() {// 判断应用的类型
  
  if (ClassUtils.isPresent(WEBFLUX_INDICATOR_CLASS, null)// 加载到DispatcherHandler
  
  && !ClassUtils.isPresent(WEBMVC_INDICATOR_CLASS, null)// mvc的DispatcherServlet
  
  && !ClassUtils.isPresent(JERSEY_INDICATOR_CLASS, null)) {// jersey的ServletContainer
  
  return WebApplicationType.REACTIVE;
  
  }
  
  for (String className : SERVLET_INDICATOR_CLASSES) {// 遍历数组:Servlet和ConfigurableWebApplicationContext
  
  if (!ClassUtils.isPresent(className, null)) {// 没有加载到Servlet相关的class
  
  return WebApplicationType.NONE;
  
  }
  
  }
  
  return WebApplicationType.SERVLET;
  
  }
  
  SpringApplication#getSpringFactoriesInstances,从类路径下 META-INF/spring.factories 下加载 SpringFactory 实例,类似的操作在 Dubbo SPI中也有:
  
  private <T> Collection<T> getSpringFactoriesInstances(Class<T> type,
  
  Class<?>[] parameterTypes, Object... args) {
  
  // 获取类加载器
  
  ClassLoader classLoader = getClassLoader();
  
  // 此处调用了SpringFactoriesLoader的loadFactoryNames()
  
  Set<String> names = new LinkedHashSet<>(
  
  SpringFactoriesLoader.loadFactoryNames(type, classLoader));
  
  // Use names and ensure unique to protect against duplicates
  
  // 实例化获取到的类
  
  List<T> instances = createSpringFactoriesInstances(type, parameterTypes,
  
  classLoader, args, names);
  
  AnnotationAwareOrderComparator.sort(instances);// 排序
  
  return instances;// 返回实例化好的对象
  
  }
  
  SpringFactoriesLoader#loadFactoryNames,加载工厂名字:
  
  public static List<String> www.yongshiyule178.com loadFactoryNames(Class<?> factoryClass, @Nullable ClassLoader classLoader) {
  
  String factoryClassName =www.mcyllpt.com/ factoryClass.getName(www.leyouzaixian2.com);
  
  return loadSpringFactories(classLoader).getOrDefault(factoryClassName, Collections.emptyList());
  
  }
  
  继续捉迷藏,到了 SpringFactoriesLoader#loadSpringFactories:下面的内容就是找到所有classpath下的 spring.factories 文件,读取里面的内容,放到缓存中,此处和Dubbo SPI中ExtensionLoader#loadDirectory几乎是一模一样,可以参考我写过的 Dubbo源码 里面的注释。
  
  private static Map<String, List<String>> loadSpringFactories(@Nullable ClassLoader classLoader) {
  
  // 从缓存中获取Map,key为classLoader
  
  MultiValueMap<String, String> result = cache.get(classLoader);
  
  if (result != null) {
  
  return result;
  
  }
  
  try {
  
  // 加载资源的urls,被加载的资源为 "META-INF/spring.factories"
  
  //先从Resources中加载,没有加载到再从SystemResources中加载
  
  Enumeration<URL> urls = (classLoader != null ?
  
  classLoader.getResources(FACTORIES_RESOURCE_LOCATION) :
  
  ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION));
  
  result = new LinkedMultiValueMap<>();
  
  while (urls.hasMoreElements()) {// 遍历加载到的 spring.factories 文件
  
  URL url = urls.nextElement();
  
  UrlResource resource = new UrlResource(url);
  
  // 读取文件到内存为Properties对象
  
  Properties properties = PropertiesLoaderUtils.loadProperties(resource);
  
  for (Map.Entry<www.michenggw.com/?, ?> entry :www.yigouyule2.cn properties.entrySet()) {
  
  // Entry的key作为工程Class的名字
  
  String factoryClassName = ((String) www.gcyl152.com/ entry.getKey()).trim();
  
  for (String factoryName : StringUtils.commaDelimitedListToStringArray((String) entry.getValue())) {
  
  // 如果有多个value,都放在Map中,注意此处为 MultiValueMap ,不是普通的Map,其实现内容的value对应一个LinkedList
  
  result.add(factoryClassName, factoryName.trim(www.gcyl159.com));
  
  }
  
  }
  
  }
  
  // 最后把读取配置的结果都放入缓存中,cache对象为一个ConcurrentReferenceHashMap
  
  cache.put(classLoader, result);
  
  return result;
  
  }
  
  catch (IOException ex) {
  
  throw new IllegalArgumentException("Unable to load factories from location [" +
  
  FACTORIES_RESOURCE_LOCATION + "]", ex);
  
  }
  
  }
  
  我们也来看一下上面读取的文件 spring.factories 的内容,大概长这个样子:
  
  # Initializers
  
  org.springframework.context.ApplicationContextInitializer=\
  
  org.springframework.boot.autoconfigure.SharedMetadataReaderFactoryContextInitializer,\
  
  org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListener
  
  # Application Listeners
  
  org.springframework.context.ApplicationListener=\
  
  org.springframework.boot.autoconfigure.BackgroundPreinitializer
  
  # Auto Configuration Import Listeners
  
  org.springframework.boot.autoconfigure.AutoConfigurationImportListener=\
  
  org.springframework.boot.autoconfigure.condition.ConditionEvaluationReportAutoConfigurationImportListener
  
  ......
  
  是时候跳出来了,回到主线,返回实例化对象后,到了 SpringApplication#deduceMainApplicationClass,获取程序当前运行堆栈,看现在运行的是哪个类的 main 方法,然后保存到上下文:
  
  private Class<?> deduceMainApplicationClass() {
  
  try {
  
  // 拿到运行时的堆栈信息
  
  StackTraceElement[] stackTrace = new RuntimeException().getStackTrace();
  
  for (StackTraceElement stackTraceElement : stackTrace) {
  
  // 如果发现哪个堆栈元素里面有运行了main方法,则返回该类
  
  if ("main".equals(stackTraceElement.getMethodName())) {
  
  return Class.forName(stackTraceElement.getClassName());
  
  }
  
  }
  
  }
  
  catch (ClassNotFoundException ex) {
  
  // Swallow and continue
  
  }
  
  return null;
  
  }
  
  至此,SpringApplication的构造函数的分析完成,后面我们继续分析SpringApplication的run()方法中做了哪些操作。

什么样的经历,才能领悟成为架构师? >>>的更多相关文章

  1. 一个4年工作经验的java程序员的困惑,怎样才能能为一个架构师,请教大神

    一个4年工作经验的java程序员的困惑,怎样才能能为一个架构师 LZ本人想往架构师发展, 业余时间也会看一些书籍, 但是感觉没有头绪, 有些书看了,也没有地方实践 我做了4年的java开发, 在一个公 ...

  2. JAVA兼职架构师

    在一些小企业或者公司人力不足的时候,经常会出现一个人干多个人的活.开发可能会干架构.测试.运维,一些小项目可能需要一个人完成.我把这些角色合并在一起称之为兼职架构师. 我用我的经历来说说兼职架构师的需 ...

  3. 向架构师进军--->系统架构设计基础知识

    如果你对项目管理.系统架构有兴趣,请加微信订阅号"softjg",加入这个PM.架构师的大家庭 在讲解系统架构设计之前,有必要补充一下架构相关的概念,因此本博文主要讲述架构.架构师 ...

  4. 谈谈.NET架构师面试及如何设计面试题

    上星期:应老东家的要求,帮其面试.NET架构师. 于是:老东家进行了一星期的简历收集: 终于:在一堆简历里,精挑细选了四个: 约了:周末上午下午各两个. 面试者年龄:在30-35岁左右,差不多10年. ...

  5. 阿里Java架构师分享自己的成长经历,教你如何快速成长为架构师

    架构师是公司的“金领”,很少需要考虑生存的问题,从而有更多的精力思考关键技术,形成“强者愈强”的良性循环.当然,冰冻三尺非一日之寒,成为一名合格的架构师是一个漫长的积累过程.对于大部分的软件开发人员来 ...

  6. 周爱民:真正的架构师是没有title的(图灵访谈)

    周爱民,现任豌豆荚架构师,国内软件开发界资深软件工程师.从1996年起开始涉足商业软件开发,历任部门经理.区域总经理.高级软件工程师.平台架构师等职,有18年的软件开发与架构.项目管理及团队建设经验, ...

  7. 架构师修炼 III - 掌握设计原则

    关于软件的设计原则有很多,对于设计原则的掌握.理解.实践及升华是架构师的一项极为之必要的修炼. 记得在12年前第一次阅读<敏捷开发>时,五大基本设计原则就深深地植入到我的脑海中一直影响至今 ...

  8. 架构师素养及从小菜进阶架构(CTO)的书籍【转】

    CTO要了解无线技术/搜索/大数据/数据库等. -- 通常定义架构有几个层次,这包括业务架构.产品架构.应用架构和技术架构: 1.业务架构:描述一个企业围绕一个行业做了哪些业务,例如支付行业的收单.退 ...

  9. 【转载】WEB架构师成长之路

    本人也是coding很多年,虽然很失败,但也总算有点失败的心得,不过我在中国,大多数程序员都是像我一样,在一直走着弯路,如果想成为一个架构师,就必须走正确的路,否则离目标越来越远,正在辛苦工作的程序员 ...

随机推荐

  1. P2629 好消息,坏消息

    题目描述 uim在公司里面当秘书,现在有n条消息要告知老板.每条消息有一个好坏度,这会影响老板的心情.告知完一条消息后,老板的心情等于之前老板的心情加上这条消息的好坏度.最开始老板的心情是0,一旦老板 ...

  2. Web版简易五子棋

    前些时候把大三写的C++版五子棋改成Web板挂到了网上,具有一定傻瓜式智能,欢迎体验使用拍砖:http://www.zhentiyuan.com/Games/QuickFiveChess.aspx 现 ...

  3. nodejs中相互引用(循环引用)的模块分析

    话不多少,直接上源码吧: modA.js: module.exports.test = 'A'; const modB = require('./05_modB'); console.log( 'mo ...

  4. 【学习笔记】深入理解js原型和闭包(0)——目录

    文章转载:https://www.cnblogs.com/wangfupeng1988/p/4001284.html 说明: 本篇文章一共16篇章,外加两篇后补的和一篇自己后来添加的学习笔记,一共19 ...

  5. 一条陌生的出路【过往d心声】

    一条陌生的出路 Vashon的心声 人生就像一列车,车上总有形形色色的人穿梭往来.你也可能会在车上遇到很多你以为有缘分的人,但是车也会有停下来的时候,总会有人从人生这列车上上下下,当你下去的时候你挥挥 ...

  6. mac osx上为qt应用生成debug symbol

    mac平台上,希望Qt编译的release程序也能包含debug symbol,这样出问题以后便于查找问题 开始按照http://doc.qt.io/qt-4.8/mac-differences.ht ...

  7. TFS2010升级至TFS2013完全指南(更换服务器)

    一.背景:         公司已使用tfs2010很长时间,目前随着公司的发展,项目越来越少,而产品越来越多,采用的开发模式,也逐渐从瀑布式.迭代式转向敏捷开发.为了更好的支持产品研发,决定将tfs ...

  8. 【HEVC简介】DB-DeBlock Filter

    参考论文:HEVC Deblocking Filter <HEVC标准介绍.HEVC帧间预测论文笔记>系列博客,目录见:http://www.cnblogs.com/DwyaneTalk/ ...

  9. VBA 连接sql server的用法

    cnnstr = "Provider=sqloledb;Data Source=192.211.21.8;Initial Catalog=pub;UID=账号;PWD=密码" VB ...

  10. laravel homestead comoser install 报错

    项目部署的时候composer install报错 说那个依赖包没有安装成功需要回滚删除但是删除不了 解决: 要配置共享文件 注:使用 NFS 的话,需要安装 vagrant-winnfsd 插件.该 ...