IOC容器加载过程及Bean的生命周期和后置处理器

SpringIOC 容器加载过程

第一步:实例化化容器:AnnotationConfigApplicationContext

@Configuration
@ComponentScan("cn.zhe")
public class MainStartTest {
    public static void main(String[] args) {
        // SpringIOC 出发点 加载Spring上下文
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainStartTest.class);
        HelloSpring bean = applicationContext.getBean(HelloSpring.class);
        bean.sayHello();
    }
}

构造函数

// 根据参数可知,可以传入多个Class,但这种情况及其少见
public AnnotationConfigApplicationContext(Class<?>... componentClasses) {
    /**
    * 调用无参构造函数
    * 主要分三步:
    *    a.调用父类构造函数
    *    b.本类构造函数 初始化注解模式下的 bean定义读取器 AnnotatedBeanDefinitionReader
    *    c.本类构造函数 初始化 classPath类型的 bean定义扫描器 AnnotatedBeanDefinitionScaner
    */
    this();
    /**
     * 注册配置类
     * 把传入的类进行注册,分为两种情况
     *     a. @Configuration的配置类
     *     b. 传入普通 Bean (基本不会这么做)
     *  Spring把配置类分为两种
     *     a. 带@Configuration注解的配置类称之为FULL配置类
     *     b. 不带@Configuration注解,是带有@Component,@Import,@ImportResouce,
     *        @Service, @ComponentScan等注解的配置类称之为Lite配置类
     */
    register(componentClasses);
    // 刷新IOC容器
    refresh();
}


this()方法分析开始

第二步:实例化工厂:DefaultListableBeanFactory

DefaultListableBeanFactory 就是我们所说的容器,里面放着beanDefinitionMap, beanDefinitionNames等

// 调用无参构造,会先调用父类GenericApplicationContext的构造函数
// 第一步调用父类构造函数,创建一个Bean工厂
public GenericApplicationContext() {
    /**
    * 调用父类的构造函数,为 ApplicationContext spring 上下文对象初始 beanFactory
    * 因为 DefaultListableBeanFactory 是最底层的实现,功能是最全的
    */
    this.beanFactory = new DefaultListableBeanFactory();
}

// 第二三步
 public class AnnotationConfigApplicationContext extends GenericApplicationContext 
     implements AnnotationConfigRegistry {
     // 注解bean定义读取器,主要作用是用来读取被注解的Bean
     private final AnnotatedBeanDefinitionReader reader;
     // 外部调用scan手动扫描的scanner对象,用处不大
     private final ClassPathBeanDefinitionScanner scanner;

     public AnnotationConfigApplicationContext() {
        /**
         * 初始化注解模式下的bean定义扫描器
         * 调用AnnotatedBeanDefinitionReader构造方法,传入的
         * 是this(AnnotationConfigApplicationContext)对象
         */
        this.reader = new AnnotatedBeanDefinitionReader(this);
        /**
        * 初始化classPath类型的bean定义扫描器
        * 使用场景极少,仅外部手动调用扫描使用,常规方式是不会用到scanner对象的
        * 此处扫描器仅用于自定义的扫描 applicationContext.scan();
        */
        this.scanner = new ClassPathBeanDefinitionScanner(this);
    }
 }

第三步:实例化 BeanDefinition 读取器 (AnnotatedBeanDefinitionReader)

主要就做了两件事情:

  • 注册内置 BeanPostProcessor
  • 注册内置相关核心的 BeanDefinition
public AnnotatedBeanDefinitionReader(BeanDefinitionRegistry registry, Environment environment {
   Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
   Assert.notNull(environment, "Environment must not be null");
   // 把ApplicationContext对象赋值给AnnotatedBeanDefinitionReader
   this.registry = registry;
   // 用户处理条件表达式计算 @Condition
   this.conditionEvaluator = new ConditionEvaluator(registry, environment, null);
   // 注册一些配置的后置处理器,并注册Spring内置的多个Bean
   AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry);
}                               

关于registerAnnotationConfigProcessors(this.registry);方法内容较多,但大多相同的判断,注册 Spring 内置的多个 Bean,以ConfigurationClassPostProcesso为例:

/**
 * 为容器中注册解析配置类的后置处理器 ConfigurationClassPostProcessor
 * org.springframework.context.annotation.internalConfigurationAnnotationProcessor
 */
if (!registry.containsBeanDefinition(CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME)) {
   RootBeanDefinition def = new RootBeanDefinition(ConfigurationClassPostProcessor.class);
   def.setSource(source);
   beanDefs.add(registerPostProcessor(registry, def, CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME));
}

/**
 * 这方法为BeanDefinition设置了一个Role,ROLE_INFRASTRUCTURE代表这是spring内部的,并非用户定义的
 * registry.registerBeanDefinition(beanName, definition);是一个接口方法,
 * 实现类是 DefaultListableBeanFactory
 *    核心工作就是
 *    a.this.beanDefinitionMap.put(beanName, beanDefinition);
 *      把beanName作为key,beanDefinition作为value,放到map里面
 *    b.beanDefinitionNames就是一个List<String>,这里就是把beanName放到List中去
 * DefaultListableBeanFactory就是我们所说的容器,里面放着beanDefinitionMap, beanDefinitionNames,
 *      beanDefinitionMap是一个hashMap,beanName作为Key,beanDefinition作为Value,
 *      beanDefinitionNames是一个集合,里面存放了beanName
 */
private static BeanDefinitionHolder registerPostProcessor(
    BeanDefinitionRegistry registry, RootBeanDefinition definition, String beanName) {

    definition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
    // 核心已在上方注释
    registry.registerBeanDefinition(beanName, definition);
    return new BeanDefinitionHolder(definition, beanName);
}

逻辑就是:

  1. 判断容器中是否已经存在了ConfigurationClassPostProcessorBean
  2. 如果不存在,就通过 RootBeanDefinition 的构造方法获得ConfigurationClassPostProcessor的BeanDefinition
  3. 执行registerPostProcessor()方法,其内部就是注册Bean(与其他 Bean 注册流程一致)

