浅谈责任链设计模式

GoF 定义

  • Chain Of Responsibility Design Pattern

Avoid coupling the sender of a request to its receiver by giving more than one object a chance to handle the request. Chain the receiving objects and pass the request along the chain until an object handles it.

借由<设计模式之美>中王争的通俗解释就是:

"在职责链模式中,多个处理器(也就是刚刚定义中说的“接收对象”)依次处理同一个请求。一个请求先经过 A 处理器处理,然后再把请求传递给 B 处理器,B 处理器处理完后再传递给 C 处理器,以此类推,形成一个链条。链条上的每个处理器各自承担各自的处理职责,所以叫作职责链模式。"

传统意义上的责任链模式有几个特点:

  • 有且仅有一个处理器处理,随着请求结束链的滑动随之终止。
  • 数据结构可以是数组 or 链表。
  • 遍历方式可以是递归 or 迭代
  • 链存在order顺序,可以排列优先级
  • 处理器选择策略可以由"钩子"方法释放。

先看一个简单的例子,然后再继续聊

public abstract class AbstractHandler {
    private AbstractHandler nextHandler;

    public final Response handleMessage(Request request) {
        Response response = null;
        //判断是否是自己的处理级别
        if (this.getHandlerLevel() == request.getBizCodeEnum()) {
            response = this.echo(request);
        } else {
            //不属于自己的处理级别,判断是否有下一个处理者
            if (this.nextHandler != null) {
                response = this.nextHandler.handleMessage(request);
            } else {
                //没有适当的处理者,再想点办法吧
            }
        }
        return response;
    }

    public void setNext(AbstractHandler _handler) {
        this.nextHandler = _handler;
    }

    protected abstract BizCodeEnum getHandlerLevel();

    protected abstract Response echo(Request request);

    public enum BizCodeEnum {
        A,
        B
    }

    @Builder
    @Getter
    public static class Request {
        //请求的等级
        BizCodeEnum bizCodeEnum;
        String name;
        String age;

    }

    @Builder
    @Getter
    public static class Response {
        boolean success;
        String code;
        String message;
    }
}

public class HandlerA extends AbstractHandler {

    @Override
    protected BizCodeEnum getHandlerLevel() {
        return BizCodeEnum.A;
    }

    @Override
    protected Response echo(Request request) {
        System.out.println("A");
        return Response.builder().code("A").build();
    }
}

public class HandlerB extends AbstractHandler {

    @Override
    protected BizCodeEnum getHandlerLevel() {
        return BizCodeEnum.B;
    }

    @Override
    protected Response echo(Request request) {
        System.out.println("B");
        return Response.builder().code("B").build();
    }
}


public class Client {

    public static void main(String[] args) {
        //声明所有的处理节点
        AbstractHandler handler1 = new HandlerA();
        AbstractHandler handler2 = new HandlerB();
        //设置链中的阶段顺序1-->2
        handler1.setNext(handler2);
        //提交请求,返回结果
        Response response = handler1.handleMessage(Request.builder().bizCodeEnum(BizCodeEnum.A).build());
        System.out.println(response.getCode() == "A");

        Response response2 = handler1.handleMessage(Request.builder().bizCodeEnum(BizCodeEnum.B).build());
        System.out.println(response2.getCode() == "B");
    }
}



如上图,一个非常正经的责任链模式。
首先定义一个抽象基类AbstractHandler,将选择策略和递归的动作封装,并且暴露抽象方法定义节点的业务身份,以及节点的处理逻辑,数据结构属于链表。

从这个例子我们可以延伸一下,首先整体解耦思想和实现方式、效果上来看,比如AbstractHandler这个类,定义method的方式是一个典型的模版设计模式(final & abstract & hook method )。其次用打标的方式选择是哪个节点处理,本质上是一个策略思想。然后,如果是单节点handle这种场景,其实就相当于观察者模式变种的Pub-Sub模式,想象一样用EventBus、MQ实现的话,对不同的consumer打tag标也差不多是这个能力。但是以上几种模式的侧重点不同,只是聊起来感觉很多东西套路类似。

所以本质要解决的问题是:

  • 将请求的发送和接收解耦
  • 无感知的拓展接收者、并且使职责单一细化
    也就是单一原则、开闭原则等。

