17-DefaultSingletonBeanRegistry

背景简介

出现的原因

我们上面介绍了:

  • AliasRegistrySimpleAliasRegistry他们分别定义和实现了对别名管理的支持
  • SingletonBeanRegistry定义了对单例 Bean 管理的支持

按照我们之前对 Spring 整合不通接口功能的套路的介绍,现在该有个类来继承SimpleAliasRegistry同时实现SingletonBeanRegistry接口,从而实现将"对单例 Bean的管理”、“对别名的管理”两个功能的整合。

职责

将"对单例 Bean的管理”、“对别名的管理”两个功能整合。

注意点

思路很简单,主要关注它里面的实例属性都有什么意义。这在后面的解决循环依赖、进行 bean 回收中都有帮助。

源码

继承关系

1.png

实例属性介绍

因为DefaultSingletonBeanRegistry通过继承SimpleAliasRegistry直接拥有了对别名的管理功能,所以在DefaultSingletonBeanRegistry中只需要将SingletonBeanRegistry的相关功能支持一遍即可。

/**
 * Cache of singleton objects: bean name --> bean instance
 * 正经的单例 bean 实例缓存,里面放的都是构造、属性填充完成、初始化好的。
 */
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);

/**
 * Cache of singleton factories: bean name --> ObjectFactory
 * 单例 bean 工厂的缓存,用于实现懒加载,需要时就拿出来,用 ObjectFactory 创建 Bean 实例并放到
 * earlySingletonObjects 等待属性填充和初始化
 */
private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);

/**
 * Cache of early singleton objects: bean name --> bean instance
 * 用于提前暴露 Bean 实例的地址,用于解决单例 bean 的循环依赖问题,这里面的东西都未执行属性填充和初始化
 */
private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);

/**
 * Set of registered singletons, containing the bean names in registration order
 * 注册的单例 Bean 的id集合,等价于 singletonObjects.keySet()
 */
//TODO 没必要专门用个集合来存储,这样维护一致性要花费不少精力
private final Set<String> registeredSingletons = new LinkedHashSet<>(256);

/**
 * 当前正在创建中的 Bean ,在提供的扩展方法——“通过一个具有创造实例、填充属性、初始化功能的工厂方法
 * 直接获得最终单例Bean”中用作记录
 * 
 */
private final Set<String> singletonsCurrentlyInCreation =
        Collections.newSetFromMap(new ConcurrentHashMap<>(16));

/**
 * 要排除创建标记的Bean
 */
private final Set<String> inCreationCheckExclusions =
        Collections.newSetFromMap(new ConcurrentHashMap<>(16));
/**
 * Disposable bean instances: bean name --> disposable instance
 */
// 这里面的bean有删除的钩子,在删除 bean 时,从注册中删除后需要调用这里的方法 
// 这里的 disposableBeans 的 key 和要回收的 bean 实例在 singletonObjects 中的 key 一样
private final Map<String, Object> disposableBeans = new LinkedHashMap<>();
/**
 * Map between containing bean names: bean name --> Set of bean names that the bean contains
 */
// key 对应的 bean 里面有变量是用的 value 中的 bean
private final Map<String, Set<String>> containedBeanMap = new ConcurrentHashMap<>(16);
/**
 * Map between dependent bean names: bean name --> Set of dependent bean names
 */
// key 对应的 bean 被 value 中的 bean 依赖,也就是说 value 中的 bean 销毁完 ,key 对应的 bean 才能开始销毁
private final Map<String, Set<String>> dependentBeanMap = new ConcurrentHashMap<>(64);
/**
 * Map between depending bean names: bean name --> Set of bean names for the bean's dependencies
 */
// key 对应的 bean 依赖了 value 中的 bean,也就是说 key 对应的 bean 销毁完,value 中的 bean 才能销毁
private final Map<String, Set<String>> dependenciesForBeanMap = new ConcurrentHashMap<>(64);
/**
 * List of suppressed Exceptions, available for associating related causes
 */
@Nullable
private Set<Exception> suppressedExceptions;
/**
 * 标记值,如果在销毁阶段,不允许新注册单例 Bean
 */
