RocketMQ 底層通信機(jī)制 源碼分析

概述

RocketMQ 底層通訊是使用Netty來(lái)實(shí)現(xiàn)的。
下面我們通過(guò)源碼分析下RocketMQ是怎么利用Netty進(jìn)行通訊的。

本文分析的是RocketMQ 最新版本 4.3.2版本。

RocketMQ 項(xiàng)目結(jié)構(gòu)

首先來(lái)看下 RocketMQ 模塊構(gòu)成。



通過(guò) RocketMQ 項(xiàng)目結(jié)構(gòu)可以看出,RocketMQ 分了好多模塊。 broker、client、filter、namesrv、remoting 等。

大家比較熟悉的幾個(gè)模塊對(duì)應(yīng)的源碼如下:
Broker Master 和 Slave 對(duì)應(yīng)的 broker 模塊。
Producer 和 Consumer 對(duì)應(yīng)的是 client 模塊。
NameSerer 服務(wù)對(duì)應(yīng)的是 namesrv 模塊。

而各個(gè)服務(wù)之間的通訊則使用的 remoting 模塊。

Remoting 模塊

通過(guò)romoting 的模塊結(jié)構(gòu)大概了解,RocketMQ 通訊使用了Netty進(jìn)行傳輸通訊。并在 org.apache.rocketmq.remoting.protocol 包中自定義了通訊協(xié)議。

通信模塊主要接口和類

RemotingService 接口

public interface RemotingService {
    //開(kāi)啟服務(wù)
    void start();
    //關(guān)閉服務(wù)
    void shutdown();
    //注冊(cè) hook (可以在調(diào)用之前和調(diào)用之后做一些擴(kuò)展處理)
    void registerRPCHook(RPCHook rpcHook);
}

RemotingService 定義了服務(wù)端和客戶端都需要的三個(gè)接口。
registerRPCHook() 方法可以注冊(cè)一個(gè) hook。可以在遠(yuǎn)程通信之前和通信之后,執(zhí)行用戶自定的一些處理。類似前置處理器和后置處理器。

RPCHook 接口

public interface RPCHook {
    void doBeforeRequest(final String remoteAddr, final RemotingCommand request);

    void doAfterResponse(final String remoteAddr, final RemotingCommand request,
        final RemotingCommand response);
}

在啟動(dòng)服務(wù)之前,可以把自己實(shí)現(xiàn)的 RPCHook 注冊(cè)到服務(wù)中,執(zhí)行遠(yuǎn)程調(diào)用的時(shí)候處理一些業(yè)務(wù)邏輯。比如打印請(qǐng)求和響應(yīng)的日志信息。

RemotingServer 和 RemotingClient 接口

RemotingServer 和 RemotingClient 接口都繼承了RemotingService 接口,并擴(kuò)展了自己特有的方法。

RemotingServer 接口

public interface RemotingServer extends RemotingService {

    //注冊(cè)一個(gè)處理請(qǐng)求的處理器, 根據(jù)requestCode, 獲取處理器,處理請(qǐng)求
    void registerProcessor(final int requestCode, final NettyRequestProcessor processor,
        final ExecutorService executor);
    //注冊(cè)一個(gè)默認(rèn)的處理器,當(dāng)根據(jù)requestCode匹配不到處理器,則使用這個(gè)默認(rèn)的處理器
    void registerDefaultProcessor(final NettyRequestProcessor processor, final ExecutorService executor);
    //獲取端口
    int localListenPort();
    //根據(jù)requestCode獲取請(qǐng)求處理器
    Pair<NettyRequestProcessor, ExecutorService> getProcessorPair(final int requestCode);
    //同步調(diào)用(同步發(fā)送消息)
    RemotingCommand invokeSync(final Channel channel, final RemotingCommand request,
        final long timeoutMillis) throws InterruptedException, RemotingSendRequestException,
        RemotingTimeoutException;
    //異步調(diào)用(異步發(fā)送消息)
    void invokeAsync(final Channel channel, final RemotingCommand request, final long timeoutMillis,
        final InvokeCallback invokeCallback) throws InterruptedException,
        RemotingTooMuchRequestException, RemotingTimeoutException, RemotingSendRequestException;
    //單向發(fā)送消息,只發(fā)送消息。不用處理發(fā)送的結(jié)果。
    void invokeOneway(final Channel channel, final RemotingCommand request, final long timeoutMillis)
        throws InterruptedException, RemotingTooMuchRequestException, RemotingTimeoutException,
        RemotingSendRequestException;

}
  • 1、registerProcessor 方法
    注冊(cè)一個(gè)處理請(qǐng)求的處理器, 存放到 HashMap中,requestCode為 Map 的 key。
    HashMap<Integer/* request code */, Pair<NettyRequestProcessor, ExecutorService>> processorTable

  • 2、registerDefaultProcessor 方法
    注冊(cè)一個(gè)默認(rèn)的處理器,當(dāng)根據(jù)requestCode匹配不到處理器,則使用這個(gè)默認(rèn)的處理器

  • 3、invokeSync 方法
    以同步的方式向客戶端發(fā)送消息。

  • 4、invokeAsync 方法
    以異步的方式向客戶端發(fā)送消息。

  • 5、invokeOneway 方法
    只向客戶端發(fā)送消息,而不處理客戶端返回的消息。該方法只是向socket中寫(xiě)入數(shù)據(jù),而不需要處理客戶端返回的消息。