internalConfigurationAnnotationProcessor

ConfigurationClassPostProcessor 是 Spring 中极其重要的一个类,它实现 BeanDefinitionRegistryPostProcessor 接口,BeanDefinitionRegistryPostProcessor 接口又扩展了 BeanFactoryPostProcessor 接口,BeanFactoryPostProcessor 是 Spring 的扩展点之一

至此加载完以下扩展点(beanDefinition -> beanDefinitionMap)

第四步:创建 BeanDefinition 扫描器 (ClassPathBeanDefinitionScanner)

初始化 classPath 类型的 BeanDefinition 扫描器,使用场景极少,仅外部手动调用扫描使用,常规方式是不会用到 scanner 对象的此处扫描器仅用于自定义的扫描applicationContext.scan()

至此this()方法结束


register(annotatedClasses);分析开始

第五步:注册配置类为BeanDefinition (register(annotatedClasses);)

/**
 * 注册配置类
 * 把传入的类进行注册,分为两种情况
 *     a. @Configuration的配置类
 *     b. 传入普通 Bean (基本不会这么做)
 * Spring把配置类分为两种
 *     a. 带@Configuration注解的配置类称之为FULL配置类
 *     b. 不带@Configuration注解,是带有@Component,@Import,@ImportResouce,
 *        @Service, @ComponentScan等注解的配置类称之为Lite配置类
 */
register(componentClasses);

public void register(Class<?>... componentClasses) {
    this.reader.register(componentClasses);
}

private <T> void doRegisterBean(Class<T> beanClass, @Nullable String name,
      @Nullable Class<? extends Annotation>[] qualifiers, @Nullable Supplier<T> supplier,
      @Nullable BeanDefinitionCustomizer[] customizers) {
   // AnnotatedGenericBeanDefinition可以理解为一种数据结构,是用来描述Bean的,这里的作用就是把传入
   // 的标记了注解的类转为AnnotatedGenericBeanDefinition数据结构,里面有一个getMetadata方法,可以拿到类上的注解
   AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(beanClass);
   // 判断是否需要跳过注解,spring中有一个@Condition注解,当不满足条件,这个bean就不会被解析
   if (this.conditionEvaluator.shouldSkip(abd.getMetadata())) {
      return;
   }

   abd.setInstanceSupplier(supplier);
   // 解析bean的作用域,如果没有设置的话,默认为单例
   ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(abd);
   abd.setScope(scopeMetadata.getScopeName());
   // 获得beanName
   String beanName = (name != null ? name : this.beanNameGenerator.generateBeanName(abd, this.registry));
   // 解析通用注解,填充到AnnotatedGenericBeanDefinition,
   // 解析的注解 Lazy,Primary,DependsOn,Role,Description
   AnnotationConfigUtils.processCommonDefinitionAnnotations(abd);
   if (qualifiers != null) {
      for (Class<? extends Annotation> qualifier : qualifiers) {
         if (Primary.class == qualifier) {
            abd.setPrimary(true);
         }
         else if (Lazy.class == qualifier) {
            abd.setLazyInit(true);
         }
         else {
            abd.addQualifier(new AutowireCandidateQualifier(qualifier));
         }
      }
   }
   if (customizers != null) {
      for (BeanDefinitionCustomizer customizer : customizers) {
         customizer.customize(abd);
      }
   }
   // 这个方法用处不大,就是把AnnotatedGenericBeanDefinition数据结构和beanName封装到一个对象中
   BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName);
   definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
   // 注册,最终会调用DefaultListableBeanFactory中的registerBeanDefinition方法去注册
   // DefaultListableBeanFactory维护着一系列信息,比如beanDefinitionNames,beanDefinitionMap
   // beanDefinitionMap是一个Map,用来保存beanName和beanDefinition
   BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, this.registry);
}

  1. 通过AnnotatedGenericBeanDefinition的构造方法,获得配置类的BeanDefinition
  2. 判断需不需要跳过注册,Spring中有一个@Condition注解,如果不满足条件,就会跳过这个类的注册
  3. 然后是解析作用域,如果没有设置的话,默认为单例
  4. 获得BeanName
  5. 解析通用注解,填充到 AnnotatedGenericBeanDefinition,解析的注解为Lazy,Primary,DependsOn,Role,Description
  6. 限定符处理,不是特指@Qualifier注解,也有可能是Primary,或者是Lazy,或者是其他(理论上是任何注解,这里没有判断注解的有效性)
  7. 把AnnotatedGenericBeanDefinition数据结构和beanName封装到一个对象中(不重要,方便传参)
  8. 注册,最终会调用 DefaultListableBeanFactory 中的 registerBeanDefinition() 方法

至此,注册配置类加载结束,配置类(MainStartTest,标记了@Configuration的类) 被放入 BeanDefinitionMap 中未实例化。(实例化都在reflesh()方法中进行)

至此register()方法结束,将我们传入的配置类加载完毕即:mainStartTest


refresh() 方法分析开始

第六步:refresh();

到这一步 Spring 还没有进行扫描,只是实例化了一个工厂,注册了一些内置的 Bean 和 配置类,这一行是至关重要的一个方法,也是内容最多的,里面做了大量的处理。