传统责任链的拓展延伸

首先谈一下维持单节点响应设计的变种

  • 单节点响应模式变种
    就以上例子来说,如果我们维持只需要一个节点响应就终止链的话。可以看到,上面的例子,有几点可以玩的。

· 优先级
如果整体是一个链表遍历的话,那么我们可以在主动挂链表的时候,将优先级更靠前的往前放。比如几率比较大的节点放在前面,可以减少链表遍历的次数,有点类似平均复杂度,你可以把绝大多数碰到的情况放在第一个节点。

· 挂链
我们可以将所有节点打上@Component注解,注入Spring环境。然后通过依赖查找的方式,从Spring环境中获取List<AbstractHandle>,再通过stream流,经过filter之后执行,filter算法就是钩子方法equals一下。那么链数据结构也变成了数组。

@Component
public class Client {
    private final List<AbstractHandler> handlerList;

    private Client(List<AbstractHandler> handlerList) {
        this.handlerList = handlerList;
    }

    public void test() {
        Request request = Request.builder().bizCodeEnum(BizCodeEnum.A).build();

        handlerList.stream().filter(f -> f.getHandlerLevel() == request.getBizCodeEnum())
            .forEach(abstractHandler -> abstractHandler.echo(request));
    }

  
}

细节一点的话,可以把filter提取一个方法深化语义,然后filter之后把流终止拿到一个Optional对象。
Preconditions.checkArgument(size>1,"节点不唯一");
Preconditions.checkArgument(isEmpty,"没有符合条件的节点");
以上判断,或者进行特殊逻辑,或者findAny终止判断一次之类的。

如果要保持链表数组结构也可以,在Spring ApplicationRunner等生命周期进行setNext操作即可,等于说将挂链的操作自动化处理一下,不需要我们去主动挂链了。这个就是属于spring生命周期做一些init操作范畴了,有点类似eventbus使用中,去做一些regist操作。

但是仍然是一个O(N)遍历,也可以改成用Map维护节点和业务身份的关系,通过"查表"这个老套路找到责任节点,不过这样就有点偏策略模式了。
个人觉得没必要太极端,一般来说链的节点个数本身就不会太多,实际上也基本上没有什么性能问题,责任链本身就是链的特性。

  private static final Map<String,AbstractHandler> CONTAINER =
        ImmutableMap.of(
            "A",new HandlerA(),
            "B",new HandlerB()
        );

    public static void main(String[] args) {

        Request request = Request.builder().bizCodeEnum(BizCodeEnum.A).build();
        //提交请求,返回结果
        Response response = CONTAINER.get("A").echo(request);
        System.out.println(response.getCode() == "A");

    }

当判断条件比较模糊的时候,还有自由组合这种场景上,责任链模式更适合。查表也只适合比较固定code这种判断。后面我们再详细看如何组合。

· 判断
上面的钩子方法属于比较明确的策略分支,也可以不使用钩子方法,用boolean变量做。

    public final void handle() {
        boolean handled = doHandle();
        if (successor != null && !handled) { successor.handle(); }
    }

    protected abstract boolean doHandle();



子类去复写doHandle,只要有一个节点执行过,链就终止了。基于这种设计,我们可以复用节点,串成不同的链,在不同的场合使用。而不必强行给节点标识业务身份,链可以无状态一些,具体要看业务需求。

  • 非单节点变种
    在很多框架中,以及日常开发需求中,责任链一般都是所有节点都跑一遍的,或者一直跑到某个节点的终止动作。比如拦截器、函数式组合等。
    这种链就比较多了,比如OkHttp、Tomcat、Rpc、Spring Interceptor等。
    我们简单看几个。


    image.png

    image.png

    image.png

我想主要介绍一下OkHttp源码中的实现,还是比较好玩的。