RemotingClient 接口

public interface RemotingClient extends RemotingService {
    //更新 NameServer 地址
    void updateNameServerAddressList(final List<String> addrs);
    //獲取 NameServer 地址
    List<String> getNameServerAddressList();
    //同步調(diào)用(同步發(fā)送消息)
    RemotingCommand invokeSync(final String addr, final RemotingCommand request,
        final long timeoutMillis) throws InterruptedException, RemotingConnectException,
        RemotingSendRequestException, RemotingTimeoutException;
    //異步調(diào)用(異步發(fā)送消息)
    void invokeAsync(final String addr, final RemotingCommand request, final long timeoutMillis,
        final InvokeCallback invokeCallback) throws InterruptedException, RemotingConnectException,
        RemotingTooMuchRequestException, RemotingTimeoutException, RemotingSendRequestException;
    //單向發(fā)送消息,只發(fā)送消息。不用處理發(fā)送的結(jié)果。
    void invokeOneway(final String addr, final RemotingCommand request, final long timeoutMillis)
        throws InterruptedException, RemotingConnectException, RemotingTooMuchRequestException,
        RemotingTimeoutException, RemotingSendRequestException;
    //注冊(cè)一個(gè)處理請(qǐng)求的處理器, 根據(jù)requestCode, 獲取處理器,處理請(qǐng)求
    void registerProcessor(final int requestCode, final NettyRequestProcessor processor,
        final ExecutorService executor);
    //設(shè)置發(fā)送異步消息的線程池,如果不設(shè)置,則使用默認(rèn)的
    void setCallbackExecutor(final ExecutorService callbackExecutor);
    //獲取線程池
    ExecutorService getCallbackExecutor();
    //判斷 channel 是否可寫(xiě)
    boolean isChannelWritable(final String addr);
}
  • 1、updateNameServerAddressList、getNameServerAddressList 方法
    更新 NameServer 地址。
    獲取 NameServer 地址。

  • 2、invokeSync、invokeAsync、invokeOneway 方法
    這三個(gè)方法參見(jiàn) RemotingServer 接口中的方法。

  • 3、setCallbackExecutor
    設(shè)置處理異步響應(yīng)消息的線程池。


服務(wù)端和客戶端的實(shí)現(xiàn)

  • NettyRemotingServer 類實(shí)現(xiàn)了RemotingServer 接口
  • NettyRemotingClient 類實(shí)現(xiàn)了RemotingClient接口

這兩個(gè)類使用Netty 來(lái)實(shí)現(xiàn)服務(wù)端和客戶端服務(wù)的。

NettyRemotingServer 解析