@Override
public void refresh() throws BeansException, IllegalStateException {
   synchronized (this.startupShutdownMonitor) {
      // 1:准备刷新上下文环境
      // 刷新预处理,和主流程关系不大,就是保存了容器的启动时间,启动标志等
      prepareRefresh();

      //2:告诉子类初始化Bean工厂(MVC),获取Bean工厂
      // 和主流程关系也不大,最终获得了DefaultListableBeanFactory
      ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();

      // 3:对Bean工厂进行填充属性
      /**
       * ①添加了两个后置处理器:
       *    a.ApplicationContextAwareProcessor
       *    b.ApplicationListenerDetector
       * ②设置忽略自动装配和允许自动装配的接口,如果不存在某个bean的时候,
       *  spring就自动注册singleton bean
       * ③ 设置了bean表达式解析器
       */
      prepareBeanFactory(beanFactory);

      try {
         // 4:空方法 留给子类去实现该接口 允许在上下文子类中对Bean工厂进行后置处理。
         postProcessBeanFactory(beanFactory);

         // 5:调用Bean工厂的后置处理器.
         // 执行自定义的BeanFactoryPostProcessor和内置的BeanFactoryPostProcessor
         invokeBeanFactoryPostProcessors(beanFactory);

         // 6:注册BeanPostProcessors
         registerBeanPostProcessors(beanFactory);

         // 7:初始化国际化资源处理器.
         initMessageSource();

         // 8:创建事件多播器
         initApplicationEventMulticaster();

         // 9:这个方法同样也是留个子类实现的springboot也是从这个方法进行启动tomcat的.
         // 模板方法,在容器刷新的时候可以自定义逻辑,不同的Spring容器做不同的事情
         onRefresh();

         // 10:将事件监听器注册到多播器上
         registerListeners();

         // 11:实例化懒加载单例Bean的,也就是Bean绝大部分都是在这里被创建出来的
         finishBeanFactoryInitialization(beanFactory);

         // 12:最后容器刷新 发布刷新事件(Spring cloud也是从这里启动的)
         finishRefresh();
      }

      catch (BeansException ex) {
         if (logger.isWarnEnabled()) {
            logger.warn("Exception encountered during context initialization - " +
                  "cancelling refresh attempt: " + ex);
         }

         // Destroy already created singletons to avoid dangling resources.
         destroyBeans();

         // Reset 'active' flag.
         cancelRefresh(ex);

         throw ex;
      }

      finally {
         // 清除元数据缓冲,实例化后就不需要了
         resetCommonCaches();
      }
   }
}

下面来逐一解析里面比较重要的几个方法,有些不重要的就仅在上面注释了

6.1 prepareBeanFactory(beanFactory)

顾名思义,BeanFactory的一些准备工作

  1. 设置了一个类加载器
  2. 设置了bean表达式解析器
  3. 添加了属性编辑器的支持
  4. 添加了一个后置处理器:ApplicationContextAwareProcessor,此后置处理器实现了BeanPostProcessor接口
  5. 设置了一些忽略自动装配的接口
  6. 设置了一些允许自动装配的接口,并且进行了赋值操作
  7. 在容器中还没有XX的 bean 的时候,帮我们注册 beanName 为 XX 的 singleton bean

6.2 invokeBeanFactoryPostProcessors(beanFactory)

首先看一下我们到这一步时 BeanDefinitionMap 里面 bean 定义的情况:

/**
 * 调用Bean工厂的后置处理器.
 * 执行自定义的 BeanFactoryPostProcessor 和内置的 BeanFactoryPostProcessor
 */
protected void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory) {
    // getBeanFactoryPostProcessors(),获得外部可以手动添加一个后置处理器,如果不添加获得的集合永远为空
   PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors());
   // Detect a LoadTimeWeaver and prepare for weaving, if found in the meantime
   // (例如通过ConfigurationClassPostProcessor注册的@Bean方法)
   if (beanFactory.getTempClassLoader() == null && beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) {
      beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory));
      beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader()));
   }
}