Response getResponseWithInterceptorChain() throws IOException {
    // Build a full stack of interceptors.
    List<Interceptor> interceptors = new ArrayList<>();
    interceptors.addAll(client.interceptors());
    interceptors.add(new RetryAndFollowUpInterceptor(client));
    interceptors.add(new BridgeInterceptor(client.cookieJar()));
    interceptors.add(new CacheInterceptor(client.internalCache()));
    interceptors.add(new ConnectInterceptor(client));
    if (!forWebSocket) {
      interceptors.addAll(client.networkInterceptors());
    }
    interceptors.add(new CallServerInterceptor(forWebSocket));

    Interceptor.Chain chain = new RealInterceptorChain(interceptors, transmitter, null, 0,
        originalRequest, this, client.connectTimeoutMillis(),
        client.readTimeoutMillis(), client.writeTimeoutMillis());

    boolean calledNoMoreExchanges = false;
    try {
      Response response = chain.proceed(originalRequest);
      if (transmitter.isCanceled()) {
        closeQuietly(response);
        throw new IOException("Canceled");
      }
      return response;
    } catch (IOException e) {
      calledNoMoreExchanges = true;
      throw transmitter.noMoreExchanges(e);
    } finally {
      if (!calledNoMoreExchanges) {
        transmitter.noMoreExchanges(null);
      }
    }
  }

  public Response proceed(Request request, Transmitter transmitter, @Nullable Exchange exchange)
      throws IOException {
    if (index >= interceptors.size()) throw new AssertionError();

    calls++;

    // If we already have a stream, confirm that the incoming request will use it.
    if (this.exchange != null && !this.exchange.connection().supportsUrl(request.url())) {
      throw new IllegalStateException("network interceptor " + interceptors.get(index - 1)
          + " must retain the same host and port");
    }

    // If we already have a stream, confirm that this is the only call to chain.proceed().
    if (this.exchange != null && calls > 1) {
      throw new IllegalStateException("network interceptor " + interceptors.get(index - 1)
          + " must call proceed() exactly once");
    }

    // Call the next interceptor in the chain.
    RealInterceptorChain next = new RealInterceptorChain(interceptors, transmitter, exchange,
        index + 1, request, call, connectTimeout, readTimeout, writeTimeout);
    Interceptor interceptor = interceptors.get(index);
    Response response = interceptor.intercept(next);

    // Confirm that the next interceptor made its required call to chain.proceed().
    if (exchange != null && index + 1 < interceptors.size() && next.calls != 1) {
      throw new IllegalStateException("network interceptor " + interceptor
          + " must call proceed() exactly once");
    }

    // Confirm that the intercepted response isn't null.
    if (response == null) {
      throw new NullPointerException("interceptor " + interceptor + " returned null");
    }

    if (response.body() == null) {
      throw new IllegalStateException(
          "interceptor " + interceptor + " returned a response with no body");
    }

    return response;
  }

  @Override public Response intercept(Chain chain) throws IOException {
    Response cacheCandidate = cache != null
        ? cache.get(chain.request())
        : null;

    long now = System.currentTimeMillis();

    CacheStrategy strategy = new CacheStrategy.Factory(now, chain.request(), cacheCandidate).get();
    Request networkRequest = strategy.networkRequest;
    Response cacheResponse = strategy.cacheResponse;

    if (cache != null) {
      cache.trackResponse(strategy);
    }

    if (cacheCandidate != null && cacheResponse == null) {
      closeQuietly(cacheCandidate.body()); // The cache candidate wasn't applicable. Close it.
    }

    // If we're forbidden from using the network and the cache is insufficient, fail.
    if (networkRequest == null && cacheResponse == null) {
      return new Response.Builder()
          .request(chain.request())
          .protocol(Protocol.HTTP_1_1)
          .code(504)
          .message("Unsatisfiable Request (only-if-cached)")
          .body(Util.EMPTY_RESPONSE)
          .sentRequestAtMillis(-1L)
          .receivedResponseAtMillis(System.currentTimeMillis())
          .build();
    }

    // If we don't need the network, we're done.
    if (networkRequest == null) {
      return cacheResponse.newBuilder()
          .cacheResponse(stripBody(cacheResponse))
          .build();
    }

    Response networkResponse = null;
    try {
      networkResponse = chain.proceed(networkRequest);
    } finally {
      // If we're crashing on I/O or otherwise, don't leak the cache body.
      if (networkResponse == null && cacheCandidate != null) {
        closeQuietly(cacheCandidate.body());
      }
    }

    // If we have a cache response too, then we're doing a conditional get.
    if (cacheResponse != null) {
      if (networkResponse.code() == HTTP_NOT_MODIFIED) {
        Response response = cacheResponse.newBuilder()
            .headers(combine(cacheResponse.headers(), networkResponse.headers()))
            .sentRequestAtMillis(networkResponse.sentRequestAtMillis())
            .receivedResponseAtMillis(networkResponse.receivedResponseAtMillis())
            .cacheResponse(stripBody(cacheResponse))
            .networkResponse(stripBody(networkResponse))
            .build();
        networkResponse.body().close();

        // Update the cache after combining headers but before stripping the
        // Content-Encoding header (as performed by initContentStream()).
        cache.trackConditionalCacheHit();
        cache.update(cacheResponse, response);
        return response;
      } else {
        closeQuietly(cacheResponse.body());
      }
    }

    Response response = networkResponse.newBuilder()
        .cacheResponse(stripBody(cacheResponse))
        .networkResponse(stripBody(networkResponse))
        .build();

    if (cache != null) {
      if (HttpHeaders.hasBody(response) && CacheStrategy.isCacheable(response, networkRequest)) {
        // Offer this request to the cache.
        CacheRequest cacheRequest = cache.put(response);
        return cacheWritingResponse(cacheRequest, response);
      }

      if (HttpMethod.invalidatesCache(networkRequest.method())) {
        try {
          cache.remove(networkRequest);
        } catch (IOException ignored) {
          // The cache cannot be written.
        }
      }
    }

    return response;
  }