private boolean singletonsCurrentlyInDestruction = false;

实现的接口方法——增

// 将 Bean 实例注册到 singletonObjects 中
@Override
public void registerSingleton(String beanName, Object singletonObject) throws IllegalStateException {
    Assert.notNull(beanName, "Bean name must not be null");
    Assert.notNull(singletonObject, "Singleton object must not be null");
    synchronized (this.singletonObjects) {
        Object oldObject = this.singletonObjects.get(beanName);
        if (oldObject != null) {
            throw new IllegalStateException("Could not register object [" + singletonObject +
                    "] under bean name '" + beanName + "': there is already object [" + oldObject + "] bound");
        }
        addSingleton(beanName, singletonObject);
    }
}

/**
 * Add the given singleton object to the singleton cache of this factory.
 * <p>To be called for eager registration of singletons.
 *
 * @param beanName        the name of the bean
 * @param singletonObject the singleton object
 */
protected void addSingleton(String beanName, Object singletonObject) {
    synchronized (this.singletonObjects) {
        this.singletonObjects.put(beanName, singletonObject);
        this.singletonFactories.remove(beanName);
        this.earlySingletonObjects.remove(beanName);
        this.registeredSingletons.add(beanName);
    }
}

实现的接口方法——查

@Override
@Nullable
// 根据 bean 的 name(这里指的是 ID ,不是 alias )获得对应的 bean 实例
public Object getSingleton(String beanName) {
    return getSingleton(beanName, true);
}

/**
 * 返回 id 对应的实例,可能是正经的实例,也可能是为了解决循环依赖提前暴露出的实例地址
 * 可以通过入参控制是否在没有得到实例的情况下提前创造实例返回地址
 *
 * @param beanName            the name of the bean to look for
 * @param allowEarlyReference 是否创造提前暴露的实例地址
 * @return the registered singleton object, or {@code null} if none found
 */
@Nullable
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
    Object singletonObject = this.singletonObjects.get(beanName); // 先看正经缓存中有没有弄好的
    if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
    // 没拿到,而且正在通过扩展的工厂方法进行获取最终实例属性
        synchronized (this.singletonObjects) {
            singletonObject = this.earlySingletonObjects.get(beanName); // 看工厂方法有没有提前丢到这里
            if (singletonObject == null && allowEarlyReference) {// 没有丢这里,看看我们给他创建个实例
                ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
                if (singletonFactory != null) {// 有创建实例的工厂,就搞一下
                    singletonObject = singletonFactory.getObject();
                    this.earlySingletonObjects.put(beanName, singletonObject);
                    this.singletonFactories.remove(beanName);
                }
            }
        }
    }
    return singletonObject;
}

/**
 * 正在通过外部工厂方法进行从头到尾的创建
 * @param beanName the name of the bean
 */
public boolean isSingletonCurrentlyInCreation(String beanName) {
    return this.singletonsCurrentlyInCreation.contains(beanName);
}


/**
 * Exposes the singleton mutex to subclasses and external collaborators.
 * <p>Subclasses should synchronize on the given Object if they perform
 * any sort of extended singleton creation phase. In particular, subclasses
 * should <i>not</i> have their own mutexes involved in singleton creation,
 * to avoid the potential for deadlocks in lazy-init situations.
 */
public final Object getSingletonMutex() {
    return this.singletonObjects;
}

@Override
public boolean containsSingleton(String beanName) {
    return this.singletonObjects.containsKey(beanName);
}

@Override
public String[] getSingletonNames() {
    synchronized (this.singletonObjects) {
        return StringUtils.toStringArray(this.registeredSingletons);
    }
}

@Override
public int getSingletonCount() {
    synchronized (this.singletonObjects) {
        return this.registeredSingletons.size();
    }
}

扩展方法——工厂懒加载功能【此工厂仅创建实例】

/**
 * 把用来创建 bean 实例的工厂方法保存至 singletonFactories 中
 * TODO : 注意了,这个 singletonFactory 仅创建实例,用来解决循环依赖,不负责初始化、填充属性什么的
 *
 * @param beanName         the name of the bean
 * @param singletonFactory the factory for the singleton object
 */
protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {
  Assert.notNull(singletonFactory, "Singleton factory must not be null");
  synchronized (this.singletonObjects) {
    if (!this.singletonObjects.containsKey(beanName)) {
      this.singletonFactories.put(beanName, singletonFactory);
      this.earlySingletonObjects.remove(beanName);
      this.registeredSingletons.add(beanName);
    }
  }
}

扩展方法——完全委托外部完成实例所有创建操作

/**
 * Return the (raw) singleton object registered under the given name,
 * creating and registering a new one if none registered yet.
 *
 * @param beanName         the name of the bean
 * @param singletonFactory the ObjectFactory to lazily create the singleton
 *                         with, if necessary
 * @return the registered singleton object
 */
// 创建 beanName 对应的单例 bean ,如果还没创建就创建,创建过就直接返回,
//
// 第二个入参不是懒加载,在这里是直接调用创建然后放到正经缓存中去来。
// 使用表达式入参的原因之一可能是方便用户自定义逻辑,包括使用闭包访问自定义变量什么的。
public Object getSingleton(String beanName, ObjectFactory<?> singletonFactory) {
    Assert.notNull(beanName, "Bean name must not be null");
    synchronized (this.singletonObjects) {
        Object singletonObject = this.singletonObjects.get(beanName);
        if (singletonObject == null) { // 缓存中没有,说明之前没创建过,那就创建一次
            if (this.singletonsCurrentlyInDestruction) { // 已经进入删除阶段,不允许创建
                throw new BeanCreationNotAllowedException(beanName,
                        "Singleton bean creation not allowed while singletons of this factory are in destruction " +
                                "(Do not request a bean from a BeanFactory in a destroy method implementation!)");
            }
            if (logger.isDebugEnabled()) {
                logger.debug("Creating shared instance of singleton bean '" + beanName + "'");
            }
      // 创建前的缓存标记——标记此 beanName 正在使用外部方法创建中
            beforeSingletonCreation(beanName); 
            boolean newSingleton = false;
            boolean recordSuppressedExceptions = (this.suppressedExceptions == null);
            if (recordSuppressedExceptions) {
                this.suppressedExceptions = new LinkedHashSet<>();
            }
            try {
                // 创建 bean 实例,并初始化。
                // 如果总全局角度来看的话,其实这个里面的逻辑就是调用的 creatBean()
                singletonObject = singletonFactory.getObject();
                newSingleton = true;
            } catch (IllegalStateException ex) {
                // Has the singleton object implicitly appeared in the meantime ->
                // if yes, proceed with it since the exception indicates that state.
                singletonObject = this.singletonObjects.get(beanName);
                if (singletonObject == null) {
                    throw ex;
                }
            } catch (BeanCreationException ex) {
                if (recordSuppressedExceptions) {
                    for (Exception suppressedException : this.suppressedExceptions) {
                        ex.addRelatedCause(suppressedException);
                    }
                }
                throw ex;
            } finally {
                if (recordSuppressedExceptions) {
                    this.suppressedExceptions = null;
                }
        // 完成创建,删除标记——将此 beanName 从正在创建的列表中删除
                afterSingletonCreation(beanName);
            }
            if (newSingleton) { // 缓存到正经缓存单例的里面
                addSingleton(beanName, singletonObject);
            }
        }
        return singletonObject;
    }
}


protected void beforeSingletonCreation(String beanName) {
    if (!this.inCreationCheckExclusions.contains(beanName) && !this.singletonsCurrentlyInCreation.add(beanName)) {
        throw new BeanCurrentlyInCreationException(beanName);
    }
}

protected void afterSingletonCreation(String beanName) {
    if (!this.inCreationCheckExclusions.contains(beanName) && !this.singletonsCurrentlyInCreation.remove(beanName)) {
        throw new IllegalStateException("Singleton '" + beanName + "' isn't currently in creation");
    }
}

扩展方法——Bean 之间关系的管理

/**
 * Register a containment relationship between two beans,
 * e.g. between an inner bean and its containing outer bean.
 * <p>Also registers the containing bean as dependent on the contained bean
 * in terms of destruction order.
 *
 * @param containedBeanName  the name of the contained (inner) bean
 * @param containingBeanName the name of the containing (outer) bean
 * @see #registerDependentBean
 */