public static void invokeBeanFactoryPostProcessors(
      ConfigurableListableBeanFactory beanFactory, List<BeanFactoryPostProcessor> beanFactoryPostProcessors) {

   // 第一步:首先调用BeanDefinitionRegistryPostProcessor的后置处理器
   // 装beanName 后续会根据这个集合来判断处理器是否已经被执行过了
   Set<String> processedBeans = new HashSet<>();

   if (beanFactory instanceof BeanDefinitionRegistry) {
      // 强行把bean工厂转为BeanDefinitionRegistry
      BeanDefinitionRegistry registry = (BeanDefinitionRegistry) beanFactory;
      // 保存BeanFactoryPostProcessor类型的后置
      List<BeanFactoryPostProcessor> regularPostProcessors = new ArrayList<>();
      List<BeanDefinitionRegistryPostProcessor> registryProcessors = new ArrayList<>();
      // 循环传递进来的 beanFactoryPostProcessors,正常情况为数据,只有手动添加了后置处理器才会有数据
      for (BeanFactoryPostProcessor postProcessor : beanFactoryPostProcessors) {
         // 判断后置处理器是不是 BeanDefinitionRegistryPostProcessor
         // 因为BeanDefinitionRegistryPostProcessor扩展了BeanFactoryPostProcessor
         if (postProcessor instanceof BeanDefinitionRegistryPostProcessor) {
            // 进行强制转换
            BeanDefinitionRegistryPostProcessor registryProcessor =
                  (BeanDefinitionRegistryPostProcessor) postProcessor;
            // 调用作为BeanDefinitionRegistryPostProcessor的处理器的后置方法
            registryProcessor.postProcessBeanDefinitionRegistry(registry);
            // 添加到用于保存的BeanDefinitionRegistryPostProcessor的集合中
            registryProcessors.add(registryProcessor);
         }
         // 若没有实现BeanDefinitionRegistryPostProcessor 接口,那么它就是BeanFactoryPostProcessor
         // 把当前的后置处理器加入到regularPostProcessors中
         else {
            regularPostProcessors.add(postProcessor);
         }
      }

      // 定义一个集合用户保存当前准备创建的BeanDefinitionRegistryPostProcessor
      List<BeanDefinitionRegistryPostProcessor> currentRegistryProcessors = new ArrayList<>();

      // 第一步:去容器中获取BeanDefinitionRegistryPostProcessor的bean的处理器名称
      // internalConfigurationAnnotationProcessor即ConfigurationAnnotationProcessor
      String[] postProcessorNames =
            beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);
      // 循环上一步获取的BeanDefinitionRegistryPostProcessor的类型名称
      for (String ppName : postProcessorNames) {
         // 判断是否实现了PriorityOrdered接口的
         if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) {
            // 显示的调用getBean()的方式获取出该对象然后加入到currentRegistryProcessors集合中去
            currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));
            // 同时也加入到processedBeans集合中去
            // 后续会根据这个集合来判断处理器是否已经被执行过了
            processedBeans.add(ppName);
         }
      }
       // 对currentRegistryProcessors集合中BeanDefinitionRegistryPostProcessor进行排序
      sortPostProcessors(currentRegistryProcessors, beanFactory);
       // 把他加入到用于保存到registryProcessors中
       // 为什么要合并,因为registryProcessors是装载BeanDefinitionRegistryPostProcessor的
       // 一开始的时候,spring只会执行BeanDefinitionRegistryPostProcessor独有的方法
       // 而不会执行BeanDefinitionRegistryPostProcessor父类的方法,即BeanFactoryProcessor的方法
       // 所以这里需要把处理器放入一个集合中,后续统一执行父类的方法
      registryProcessors.addAll(currentRegistryProcessors);
      /**
       * 在这里典型的BeanDefinitionRegistryPostProcessor就是
       * ConfigurationClassPostProcessor
       * 用于进行bean定义的加载 比如我们的包扫描,@import 等等
       */
       // Spring热插播的体现,像ConfigurationClassPostProcessor就相当于一个组件
       // Spring很多事情就是交给组件去管理,如果不想用这个组件,直接去掉注册组件就行
      invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);
      // 调用完之后,马上clear掉,临时变量需要清除
      // list.clear()只清除对象的引用使其变为垃圾,与list = null 集合也会置空
      currentRegistryProcessors.clear();

      // 接下来,去容器中获取BeanDefinitionRegistryPostProcessor的bean的处理器名称
      postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);
      // 循环上一步获取的BeanDefinitionRegistryPostProcessor的类型名称
      for (String ppName : postProcessorNames) {
         // 没有被处理过,且实现了Ordered接口的
         if (!processedBeans.contains(ppName) && beanFactory.isTypeMatch(ppName, Ordered.class)) {
            // 显示的调用getBean()的方式获取出该对象然后加入到currentRegistryProcessors集合中去
            currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));
            // 同时也加入到processedBeans集合中去
            processedBeans.add(ppName);
         }
      }
      // 对currentRegistryProcessors集合中BeanDefinitionRegistryPostProcessor进行排序
      sortPostProcessors(currentRegistryProcessors, beanFactory);
      // 把他加入到用于保存到registryProcessors中
      registryProcessors.addAll(currentRegistryProcessors);
      // 调用他的后置处理方法
      invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);
      // 调用完之后,马上clear掉
      currentRegistryProcessors.clear();

      // 调用没有实现任何优先级接口的BeanDefinitionRegistryPostProcessor
      // 定义一个重复处理的开关变量 默认值为true
      // Finally, invoke all other BeanDefinitionRegistryPostProcessors until no further ones appear.
      boolean reiterate = true;
      // 第一次就可以进来
      while (reiterate) {
         // 进入循环马上把开关变量给改为fasle
         reiterate = false;
         // 去容器中获取BeanDefinitionRegistryPostProcessor的bean的处理器名称
         // 根据类型查 beanName 一般情况下只会获取到一个
org.springframework.context.annotation.internalConfigurationAnnotationProcessor,
          也就是 ConfigurationAnnotationProcessor
         postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);
         // 循环上一步获取的BeanDefinitionRegistryPostProcessor的类型名称
         for (String ppName : postProcessorNames) {
            // 没有被处理过的
            if (!processedBeans.contains(ppName)) {
               // 显示的调用getBean()的方式获取出该对象然后加入到currentRegistryProcessors集合中去
               currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class));
               // 同时也加入到processedBeans集合中去
               processedBeans.add(ppName);
               // 再次设置为true
               reiterate = true;
            }
         }
         // 对currentRegistryProcessors集合中BeanDefinitionRegistryPostProcessor进行排序
         sortPostProcessors(currentRegistryProcessors, beanFactory);
         // 把他加入到用于保存到registryProcessors中
         registryProcessors.addAll(currentRegistryProcessors);
         // 调用他的后置处理方法
         invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);
         // 调用完之后,马上clear掉
         currentRegistryProcessors.clear();
      }

      // 调用实现了BeanDefinitionRegistryPostProcessor的接口 他是他也同时实现了BeanFactoryPostProcessor的方法
      invokeBeanFactoryPostProcessors(registryProcessors, beanFactory);
      // 调用BeanFactoryPostProcessor成品的不是通过getBean的
      invokeBeanFactoryPostProcessors(regularPostProcessors, beanFactory);
   }

   else {
      // 若当前的beanFactory没有实现了BeanDefinitionRegistry 直接调用
      // 直接调用 beanFactoryPostProcessor 接口的方法进行后置处理
      invokeBeanFactoryPostProcessors(beanFactoryPostProcessors, beanFactory);
   }

   // 获取容器中所有的 BeanFactoryPostProcessor
   String[] postProcessorNames =
         beanFactory.getBeanNamesForType(BeanFactoryPostProcessor.class, true, false);

   // 保存BeanFactoryPostProcessor类型实现了priorityOrdered
   List<BeanFactoryPostProcessor> priorityOrderedPostProcessors = new ArrayList<>();
   // 保存BeanFactoryPostProcessor类型实现了Ordered接口的
   List<String> orderedPostProcessorNames = new ArrayList<>();
   // 保存BeanFactoryPostProcessor没有实现任何优先级接口的
   List<String> nonOrderedPostProcessorNames = new ArrayList<>();
   for (String ppName : postProcessorNames) {
      // processedBeans包含的话,表示在上面处理BeanDefinitionRegistryPostProcessor的时候处理过了
      if (processedBeans.contains(ppName)) {
         // skip - already processed in first phase above
      }
      // 判断是否实现了PriorityOrdered
      else if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) {
         priorityOrderedPostProcessors.add(beanFactory.getBean(ppName, BeanFactoryPostProcessor.class));
      }
      // 判断是否实现了Ordered
      else if (beanFactory.isTypeMatch(ppName, Ordered.class)) {
         orderedPostProcessorNames.add(ppName);
      }
      // 没有实现任何的优先级接口的
      else {
         nonOrderedPostProcessorNames.add(ppName);
      }
   }

   // 首先,先调用BeanFactoryPostProcessor实现了 PriorityOrdered接口的
   sortPostProcessors(priorityOrderedPostProcessors, beanFactory);
   invokeBeanFactoryPostProcessors(priorityOrderedPostProcessors, beanFactory);

   // 再调用BeanFactoryPostProcessor实现了 Ordered.
   List<BeanFactoryPostProcessor> orderedPostProcessors = new ArrayList<>(orderedPostProcessorNames.size());
   for (String postProcessorName : orderedPostProcessorNames) {
      orderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class));
   }
   sortPostProcessors(orderedPostProcessors, beanFactory);
   invokeBeanFactoryPostProcessors(orderedPostProcessors, beanFactory);

   // 最后调用没有实现任何方法接口的
   List<BeanFactoryPostProcessor> nonOrderedPostProcessors = new ArrayList<>(nonOrderedPostProcessorNames.size());
   for (String postProcessorName : nonOrderedPostProcessorNames) {
      nonOrderedPostProcessors.add(beanFactory.getBean(postProcessorName, BeanFactoryPostProcessor.class));
   }
   invokeBeanFactoryPostProcessors(nonOrderedPostProcessors, beanFactory);

   // Clear cached merged bean definitions since the post-processors might have
   // modified the original metadata, e.g. replacing placeholders in values...
   beanFactory.clearMetadataCache();
}