可以看到Okhttp中,首先是一个双向责任链,也就是执行前按顺序执行所有拦截器,然后执行后再反过来执行一遍。想要实现双向,其实有两种方案,递归或者透传对象。

public interface Interceptor {
  Response intercept(Chain chain) throws IOException;

  interface Chain {
    Request request();

    Response proceed(Request request) throws IOException;

    /**
     * Returns the connection the request will be executed on. This is only available in the chains
     * of network interceptors; for application interceptors this is always null.
     */
    @Nullable Connection connection();

    Call call();

    int connectTimeoutMillis();

    Chain withConnectTimeout(int timeout, TimeUnit unit);

    int readTimeoutMillis();

    Chain withReadTimeout(int timeout, TimeUnit unit);

    int writeTimeoutMillis();

    Chain withWriteTimeout(int timeout, TimeUnit unit);
  }
}

Okhttp的设计是用传递Chain这个接口,进行递归。每一层拦截器,都会去执行
chain.proceed();proceed内部又是通过index + 1再找下一个拦截器执行。于是整个过程就跟二叉树的前中后序遍历一样,那么在chain.proceed()之前执行的代码就是前置Interceptor逻辑,后面执行的就是后置Interceptor逻辑。

在Okhttp整体设计中,是将自定义的拦截器先加入List,后面依次追加,这就是前面那张图的执行顺序由来。

    List<Interceptor> interceptors = new ArrayList<>();
    interceptors.addAll(client.interceptors());
    interceptors.add(new RetryAndFollowUpInterceptor(client));
    interceptors.add(new BridgeInterceptor(client.cookieJar()));
    interceptors.add(new CacheInterceptor(client.internalCache()));
    interceptors.add(new ConnectInterceptor(client));

然后Interceptor也是可以提前退出链的,比如CacheInterceptor如果命中了,就不会再去走网络拦截器了,可以看到源码里return response了,彼时还没有走到proceed。

整体来说,Chain的设计比较优雅,其次双向的实现也可以借鉴,还有就是list循环用index进行推进。

Tomcat源码也是通过pos下标去跑的,定位链执行到了哪个节点。

Spring Interceptor实现的简单一些,直接平铺了几个方法。

在DispatcherServlet 的 doDispatch() 方法来分发请求,它在真正的业务逻辑执行前后,执行 HandlerExecutionChain 中的 applyPreHandle() 和 applyPostHandle() 函数,用来实现拦截的功能。

以上就是几种常见的框架责任链的玩法了。

职责链模式常用在框架的开发中,为框架提供扩展点,让框架的使用者在不修改框架源码的情况下,基于扩展点添加新的功能。