通過(guò) NettyRemotingServer類中的start() 方法開(kāi)啟一個(gè) Netty 的服務(wù)端。
代碼如下:

    @Override
    public void start() {
        this.defaultEventExecutorGroup = new DefaultEventExecutorGroup(
            nettyServerConfig.getServerWorkerThreads(),
            new ThreadFactory() {
                private AtomicInteger threadIndex = new AtomicInteger(0);
                @Override
                public Thread newThread(Runnable r) {
                    return new Thread(r, "NettyServerCodecThread_" + this.threadIndex.incrementAndGet());
                }
            });

        ServerBootstrap childHandler =
            this.serverBootstrap.group(this.eventLoopGroupBoss, this.eventLoopGroupSelector)
                .channel(useEpoll() ? EpollServerSocketChannel.class : NioServerSocketChannel.class)
                .option(ChannelOption.SO_BACKLOG, 1024)
                .option(ChannelOption.SO_REUSEADDR, true)
                .option(ChannelOption.SO_KEEPALIVE, false)
                .childOption(ChannelOption.TCP_NODELAY, true)
                .childOption(ChannelOption.SO_SNDBUF, nettyServerConfig.getServerSocketSndBufSize())
                .childOption(ChannelOption.SO_RCVBUF, nettyServerConfig.getServerSocketRcvBufSize())
                .localAddress(new InetSocketAddress(this.nettyServerConfig.getListenPort()))
                .childHandler(new ChannelInitializer<SocketChannel>() {
                    @Override
                    public void initChannel(SocketChannel ch) throws Exception {
                        ch.pipeline()
                            .addLast(defaultEventExecutorGroup, HANDSHAKE_HANDLER_NAME,
                                new HandshakeHandler(TlsSystemConfig.tlsMode))
                            .addLast(defaultEventExecutorGroup,
                                //編碼
                                new NettyEncoder(),
                                //解碼
                                new NettyDecoder(),
                                //心跳檢測(cè)
                                new IdleStateHandler(0, 0, nettyServerConfig.getServerChannelMaxIdleTimeSeconds()),
                                //連接管理handler,處理connect, disconnect, close等事件
                                new NettyConnectManageHandler(),
                                //處理接收到RemotingCommand消息后的事件, 收到服務(wù)器端響應(yīng)后的相關(guān)操作
                                new NettyServerHandler()
                            );
                    }
                });

        if (nettyServerConfig.isServerPooledByteBufAllocatorEnable()) {
            childHandler.childOption(ChannelOption.ALLOCATOR, PooledByteBufAllocator.DEFAULT);
        }

        try {
            ChannelFuture sync = this.serverBootstrap.bind().sync();
            InetSocketAddress addr = (InetSocketAddress) sync.channel().localAddress();
            this.port = addr.getPort();
        } catch (InterruptedException e1) {
            throw new RuntimeException("this.serverBootstrap.bind().sync() InterruptedException", e1);
        }

        if (this.channelEventListener != null) {
            this.nettyEventExecutor.start();
        }

        this.timer.scheduleAtFixedRate(new TimerTask() {
            @Override
            public void run() {
                try {
                    NettyRemotingServer.this.scanResponseTable();
                } catch (Throwable e) {
                    log.error("scanResponseTable exception", e);
                }
            }
        }, 1000 * 3, 1000);
    }

從 start 方法中啟動(dòng)一個(gè)Netty 的服務(wù)端。

  • 通過(guò)設(shè)置的自定義的 NettyEncoder對(duì)發(fā)送的消息進(jìn)行編碼(序列化)。
  • 通過(guò)NettyDecoder 對(duì)接收的消息進(jìn)行解碼操作(反序列化)
  • 最后再把反序列化的對(duì)象交給 NettyServerHandler 進(jìn)行處理。

NettyRemotingClient 解析

通過(guò) NettyRemotingClient 類中的 start 方法開(kāi)啟一個(gè) netty 客戶端