总结:

首先,之前已经了解过 BeanDefinition 的两个扩展点 postProcessBeanFactorypostProcessBeanDefinitionRegistry。前者是可以修改 BeanDefinition,重写方法,后者可以多添加 BeanDefinition

1、定义了一个 Set,processedBeans 装载BeanName,后面会根据此 Set 来判断后置处理器是否被执行过

2、判断当前的 beanFactory 有没有实现 BeanDefinitionRegistry,当然是肯定的, 定义了两个 List 一个是 regularPostProcessors,用来装载 BeanFactoryPostProcessor。它只有一个实现 方法postProcessBeanFactory()。一个是 registryProcessors, 用来装载 BeanDefinitionRegistryPostProcessor。因为 它继承了 BeanFactoryPostProcessor 它不仅有postProcessBeanFactory()还有postProcessBeanDefinitionRegistry()

3、循环传进来的beanFactoryPostProcessors,一般情况下都是空的,除非自己 add 了 beanFactory 的后置处理器。假设有数据,先判断是否是BeanDefinitionRegistryPostProcessor如果是调用postProcessBeanDefinitionRegistry()方法,并添加到集合 registryProcessors 中,否的话直接加入到集合 regularPostProcessors。(postProcessBeanFactory()会在后面执行,先存起来)

4、 定义一个集合(List 临时变量) currentRegistryProcessors用户保存当前准备创建的 BeanDefinitionRegistryPostProcessor

5、去容器中获取BeanDefinitionRegistryPostProcessor的bean的处理器名称internalConfigurationAnnotationProcessorConfigurationAnnotationProcessor一般情况都只会获取到一个。此时Spring还未扫描完成,扫描是在ConfigurationClassPostProcessor类完成的,就是下面的第一个invokeBeanDefinitionRegistryPostProcessors()方法

6、循环 postProcessorNames internalConfigurationAnnotationProcessor判断是否实现了 PriorityOrdered,实现了添加到currentRegistryProcessors和processedBeans表示它们被处理过了(下一步才处理)

7、对 currentRegistryProcessors 集合中BeanDefinitionRegistryPostProcessor 进行排序

8、将currentRegistryProcessors集合加到registryProcessors集合中,因为registryProcessors是装载BeanDefinitionRegistryPostProcessor,一开始的时候,spring只会执行BeanDefinitionRegistryPostProcessor独有的方法postProcessBeanDefinitionRegistry()。而不会执行 BeanDefinitionRegistryPostProcessor 父类的方法,即 BeanFactoryProcessor 的方法postProcessBeanFactory()。所有在此统一放到一起等待后续执行

9、internalConfigurationAnnotationProcessor(currentRegistryProcessors, registry),执行currentRegistryProcessors中的ConfigurationClassPostProcessor中的postProcessBeanDefinitionRegistry()方法,这里体现了 Spring 中热插拔,插件化开发的思想,如果不想用这个,不添加就行了。从下图可以看到,执行完该方法后 bean定义 被加载到了BeanDefinitionMap中

10、清空currentRegistryProcessors,用完了就需要清空,给后面的其他的重复使用

11、最后会重复上面的逻辑,调用顺序如下:

  • 实现了PriorityOrdered接口的
  • 实现了Ordered接口的
  • 没有实现任何的优先级接口的

如果实现了多个的话,将在最先实现的地方调用,第二次将会判断是否已经处理过

再来看一下postProcessBeanFactory()方法,它调用了一个会解析我们BeanDefinition的方法processConfigBeanDefinitions((BeanDefinitionRegistry) beanFactory);

@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) {
    int factoryId = System.identityHashCode(beanFactory);
    if (this.factoriesPostProcessed.contains(factoryId)) {
        throw new IllegalStateException(
            "postProcessBeanFactory already called on this post-processor against " + beanFactory);
    }
    this.factoriesPostProcessed.add(factoryId);
    if (!this.registriesPostProcessed.contains(factoryId)) {
        // BeanDefinitionRegistryPostProcessor hook apparently not supported...
        // Simply call processConfigurationClasses lazily at this point then.
        processConfigBeanDefinitions((BeanDefinitionRegistry) beanFactory);
    }
    // 为属性为full的Bean定义做CGLIB增强
    enhanceConfigurationClasses(beanFactory);
    beanFactory.addBeanPostProcessor(new ImportAwareBeanPostProcessor(beanFactory));
}