public void registerContainedBean(String containedBeanName, String containingBeanName) {
    synchronized (this.containedBeanMap) {
        Set<String> containedBeans =
                this.containedBeanMap.computeIfAbsent(containingBeanName, k -> new LinkedHashSet<>(8));
        if (!containedBeans.add(containedBeanName)) {
            return;
        }
    }
    registerDependentBean(containedBeanName, containingBeanName);
}

/**
 * Register a dependent bean for the given bean,
 * to be destroyed before the given bean is destroyed.
 *
 * @param beanName          the name of the bean
 * @param dependentBeanName the name of the dependent bean
 */
public void registerDependentBean(String beanName, String dependentBeanName) {
    String canonicalName = canonicalName(beanName);

    synchronized (this.dependentBeanMap) {
        Set<String> dependentBeans =
                this.dependentBeanMap.computeIfAbsent(canonicalName, k -> new LinkedHashSet<>(8));
        if (!dependentBeans.add(dependentBeanName)) {
            return;
        }
    }

    synchronized (this.dependenciesForBeanMap) {
        Set<String> dependenciesForBean =
                this.dependenciesForBeanMap.computeIfAbsent(dependentBeanName, k -> new LinkedHashSet<>(8));
        dependenciesForBean.add(canonicalName);
    }
}

很容易理解,不在详细介绍,我觉得看实例属性介绍已经够了。

扩展方法——检测依赖是否存在

这个方法常常用来检测循环依赖的存在,其实 Spring 框架是做了针对单例 bean 的解决循环依赖的操作的:通过提前暴露地址解决依赖——这样就不会轮到A—>B—>A时最后回头创建 A,可以直接拿到A实例。

但是上面仅仅是解决属性上的循环依赖,如果连实例都创建不出来,这个循环依赖就没法解决了,所以我们提供一个用来检测依赖的工具函数还有有必要的:

扩展:两种导致无法创建实例的循环依赖

  1. dependOn,手动配置,A dependOn B表明B要在A之前完整构建完。这种依赖如果出现循环的话,是一个实例都创建不出来的。
  2. 依赖的传递是用来创建实例的构造函数入参

另外,我们在解决循环依赖失败时也要判断影响范围,所以也需要

/**
 * 检测依赖
 *
 * @param beanName          the name of the bean to check
 * @param dependentBeanName the name of the dependent bean
 * @since 4.0
 */
protected boolean isDependent(String beanName, String dependentBeanName) {
    synchronized (this.dependentBeanMap) {
        return isDependent(beanName, dependentBeanName, null);
    }
}

// 从 dependentBeanMap 中找,看能否找到从 beanName-->dependentBeanName 的直接/间接映射
// 1. 递归
// 2. 树的深度遍历
// 思路很简单,和 SimpleAliasRegistry 的那个遍历差不多
private boolean isDependent(String beanName, String dependentBeanName, @Nullable Set<String> alreadySeen) {
    if (alreadySeen != null && alreadySeen.contains(beanName)) {
        return false;
    }
    String canonicalName = canonicalName(beanName);
    Set<String> dependentBeans = this.dependentBeanMap.get(canonicalName);
    if (dependentBeans == null) {
        return false;
    }
    if (dependentBeans.contains(dependentBeanName)) {
        return true;
    }
    for (String transitiveDependency : dependentBeans) {
        if (alreadySeen == null) {
            alreadySeen = new HashSet<>();
        }
        alreadySeen.add(beanName);
        if (isDependent(transitiveDependency, dependentBeanName, alreadySeen)) {
            return true;
        }
    }
    return false;
}

/**
 * Determine whether a dependent bean has been registered for the given name.
 *
 * @param beanName the name of the bean to check
 */
// 判断 beanName 是否被依赖
protected boolean hasDependentBean(String beanName) {
    return this.dependentBeanMap.containsKey(beanName);
}