@Override
    public void start() {
        this.defaultEventExecutorGroup = new DefaultEventExecutorGroup(
            nettyClientConfig.getClientWorkerThreads(),
            new ThreadFactory() {

                private AtomicInteger threadIndex = new AtomicInteger(0);

                @Override
                public Thread newThread(Runnable r) {
                    return new Thread(r, "NettyClientWorkerThread_" + this.threadIndex.incrementAndGet());
                }
            });

        Bootstrap handler = this.bootstrap.group(this.eventLoopGroupWorker).channel(NioSocketChannel.class)
            .option(ChannelOption.TCP_NODELAY, true)
            .option(ChannelOption.SO_KEEPALIVE, false)
            .option(ChannelOption.CONNECT_TIMEOUT_MILLIS, nettyClientConfig.getConnectTimeoutMillis())
            .option(ChannelOption.SO_SNDBUF, nettyClientConfig.getClientSocketSndBufSize())
            .option(ChannelOption.SO_RCVBUF, nettyClientConfig.getClientSocketRcvBufSize())
            .handler(new ChannelInitializer<SocketChannel>() {
                @Override
                public void initChannel(SocketChannel ch) throws Exception {
                    ChannelPipeline pipeline = ch.pipeline();
                    if (nettyClientConfig.isUseTLS()) {
                        if (null != sslContext) {
                            pipeline.addFirst(defaultEventExecutorGroup, "sslHandler", sslContext.newHandler(ch.alloc()));
                            log.info("Prepend SSL handler");
                        } else {
                            log.warn("Connections are insecure as SSLContext is null!");
                        }
                    }
                    pipeline.addLast(
                        defaultEventExecutorGroup,
                        //發(fā)送消息編碼
                        new NettyEncoder(),
                        //接收消息解碼
                        new NettyDecoder(),
                        //心跳監(jiān)測(cè)
                        new IdleStateHandler(0, 0, nettyClientConfig.getClientChannelMaxIdleTimeSeconds()),
                        //連接管理handler,處理connect, disconnect, close等事件
                        new NettyConnectManageHandler(),
                       //處理接收到RemotingCommand消息后的事件, 收到服務(wù)器端響應(yīng)后的相關(guān)操作
                        new NettyClientHandler());
                }
            });

        this.timer.scheduleAtFixedRate(new TimerTask() {
            @Override
            public void run() {
                try {
                    NettyRemotingClient.this.scanResponseTable();
                } catch (Throwable e) {
                    log.error("scanResponseTable exception", e);
                }
            }
        }, 1000 * 3, 1000);

        if (this.channelEventListener != null) {
            this.nettyEventExecutor.start();
        }
    }