这个方法中会引申出一个知识点

注册配置类 把传入的类进行注册,分为两种情况

  • a. @Configuration 的配置类
  • b. 传入普通 Bean (基本不会这么做)

Spring 把配置类分为两种

  • a. 带 @Configuration 注解的配置类称之为 FULL 配置类
  • b. 不带 @Configuration 注解,而是带有 @Component,@Import,@ImportResouce, @Service, @ComponentScan 等注解的配置类称之为 Lite 配置类

如果我们注册了Full 配置类,我们 getBean 这个配置类,会发现它已经不是原本那个配置类了,而是已经被 CGLIB 代理的类

例如:写一个A类,其中有一个构造方法,打印出“HelloSpring”,再写一个配置类,里面有两个 带 @Bean 的方法。假设其中一个方法getA()new A(),并且返回A的对象。第二个方法又调用了getA()。如果配置类是 Lite 配置类,会发现打印了两次“HelloSpring”,即 A 类被 new 了两次。如果配置类是 FULL 配置类,会发现只打印一次,因为这个类被CGLIB代理了。

public void processConfigBeanDefinitions(BeanDefinitionRegistry registry) {
        List<BeanDefinitionHolder> configCandidates = new ArrayList<>();
        // 获取IOC 容器中目前所有bean定义的名称
        String[] candidateNames = registry.getBeanDefinitionNames();

        // 循环上一步获取的所有的Bean定义信息
        for (String beanName : candidateNames) {
            // 通过Bean的名称来获取我们的bean定义对象
            BeanDefinition beanDef = registry.getBeanDefinition(beanName);
            // 判断是否有没有解析过
            if (beanDef.getAttribute(ConfigurationClassUtils.CONFIGURATION_CLASS_ATTRIBUTE) != null) {
                if (logger.isDebugEnabled()) {
                    logger.debug("Bean definition has already been processed as a configuration class: " + beanDef);
                }
            }
            // 进行正在的解析判断是不是完全的配置类 还是一个非正式的配置类
            else if (ConfigurationClassUtils.checkConfigurationClassCandidate(beanDef, this.metadataReaderFactory)) {
                // 满足添加就加入到候选的配置类集合中
                configCandidates.add(new BeanDefinitionHolder(beanDef, beanName));
            }
        }

        // Return immediately if no @Configuration classes were found
        // 若没有找到配置类直接返回
        if (configCandidates.isEmpty()) {
            return;
        }

        // Sort by previously determined @Order value, if applicable
        // 对配置类进行Order排序
        configCandidates.sort((bd1, bd2) -> {
            int i1 = ConfigurationClassUtils.getOrder(bd1.getBeanDefinition());
            int i2 = ConfigurationClassUtils.getOrder(bd2.getBeanDefinition());
            return Integer.compare(i1, i2);
        });

        // Detect any custom bean name generation strategy supplied through the enclosing application context
        // 创建我们通过@CompentScan导入进来的bean name的生成器
        // 创建我们通过@Import导入进来的bean的名称
        SingletonBeanRegistry sbr = null;
        if (registry instanceof SingletonBeanRegistry) {
            sbr = (SingletonBeanRegistry) registry;
            if (!this.localBeanNameGeneratorSet) {
                BeanNameGenerator generator = (BeanNameGenerator) sbr.getSingleton(
                        AnnotationConfigUtils.CONFIGURATION_BEAN_NAME_GENERATOR);
                if (generator != null) {
                    // 设置@CompentScan导入进来的bean的名称生成器
                    this.componentScanBeanNameGenerator = generator;
                    // 设置@Import导入进来的bean的名称生成器
                    this.importBeanNameGenerator = generator;
                }
            }
        }

        if (this.environment == null) {
            this.environment = new StandardEnvironment();
        }

        // 创建一个配置类解析器对象
        // Parse each @Configuration class
        ConfigurationClassParser parser = new ConfigurationClassParser(
                this.metadataReaderFactory, this.problemReporter, this.environment,
                this.resourceLoader, this.componentScanBeanNameGenerator, registry);
        // 创建一个集合用于保存我们的配置类BeanDefinitionHolder集合默认长度是配置类集合的长度
        Set<BeanDefinitionHolder> candidates = new LinkedHashSet<>(configCandidates);
        // 创建一个集合用于保存我们的已经解析的配置类,长度默认为解析出来默认的配置类的集合长度
        Set<ConfigurationClass> alreadyParsed = new HashSet<>(configCandidates.size());
        //do while 会进行第一次解析
        do {
            // 解析配置类
            // 经过这一步,会将@ComponentScans、@ComponentScan、@Bean、@Import等注解要注册的类扫描出来
            parser.parse(candidates);
            parser.validate();
            // 解析出来的配置类
            Set<ConfigurationClass> configClasses = new LinkedHashSet<>(parser.getConfigurationClasses());
            configClasses.removeAll(alreadyParsed);

            // Read the model and create bean definitions based on its content
            if (this.reader == null) {
                this.reader = new ConfigurationClassBeanDefinitionReader(
                        registry, this.sourceExtractor, this.resourceLoader, this.environment,
                        this.importBeanNameGenerator, parser.getImportRegistry());
            }
            // 把解析出来的配置类注册到容器中
            // 经过这一步会将@Bean、@import 注册的类变成BeanDefinition
            this.reader.loadBeanDefinitions(configClasses);
            // 加入到已经解析的集合中
            alreadyParsed.addAll(configClasses);

            candidates.clear();
            //判断我们IOC容器中的是不是>候选原始的bean定义的个数
            if (registry.getBeanDefinitionCount() > candidateNames.length) {
                // 获取所有的bean定义
                String[] newCandidateNames = registry.getBeanDefinitionNames();
                // 原始的老的候选的bean定义
                Set<String> oldCandidateNames = new HashSet<>(Arrays.asList(candidateNames));
                Set<String> alreadyParsedClasses = new HashSet<>();
                // 赋值已经解析的
                for (ConfigurationClass configurationClass : alreadyParsed) {
                    alreadyParsedClasses.add(configurationClass.getMetadata().getClassName());
                }
                for (String candidateName : newCandidateNames) {
                    // 表示当前循环的还没有被解析过
                    if (!oldCandidateNames.contains(candidateName)) {
                        BeanDefinition bd = registry.getBeanDefinition(candidateName);
                        // 判断有没有被解析过
                        // checkConfigurationClassCandidate 此时为Bean定义标识为full或lite,在后面根据属性潘森是否需要用CGLIB增强
                        if (ConfigurationClassUtils.checkConfigurationClassCandidate(bd, this.metadataReaderFactory) &&
                                !alreadyParsedClasses.contains(bd.getBeanClassName())) {
                            candidates.add(new BeanDefinitionHolder(bd, candidateName));
                        }
                    }
                }
                candidateNames = newCandidateNames;
            }
        }
        // 存在没有解析过的 需要循环解析
        while (!candidates.isEmpty());

        // Register the ImportRegistry as a bean in order to support ImportAware @Configuration classes
        if (sbr != null && !sbr.containsSingleton(IMPORT_REGISTRY_BEAN_NAME)) {
            sbr.registerSingleton(IMPORT_REGISTRY_BEAN_NAME, parser.getImportRegistry());
        }

        if (this.metadataReaderFactory instanceof CachingMetadataReaderFactory) {
            // Clear cache in externally provided MetadataReaderFactory; this is a no-op
            // for a shared cache since it'll be cleared by the ApplicationContext.
            ((CachingMetadataReaderFactory) this.metadataReaderFactory).clearCache();
        }
    }