/**
 * Return the names of all beans which depend on the specified bean, if any.
 *
 * @param beanName the name of the bean
 * @return the array of dependent bean names, or an empty array if none
 */
public String[] getDependentBeans(String beanName) {
    Set<String> dependentBeans = this.dependentBeanMap.get(beanName);
    if (dependentBeans == null) {
        return new String[0];
    }
    synchronized (this.dependentBeanMap) {
        return StringUtils.toStringArray(dependentBeans);
    }
}

扩展方法——清除指定/所有单例 bean

public void destroySingletons() {
    if (logger.isDebugEnabled()) {
        logger.debug("Destroying singletons in " + this);
    }
    synchronized (this.singletonObjects) {
        this.singletonsCurrentlyInDestruction = true;
    }

    String[] disposableBeanNames;
    synchronized (this.disposableBeans) {
        disposableBeanNames = StringUtils.toStringArray(this.disposableBeans.keySet());
    }
    for (int i = disposableBeanNames.length - 1; i >= 0; i--) {
        destroySingleton(disposableBeanNames[i]);
    }

    this.containedBeanMap.clear();
    this.dependentBeanMap.clear();
    this.dependenciesForBeanMap.clear();

    clearSingletonCache();
}

/**
 * Clear all cached singleton instances in this registry.
 *
 * @since 4.3.15
 */
protected void clearSingletonCache() {
    synchronized (this.singletonObjects) {
        this.singletonObjects.clear();
        this.singletonFactories.clear();
        this.earlySingletonObjects.clear();
        this.registeredSingletons.clear();
        this.singletonsCurrentlyInDestruction = false;
    }
}

/**
 * Destroy the given bean. Delegates to {@code destroyBean}
 * if a corresponding disposable bean instance is found.
 *
 * @param beanName the name of the bean
 * @see #destroyBean
 */
// 这里可以优化吧,这里算作是树形的递归调用,最开始调用起这个函数时也是用的循环,
// 在方法头加一个判断,如果remove发现没有,说明在删除其他单例时就提前删完此单例bean了,直接退出
public void destroySingleton(String beanName) {
    // 此函数主要目的是取消单例和对应的 DisposableBean 的注册

    // Remove a registered singleton of the given name, if any.
    removeSingleton(beanName);

    // Destroy the corresponding DisposableBean instance.
    DisposableBean disposableBean;
    synchronized (this.disposableBeans) {
        disposableBean = (DisposableBean) this.disposableBeans.remove(beanName);
    }
    // 将具体的销毁工作委托出去
    destroyBean(beanName, disposableBean);
}

/**
 * Destroy the given bean. Must destroy beans that depend on the given
 * bean before the bean itself. Should not throw any exceptions.
 *
 * @param beanName the name of the bean
 * @param bean     the bean instance to destroy
 */
protected void destroyBean(String beanName, @Nullable DisposableBean bean) {
    // 先将依赖 beanName 的单例销毁才能销毁 beanName

    // Trigger destruction of dependent beans first...
    Set<String> dependencies;
    synchronized (this.dependentBeanMap) {
        // Within full synchronization in order to guarantee a disconnected Set
        dependencies = this.dependentBeanMap.remove(beanName);
    }
    if (dependencies != null) {
        if (logger.isDebugEnabled()) {
            logger.debug("Retrieved dependent beans for bean '" + beanName + "': " + dependencies);
        }
        for (String dependentBeanName : dependencies) { // 先递归销毁依赖 beanName 的单例
            destroySingleton(dependentBeanName);
        }
    }
    // 完成依赖此单例的项目的销毁

    // Actually destroy the bean now...
    if (bean != null) {
        try {
            bean.destroy(); // 销毁 beanName
        } catch (Throwable ex) {
            logger.error("Destroy method on bean with name '" + beanName + "' threw an exception", ex);
        }
    }

    // 将包含此单例的项目进行销毁
    // TODO 感觉怪怪的,销毁完 beanName 后 containedBeans 应该不能继续工作了,怎么能把这个放在最后面?
    // Trigger destruction of contained beans...
    Set<String> containedBeans;
    synchronized (this.containedBeanMap) {
        // Within full synchronization in order to guarantee a disconnected Set
        containedBeans = this.containedBeanMap.remove(beanName);
    }
    if (containedBeans != null) {
        for (String containedBeanName : containedBeans) {
            destroySingleton(containedBeanName);
        }
    }

    // 完成包含此单例的项目的销毁

    // 做一些善后操作,beanName 依赖了一些 bean ,把这些映射移除即可

    // Remove destroyed bean from other beans' dependencies.
    synchronized (this.dependentBeanMap) {
        for (Iterator<Map.Entry<String, Set<String>>> it = this.dependentBeanMap.entrySet().iterator(); it.hasNext(); ) {
            Map.Entry<String, Set<String>> entry = it.next();
            Set<String> dependenciesToClean = entry.getValue();
            dependenciesToClean.remove(beanName);
            if (dependenciesToClean.isEmpty()) {
                it.remove();
            }
        }
    }

    // Remove destroyed bean's prepared dependency information.
    this.dependenciesForBeanMap.remove(beanName);
}