责任链与函数式编程

其实工作中很多时候遇到比较简单的场景,我们可以借由java.util.function包里面的函数式接口,结合传统的设计模式,进行简化。

像java 8推出的很多函数式接口,本身就是推荐我们使用组合等方式进行编程的,最早应该是google一些类库先推的一些api,8借鉴过来了,java也常常干这种事,比如ListenableFuture -> CompleableFuture ,LocalDate之类的。挺好的,最终java版本越来越演进,我们也可以更使用更好的范式。

比如现在有这样一个需求,在客户发起一个操作之前,我们会判断这个客户是否可以进行这个动作。比方说客户要注销某个账户,而这个账户还有未结清的账单,或者这个账户仍然有一些在途的操作,此时我们应该返回相应的提示,让客户结清账单或者取消在途订单再来注销。

我们首先定义三个Predicate

 private Predicate<CreditQuitSDO> onGoingCreditQuitPredicate() {
        return creditQuit -> {
            CreditQuitFindParam creditQuitFindParam = assemblerCreditQuitFindParam(creditQuit);
            CreditQuitSDO existOrder = creditQuitRepository.find(creditQuitFindParam);
            if (existOrder.isUnFinishCreditQuit) {
                throw new CreditQuitRuleCheckException("ON_GOING_SERVICE_CLOSE_EXIST",
                    Lists.newArrayList("ON_GOING_SERVICE_CLOSE_EXIST"), "重复关闭请求");
            }
            return true;
        };
    }

    private Predicate<CreditQuitSDO> onGoingLoanPredicate() {
        return creditQuit -> {
            List<LoanSDO> existOrderList = loanRepository.findAll(assemblerLoanUserQuery(creditQuit));

            Optional.ofNullable(existOrderList).ifPresent(list -> list.forEach(existOrder -> {
                if (existOrder.isUnContractFinished()) {
                    throw new CreditQuitRuleCheckException("ON_GOING_LOAN_EXIST",
                        Lists.newArrayList("ON_GOING_LOAN_EXIST"), "您有未结清或已申请未放款业务,不能申请退出服务");
                }
            }));

            return true;
        };
    }

    private Predicate<CreditQuitSDO> onGoingCreditPredicate() {
        return creditQuit -> {
            CreditFindParam creditFindParam = assemblerCreditFindParam(creditQuit);
            CreditSDO existOrder = creditRepository.find(creditFindParam);
            if (belongSuitableStatus(existOrder.getStatus())) {
                throw new CreditQuitRuleCheckException("ON_GOING_CREDIT_EXIST",
                    Lists.newArrayList("ON_GOING_CREDIT_EXIST"), "在途授信流程未结束,无法进行服务关闭");
            }
            return true;
        };
    }

然后我们继续使用tmf添加应用层拓展的更多准入条件,不同的业务身份添加的自定义拓展准入条件可以不同。

  @AbilityExtension(code = EXT_POINT_CHECK_RULES,
        name = "服务关闭校验规则定义",
        desc = "服务关闭校验规则定义")
    default List<Predicate<CreditQuitSDO>> getCheckPredicates(CreditQuitSDO creditQuit) {
        return Lists.newArrayList();
    }
    public void check(CreditQuitSDO creditQuit) {
        List<Predicate<CreditQuitSDO>> checkPredicates =
            Lists.newArrayList(onGoingCreditPredicate(), onGoingCreditQuitPredicate(), onGoingLoanPredicate());

        CreditQuitExtPoints extension = getExtensionPoints(creditQuit.getBizCode());
        List<Predicate<CreditQuitSDO>> extPredicates = extension.getCheckPredicates(creditQuit);

        if (CollectionUtils.isNotEmpty(extPredicates)) {
            checkPredicates.addAll(extPredicates);
        }

        checkPredicates.forEach(predicate -> predicate.test(creditQuit));
    }

以上我们就相当于得到了小业务中台复用的准入条件,加上tmf中自定义的准入条件,结合到一起的Predicate集合,然后进行判断即可,在上层捕捉异常即可返回命中的那条失败信息,比如说onGoingCreditPredicate没有通过,就会返回用户:"在途授信流程未结束,无法进行服务关闭"。