從 start 方法中啟動(dòng)一個(gè)Netty 客戶端服務(wù)。

  • 通過(guò)設(shè)置的自定義的 NettyEncoder對(duì)發(fā)送的消息進(jìn)行編碼(序列化)。
  • 通過(guò)NettyDecoder對(duì)接收的消息進(jìn)行解碼操作(反序列化)
  • 最后再把反序列化的對(duì)象交給 NettyServerHandler` 進(jìn)行處理。

序列化反序列化

通過(guò)分析 RemotingServerRemotingClient 接口及實(shí)現(xiàn)可以發(fā)現(xiàn),發(fā)送消息和接收到的消息都是 RemotingCommand 對(duì)象。
經(jīng)過(guò)分析 NettyEncoderNettyDecoder 發(fā)現(xiàn),序列化和反序列化調(diào)用的是 RemotingCommand 對(duì)象的 encodedecode 方法

消息格式

  • 第一部分是消息的長(zhǎng)度,占用4個(gè)字節(jié)。等于第二、三、四部分長(zhǎng)度的總和。
  • 第二部分是消息頭的長(zhǎng)度,占用4個(gè)字節(jié)。等于第三部分長(zhǎng)度大小。
  • 第三部分是通過(guò)Json序列化的消息頭的數(shù)據(jù)。
  • 第四部分是序列化的消息數(shù)據(jù)。

具體的消息格式我們通過(guò) RemotingCommand類的 encodedecode 方法進(jìn)行分析。

RemotingCommand.encode() 方法

    public ByteBuffer encode() {
        // 1> header length size
        int length = 4;

        // 2> header data length
        byte[] headerData = this.headerEncode();
        length += headerData.length;

        // 3> body data length
        if (this.body != null) {
            length += body.length;
        }

        ByteBuffer result = ByteBuffer.allocate(4 + length);

        // length
        result.putInt(length);

        // header length
        result.put(markProtocolType(headerData.length, serializeTypeCurrentRPC));

        // header data
        result.put(headerData);

        // body data;
        if (this.body != null) {
            result.put(this.body);
        }

        result.flip();

        return result;
    }

1、定義消息頭的長(zhǎng)度為 length = 4
2、通過(guò) this.headerEncode() 獲取序列化的 header data。
3、然后申請(qǐng)一個(gè)長(zhǎng)度為 length + header length + header data +body 大小的ByteBuffer。
ByteBuffer result = ByteBuffer.allocate(4 + length);
4、然后向 ByteBuffer result 中填充數(shù)據(jù)

headerEncode 方法

該方法主要是實(shí)現(xiàn)了消息頭的序列化。

private byte[] headerEncode() {
        this.makeCustomHeaderToNet();
        if (SerializeType.ROCKETMQ == serializeTypeCurrentRPC) {
            return RocketMQSerializable.rocketMQProtocolEncode(this);
        } else {
            return RemotingSerializable.encode(this);
        }
    }

序列化消息頭有兩種方式SerializeType.ROCKETMQ 和 SerializeType.JSON。
如果是SerializeType.JSON方式序列化比較簡(jiǎn)單。

RemotingSerializable.encode 方法

SerializeType.JSON 類型序列化。

    public static byte[] encode(final Object obj) {
        final String json = toJson(obj, false);
        if (json != null) {
            return json.getBytes(CHARSET_UTF8);
        }
        return null;
    }

直接把對(duì)象轉(zhuǎn)換成json字符串,然后轉(zhuǎn)換成 byte[] 數(shù)組

RocketMQSerializable.rocketMQProtocolEncode 方法

SerializeType.ROCKETMQ 類型序列化。

    public static byte[] rocketMQProtocolEncode(RemotingCommand cmd) {
        // String remark
        byte[] remarkBytes = null;
        int remarkLen = 0;
        if (cmd.getRemark() != null && cmd.getRemark().length() > 0) {
            remarkBytes = cmd.getRemark().getBytes(CHARSET_UTF8);
            remarkLen = remarkBytes.length;
        }

        // HashMap<String, String> extFields
        byte[] extFieldsBytes = null;
        int extLen = 0;
        if (cmd.getExtFields() != null && !cmd.getExtFields().isEmpty()) {
            extFieldsBytes = mapSerialize(cmd.getExtFields());
            extLen = extFieldsBytes.length;
        }

        int totalLen = calTotalLen(remarkLen, extLen);

        ByteBuffer headerBuffer = ByteBuffer.allocate(totalLen);
        // int code(~32767)
        headerBuffer.putShort((short) cmd.getCode());
        // LanguageCode language
        headerBuffer.put(cmd.getLanguage().getCode());
        // int version(~32767)
        headerBuffer.putShort((short) cmd.getVersion());
        // int opaque
        headerBuffer.putInt(cmd.getOpaque());
        // int flag
        headerBuffer.putInt(cmd.getFlag());
        // String remark
        if (remarkBytes != null) {
            headerBuffer.putInt(remarkBytes.length);
            headerBuffer.put(remarkBytes);
        } else {
            headerBuffer.putInt(0);
        }
        // HashMap<String, String> extFields;
        if (extFieldsBytes != null) {
            headerBuffer.putInt(extFieldsBytes.length);
            headerBuffer.put(extFieldsBytes);
        } else {
            headerBuffer.putInt(0);
        }

        return headerBuffer.array();
    }

可以看到 代碼把 RemotingCommand 對(duì)象中的數(shù)據(jù)按照一定的順序轉(zhuǎn)換成字節(jié)存儲(chǔ)到ByteBuffer 中。

從代碼中可以看出消息頭中包括,request code、請(qǐng)求端實(shí)現(xiàn)語(yǔ)言、版本等信息。

RemotingCommand.decode() 方法

    public static RemotingCommand decode(final ByteBuffer byteBuffer) {
        int length = byteBuffer.limit();
        int oriHeaderLen = byteBuffer.getInt();
        int headerLength = getHeaderLength(oriHeaderLen);

        byte[] headerData = new byte[headerLength];
        byteBuffer.get(headerData);

        RemotingCommand cmd = headerDecode(headerData, getProtocolType(oriHeaderLen));

        int bodyLength = length - 4 - headerLength;
        byte[] bodyData = null;
        if (bodyLength > 0) {
            bodyData = new byte[bodyLength];
            byteBuffer.get(bodyData);
        }
        cmd.body = bodyData;

        return cmd;
    }

這里的byteBuffer中的數(shù)據(jù)包含 header length + header data +body data

為什么不是包含 length+header length + header data +body data呢?
因?yàn)閚etty在獲取這條消息的時(shí)候是通過(guò)io.netty.handler.codec.LengthFieldBasedFrameDecoder進(jìn)行拆包的。該拆包的原理就是通過(guò) 消息的 length長(zhǎng)度進(jìn)行拆分的。所以拆分出來(lái)的數(shù)據(jù)就是header length + header data +body data這部分。

1、從byteBuffer中獲取header length 長(zhǎng)度。
2、然后再通過(guò)header length 長(zhǎng)度從 byteBuffer 獲取 header data。
3、剩下的byteBuffer數(shù)據(jù)就是body的數(shù)據(jù)。
把解析出來(lái)的數(shù)據(jù)轉(zhuǎn)換成 RemotingCommand 對(duì)象。

    private static RemotingCommand headerDecode(byte[] headerData, SerializeType type) {
        switch (type) {
            case JSON:
                RemotingCommand resultJson = RemotingSerializable.decode(headerData, RemotingCommand.class);
                resultJson.setSerializeTypeCurrentRPC(type);
                return resultJson;
            case ROCKETMQ:
                RemotingCommand resultRMQ = RocketMQSerializable.rocketMQProtocolDecode(headerData);
                resultRMQ.setSerializeTypeCurrentRPC(type);
                return resultRMQ;
            default:
                break;
        }

        return null;
    }

判斷該數(shù)據(jù)是通過(guò) SerializeType.ROCKETMQ 序列化還是 SerializeType.JSON 序列化的。
然后根據(jù)類型進(jìn)行反序列化操作。

RemotingSerializable.decode 方法

SerializeType.JSON 反序列化。

    public static <T> T decode(final byte[] data, Class<T> classOfT) {
        final String json = new String(data, CHARSET_UTF8);
        return fromJson(json, classOfT);
    }

直接把 json 數(shù)據(jù)反序列化成對(duì)象。

RocketMQSerializable.rocketMQProtocolDecode 方法

SerializeType.ROCKETMQ 反序列化。

    public static RemotingCommand rocketMQProtocolDecode(final byte[] headerArray) {
        RemotingCommand cmd = new RemotingCommand();
        ByteBuffer headerBuffer = ByteBuffer.wrap(headerArray);
        // int code(~32767)
        cmd.setCode(headerBuffer.getShort());
        // LanguageCode language
        cmd.setLanguage(LanguageCode.valueOf(headerBuffer.get()));
        // int version(~32767)
        cmd.setVersion(headerBuffer.getShort());
        // int opaque
        cmd.setOpaque(headerBuffer.getInt());
        // int flag
        cmd.setFlag(headerBuffer.getInt());
        // String remark
        int remarkLength = headerBuffer.getInt();
        if (remarkLength > 0) {
            byte[] remarkContent = new byte[remarkLength];
            headerBuffer.get(remarkContent);
            cmd.setRemark(new String(remarkContent, CHARSET_UTF8));
        }

        // HashMap<String, String> extFields
        int extFieldsLength = headerBuffer.getInt();
        if (extFieldsLength > 0) {
            byte[] extFieldsBytes = new byte[extFieldsLength];
            headerBuffer.get(extFieldsBytes);
            cmd.setExtFields(mapDeserialize(extFieldsBytes));
        }
        return cmd;
    }

根據(jù) encode 的順序進(jìn)行反序列化操作。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,967評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,273評(píng)論 3 415
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 175,870評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 62,742評(píng)論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,527評(píng)論 6 407
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 55,010評(píng)論 1 322
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,108評(píng)論 3 440
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,250評(píng)論 0 288
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,769評(píng)論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,656評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,853評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,371評(píng)論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,103評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,472評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,717評(píng)論 1 281
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,487評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,815評(píng)論 2 372

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,781評(píng)論 18 139
  • Netty的簡(jiǎn)單介紹 Netty 是一個(gè) NIO client-server(客戶端服務(wù)器)框架,使用 Netty...
    AI喬治閱讀 8,429評(píng)論 1 101
  • Spring Web MVC Spring Web MVC 是包含在 Spring 框架中的 Web 框架,建立于...
    Hsinwong閱讀 22,490評(píng)論 1 92
  • 枯萎的紅 無(wú)法觸碰,參差的脈絡(luò) 你披著帶刺的嫁衣 尋找?guī)е端睦杳?沒(méi)有人為你解釋 也沒(méi)有人為你祝福 是誰(shuí)把玫瑰...
    二兩酒仙閱讀 272評(píng)論 6 12
  • 第一次是大花這樣說(shuō)的,她的原話是“我的男友力好強(qiáng)”。那天是我去她家,她剛洗完澡穿著睡衣,松散著有一些潮濕的頭發(fā)坐在...
    修坯刀閱讀 327評(píng)論 0 0