总结记录

此类主要关注那些实例属性即可,把那些Map的功能梳理清楚,方法都是见名知意的。

问题

冗余数据结构定义问题

我们在介绍上面的实例属性时很容易发现:我们定义的Map,Set那一大串虽然很多很全,但是有点过多了,很多属性我们如果仔细品味的话还是有问题的——有的属性我们觉得一样,实际上可能真的有点冗余;有的属性我们觉得一样,但是他们的职能、使用场景完全不同。

singletonObjectsregisteredSingletons

/**
 * Cache of singleton objects: bean name --> bean instance
 */
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);

/**
 * Set of registered singletons, containing the bean names in registration order
 */
//TODO 没必要专门用个集合来存储,这样维护一致性要花费不少精力
private final Set<String> registeredSingletons = new LinkedHashSet<>(256);

个人认为没必要专门创建下面的Set直接用singletonObjects.keySet足以,原因:

  1. registeredSingletons为私有且未向外暴露,不会出现线程安全之类的其他的的情况,不会拖singletonObjects的后腿
  2. 线程的集合类接口,没有理由不用

Bean 销毁的顺序问题

destroyBean中,销毁的先后顺序是:

  1. 依赖此beanNamebean
  2. beanName对应的bean
  3. 包含此beanName的外部容器

个人感觉应该是3,1,2/1,3,2而不是123。

先销毁了 beanName对应bean,外面的容器就无法正常工作了,会造成请求失败。

扩展

冗余数据结构定义问题

dependentBeanMapdependenciesForBeanMap

理论上说吧你用逻辑运算也是能算出来这两个是有逻辑重复的。但是如果合并成一个数据结构的话,那就是一个图了,可能要用一个二维矩阵来表示,这需要专门封装很多东西。而且,你获得一个依赖集,遍历那么多,有点复杂了吧。

综上,个人认为这个不能随便替代

singletonsCurrentlyInCreationearlySingletonObjectssingletonFactories

singletonsCurrentlyInCreation这个钩子是专门为扩展方法中的那个将 Bean 的所有实例化、初始化都委托给外部函数时用的记录位置。

earlySingletonObjectssingletonFactories是一起用的,用来解决单例 bean 的循环依赖问题。一般在上面的"将 Bean 的所有实例化、初始化都委托给外部函数"中的外部函数中自行使用。

线程安全问题

只有singletonObjects采用了线程安全的集合,此类的复杂操作时都自行使用了隐式锁。这点做的很全面。【在个别方法没有用锁的(例如isDependent)是因为用private,在调用它的public中加了锁。】

因为singletonObjects要暴露出去,所以采用线程安全的集合也是完全可以理解的。

单例 Bean 的循环依赖问题

一般,我们在进行单例 Bean 的生成和注册时都是用的提供的扩展方法,传入一个创建、初始化的方法来做。在这种情况下,一个单例 Bean 的流转顺序我们在DefaultSingletonRegistry的角度看大概是这样的:

2.png

感觉自己时序图画的完全不符合规范。

主要目的是说明在各个缓存块之间的流转关系,这个后面想个好点的、更直观的表达方式吧。

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

推荐阅读更多精彩内容