至此自定义的配置类都加载到了beanDefinitonMap 中,但仍未初始化

此处会实例化以下几个内置 Bean

6.3 registerBeanPostProcessors(beanFactory);

实例化和注册 beanFactory 中扩展了 BeanPostProcessor 的bean。

例如:

AutowiredAnnotationBeanPostProcessor(处理被@Autowired注解修饰的bean并注入)

RequiredAnnotationBeanPostProcessor(处理被@Required注解修饰的方法)

CommonAnnotationBeanPostProcessor(处理@PreDestroy、@PostConstruct、@Resource等多个注解的作用)等。

此处会实例化以下几个内置 Bean

6.4 initApplicationEventMulticaster(); 和 registerListeners();

创建事件多播器

注册监听器,广播early application events

后续监听机制再来看这两个

6.5 finishBeanFactoryInitialization(beanFactory);

实例化 非懒加载单例 Bean ,也就是我们的 Bean 都是在这里被创建出来的。包括(实例化、填充属性、初始化)

里面会有一个方法preInstantiateSingletons()是一个接口方法,这个方法只有一个实现类DefaultListableBeanFactory,里面最重要的就是getBean();。这中间还会去判断是否是一个特殊的Bean(即:FactoryBean)一旦一个类实现了 FactoryBean 并从写了getObject() 方法那么,IOC容器拿到的实例就是调用getObject方法得到的特殊的实例,没有实现这个接口时注册到IOC容器中的就是一个普通的Bean,当实现后,IOC容器会调用getObject方法返回的实例(工厂模式)

// 初始化所有的非懒加载单例Bean
beanFactory.preInstantiateSingletons();
// 执行 getBean流程
getBean(beanName);
复制代码

里面调用的是 AbstractBeanFactory.java 里面的getBean();

@Override
public Object getBean(String name) throws BeansException {
   return doGetBean(name, null, null, false);
}

doGetBean 方法太多了,这里挑出主要创建 单例bean 的逻辑

// 创建单例bean
if (mbd.isSingleton()) {
    // 把beanName 和一个 singletonFactory 并且传入一个回调对象用于回调
    sharedInstance = getSingleton(beanName, () -> {
        try {
            // 进入创建bean的逻辑
            return createBean(beanName, mbd, args);
        }
        catch (BeansException ex) {
            // 创建bean的过程中发生异常,需要销毁关于当前bean的所有信息
            destroySingleton(beanName);
            throw ex;
        }
    });
    bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
}

这里的creatBean()又是一个接口方法,但也仅仅只有一个类对其做了实现AbstractAutowireCapableBeanFactory。该方法前面也会进行一大堆的判断,我们再次挑出关键步骤

// 真正的开始创建Bean实例对象
Object beanInstance = doCreateBean(beanName, mbdToUse, args);
if (logger.isTraceEnabled()) {
   logger.trace("Finished creating instance of bean '" + beanName + "'");
}
return beanInstance;

点进方法doCreateBean(beanName, mbdToUse, args);发现也在该类下面,里面又做了一大堆的事情,我们主要调出机构关键点

创建实例

// 使用合适的实例化策略来创建新的实例:工厂方法、构造函数自动注入、简单初始化 
instanceWrapper = createBeanInstance(beanName, mbd, args);

填充属性及初始化

// 给我们的属性进行赋值(调用set方法进行赋值)
populateBean(beanName, mbd, instanceWrapper);
// 进行对象初始化操作(在这里可能生成代理对象)
exposedObject = initializeBean(beanName, exposedObject, mbd);

initializeBean(beanName, exposedObject, mbd);中,又调用了invokeInitMethods(beanName, wrappedBean, mbd);invokeAwareMethods(beanName, bean);