如果不想抛异常,想提高性能,也可以返回false,然后通过别的方式透传失败话术,视场景而定,我这里性能可接受,所以选择最方便的方式上抛校验话术。

有一些场景我们也可以直接用static修饰predicate,在编译期就通过静态map编排,查表进行策略,或者switch case编排之类的。如果说在静态static predicate中有spring依赖无法引入的问题(就是static作用域引用autowared的对象,编译报错),也比较简单。

@Component
public class InstanceLocator extends SpringInitializingBean {

    public static <T> T getInstance(Class<T> requiredType) {
        return applicationContext.getBean(requiredType);
    }

    public static <T> T getInstance(String name, Class<T> requiredType) {
        return applicationContext.getBean(name, requiredType);
    }

    public static <T> Map<String, T> getBeansOfType(Class<T> type) {
        return applicationContext.getBeansOfType(type);
    }

    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }
}


在static修饰的predicate中使用InstanceLocator.getBean即可,因为运行期间已经能够从spring容器中获取bean了,绕过这个限制即可。其实底层都是从DefaultListableBeanFactory获取的,ApplicationContext这个bean工厂也是通过DefaultListableBeanFactory委托管理bean的,他们是同一个接口实现类。

以上也是比较简单的玩法,当然Predicate也可以组合着玩。
比如:

onGoingCreditQuitPredicate()
            .and(onGoingLoanPredicate())
            .and(onGoingCreditPredicate()).test(creditQuit);

这也是一种挂链的方式。

如果我们进一步封装,可以将Predicate都放在PredicateFactory中维护,由Factory与上层交互,面向新增就在factory继续铺predicate。其实没有必要很纠结原则,比如说Predicate没有单独放一个类,设计模式是比较灵活的,在函数名和函数内部方法名很合理的情况下,可读性完全没有问题。并且有一个优势在于,多个predicate公用的方法可以提取在一个类的作用域,而私有的方法可以直接包裹在predicate内部。
比如:

    private Predicate<CreditQuitSDO> onGoingCreditPredicate() {
        return new Predicate<CreditQuitSDO>() {
            @Override
            public boolean test(CreditQuitSDO creditQuit) {
                CreditFindParam creditFindParam = assemblerCreditFindParam(creditQuit);
                CreditSDO existOrder = creditRepository.find(creditFindParam);
                if (belongSuitableStatus(existOrder.getStatus())) {
                    throw new CreditQuitRuleCheckException("ON_GOING_CREDIT_EXIST",
                        Lists.newArrayList("ON_GOING_CREDIT_EXIST"), "在途授信流程未结束,无法进行服务关闭");
                }

                return true;
            }

            private  boolean belongSuitableStatus(CreditStatus creditStatus) {
                return !ENABLE_QUIT_STATUS.contains(creditStatus);
            }

            private  CreditFindParam assemblerCreditFindParam(CreditQuitSDO creditQuit) {
                CreditFindParam creditFindParam = new CreditFindParam();
                creditFindParam.setCustomerId(creditQuit.getCustomer().getCustomerId());
                return creditFindParam;
            }
        };
    }

我们可以把belongSuitableStatus和assemblerCreditFindParam两个仅属于这个predicate私有的方法内置,然后将多个predicate公用的方法外置,来进一步细腻,可以和前面一张图对比一下,其实就已经把Predicate当一个类用了,比较简洁,替代传统设计模式的类。

回顾

  • 责任链模式官方定义
    将请求的发送和接收解耦,让多个接收对象都有机会处理这个请求。将这些接收对象串成一条链,并沿着这条链传递这个请求,直到链上的某个接收对象能够处理它为止。

  • 实际业务实现
    可以单节点处理,也可以全部过一遍,可以数组也可以链表。

  • Spring、Functional结合
    可以通过Spring实现节点的自动注册发现、自动串联等。
    可以通过Predicate等函数,进行简易的责任链实现,Predicate也是很适合作为tmf ExtendPoint定义的。
    关于更多Spring和函数式小技巧单独再写一篇分享,还是有很多设计模式或者框架二次封装,可以通过Spring和Functional方式玩一些的。

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