protected Object initializeBean(final String beanName, final Object bean, @Nullable RootBeanDefinition mbd) {
   if (System.getSecurityManager() != null) {
      AccessController.doPrivileged((PrivilegedAction<Object>) () -> {
         invokeAwareMethods(beanName, bean);
         return null;
      }, getAccessControlContext());
   }
   else {
      // 若我们的bean实现了XXXAware接口进行方法的回调
      invokeAwareMethods(beanName, bean);
   }

   Object wrappedBean = bean;
   if (mbd == null || !mbd.isSynthetic()) {
      // 调用我们的bean的后置处理器的postProcessorsBeforeInitialization方法  @PostCust注解的方法
      wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);
   }

   try {
      // 调用初始化方法
      invokeInitMethods(beanName, wrappedBean, mbd);
   }
   catch (Throwable ex) {
      throw new BeanCreationException(
            (mbd != null ? mbd.getResourceDescription() : null),
            beanName, "Invocation of init method failed", ex);
   }
   if (mbd == null || !mbd.isSynthetic()) {
      // 调用我们bean的后置处理器的PostProcessorsAfterInitialization方法
      wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);
   }

   return wrappedBean;
}

private void invokeAwareMethods(final String beanName, final Object bean) {
   if (bean instanceof Aware) {
      // 此bean实现了BeanNameAware
      if (bean instanceof BeanNameAware) {
         ((BeanNameAware) bean).setBeanName(beanName);
      }
      // 实现了BeanClassLoaderAware接口
      if (bean instanceof BeanClassLoaderAware) {
         ClassLoader bcl = getBeanClassLoader();
         if (bcl != null) {
            ((BeanClassLoaderAware) bean).setBeanClassLoader(bcl);
         }
      }
      // 实现了BeanFactoryAware
      if (bean instanceof BeanFactoryAware) {
         ((BeanFactoryAware) bean).setBeanFactory(AbstractAutowireCapableBeanFactory.this);
      }
   }
}

至此剩余的Bean也全部初始化完成至 IOC 容器中

验证Spring Bean 的生命周期

定义一个SpringBean

@ComponentScan
public  class SpringBean
        implements InitializingBean, DisposableBean, BeanNameAware, BeanFactoryAware, BeanClassLoaderAware {

    // 就是一个普通的被@Component标注的类
    @Autowired
    AutoBean autoBean;

    public SpringBean() {
        System.out.println("SpringBean Constructor Method:" + autoBean);
        System.out.println("SpringBean()");
    }

    @Override
    public void setBeanClassLoader(ClassLoader classLoader) {
        System.out.println("ClassLoader");
    }

    @Override
    public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
        System.out.println("setBeanFactory");
    }

    @Override
    public void setBeanName(String name) {
        System.out.println("setBeanName:" + autoBean);
        System.out.println("setBeanName");
    }

    @Override
    public void destroy() throws Exception {
        System.out.println("destroy");
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        System.out.println("afterPropertiesSet");
    }

    public void initMethod() {
        System.out.println("initMethod");
    }

    public void destroyMethod() {
        System.out.println("destroyMethod");
    }
}

再定义一个BeanPostProcessor,在重写的两个方法中进行了判断,如果传进来的 beanName 是 springBean 才进行打印

@Component
public class MyBeanPostProcessor implements BeanPostProcessor {

   @Override
   public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
      if(beanName.equals("springBean")) {
         System.out.println("postProcessBeforeInitialization");
      }
      return bean;
   }

   @Override
   public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
      if(beanName.equals("springBean")) {
         System.out.println("postProcessAfterInitialization");
      }
      return bean;
   }
}

定义一个配置类,完成自动扫描,但是SpringBean是手动注册的,并且声明了initMethod和destroyMethod:

@Configuration
@ComponentScan
public class MainConfig {
   @Bean(initMethod = "initMethod",destroyMethod = "destroyMethod")
   public SpringBean springBean() {
      return new SpringBean();
   }
}

然后是启动类:

public class Main {
   public static void main(String[] args) throws Exception {
       AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(MainConfig.class);
       // Spring 的 destroy() 方法已过时会报错,推荐使用 registerShutdownHook() 优雅的关闭 IOC
       // context.registerShutdownHook() 是一个钩子方法,当jvm关闭退出的时候会调用这个钩子方法
       // 当然 context.close() 也可以销毁容器
       context.registerShutdownHook();
   }
}

运行结果:

由此我们可以推导出Spring的生命周期

  1. 实例化Bean对象,这个时候 Bean 的对象是非常低级的,基本不能够被我们使用,因为连最基本的属性都没有设置,可以理解为 连Autowired注解都是没有解析的
  2. 填充属性,当做完这一步,Bean对象基本是完整的了,可以理解为Autowired注解已经解析完毕,依赖注入完成了
  3. 如果Bean实现了BeanNameAware接口,则调用setBeanName方法
  4. 如果Bean实现了BeanClassLoaderAware接口,则调用setBeanClassLoader方法
  5. 如果Bean实现了BeanFactoryAware接口,则调用setBeanFactory方法
  6. 调用BeanPostProcessor的postProcessBeforeInitialization方法
  7. 如果Bean实现了InitializingBean接口,调用afterPropertiesSet方法
  8. 如果Bean定义了init-method方法,则调用Bean的init-method方法
  9. 调用BeanPostProcessor的postProcessAfterInitialization方法。当进行到这一步,Bean已经被准备就绪了,一直停留在应用的上下文中,直到被销毁
  10. 如果应用的上下文被销毁了,如果Bean实现了DisposableBean接口,则调用destroy方法,如果Bean定义了destory-method声明了销毁方法也会被调用

思考

配置类 @Configuration 加与不加的区别,加上会创建 CGLIB 动态代理,保证配置类中的 Bean 是受 IOC容器控制的,是单例的,如果配置类中重复的用某一个类,不加的话就是是重复调用方法,多次创建,不受 IOC 容器控制。

重复 beanName 覆盖原则,如果是通过 Scanner 扫描到的同一包下两个相同的 beanName 会抛异常,如果一个是 @Compontent 扫描的 bean,一个是通过 @Bean 配置的 bean,那么 @Bean 配置的 bean 会覆盖前面的 bean,因为它是后创建的。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容