lagou 爪哇 3-2 zookeeper 笔记

Zookeeper 简介

分布式系统的协调工作就是通过某种方式,让每个节点的信息能够同步和共享。这依赖于服务进程之间的通信。通信方式有两种:

  • 通过网络进行信息共享
    这就像现实中,开发leader在会上把任务传达下去,组员通过听leader命令或者看leader的邮件知道自己要干什么。当任务分配有变化时,leader会单独告诉组员,或者再次召开会议。信息通过人与人之间的直接沟通,完成传递。

  • 通过共享存储
    这就好比开发leader按照约定的时间和路径,把任务分配表放到了svn上,组员每天去svn上拉取最新的任务分配表,然后干活。其中svn就是共享存储。更好一点的做法是,当svn文件版本更新时,触发邮件通知,每个组员再去拉取最新的任务分配表。这样做更好,因为每次更新,组员都能第一时间得到消
    息,从而让自己手中的任务分配表永远是最新的。此种方式依赖于中央存储。

ZooKeeper如何解决分布式系统面临的问题
ZooKeeper对分布式系统的协调,使用的是第二种方式,共享存储。其实共享存储,分布式应用也需要
和存储进行网络通信。

注:Slave节点要想获取ZooKeeper的更新通知,需事先在关心的数据节点上设置观察点。

大多数分布式系统中出现的问题,都源于信息的共享出了问题。如果各个节点间信息不能及时共享和同步,那么就会在协作过程中产生各种问题。ZooKeeper解决协同问题的关键,就是在于保证分布式系统信息的一致性。

zookeeper的基本概念

Zookeeper是一个开源的分布式协调服务,其设计目标是将那些复杂的且容易出错的分布式一致性服务封装起来,构成一个高效可靠的原语集,并以一些简单的接口提供给用户使用。zookeeper是一个典型的分布式数据一致性的解决方案,分布式应用程序可以基于它实现诸如数据订阅/发布、负载均衡、命名服务、集群管理、分布式锁和分布式队列等功能
基本概念

①集群角色
通常在分布式系统中,构成一个集群的每一台机器都有自己的角色,最典型的集群就是Master/Slave模式(主备模式),此情况下把所有能够处理写操作的机器称为Master机器,把所有通过异步复制方式获取最新数据,并提供读服务的机器为Slave机器。

而在Zookeeper中,这些概念被颠覆了。它没有沿用传递的Master/Slave概念,而是引入了Leader、Follower、Observer三种角色。Zookeeper集群中的所有机器通过Leader选举来选定一台被称为Leader的机器,Leader服务器为客户端提供读和写服务,除Leader外,其他机器包括Follower和Observer,Follower和Observer都能提供读服务,唯一的区别在于Observer不参与Leader选举过程,不参与写操作的过半写成功策略,因此Observer可以在不影响写性能的情况下提升集群的性能。

②会话(session)
Session指客户端会话,一个客户端连接是指客户端和服务端之间的一个TCP长连接, Zookeeper对外的服务端口默认为2181,客户端启动的时候,首先会与服务器建立一个TCP连接,从第一次连接建立开始,客户端会话的生命周期也开始了,通过这个连接,客户端能够心跳检测与服务器保持有效的会话,也能够向 Zookeeper服务器发送请求并接受响应,同时还能够通过该连接接受来自服务器的 Watch事件通知。

③数据节点(Znode)
在谈到分布式的时候,我们通常说的“节点”是指组成集群的每一台机器。然而,在 ZooKeeper中,“节点分为两类,第一类同样是指构成集群的机器,我们称之为机器节点;第二类则是指数据模型中的数据单元,我们称之为数据节点——ZNode。 ZooKeeper将所有数据存储在内存中,数据模型是一棵树
(ZNode Tree),由斜杠(/)进行分割的路径,就是一个node,例如app/path.每个node上都会保存自己的数据内容,同时还会保存一系列属性信息。

④版本
刚刚我们提到,Zookeeper的每个Znode上都会存储数据,对于每个ZNode,Zookeeper都会为其维护一个叫作 Stat 的数据结构,Stat记录了这个ZNode的三个数据版本,分别是version(当前ZNode的版本)、cversion(当前ZNode子节点的版本)、aversion(当前ZNode的ACL版本)。

⑤Watcher(事件监听器)
Wathcer(事件监听器),是Zookeeper中一个很重要的特性,Zookeeper允许用户在指定节点上注册一些Watcher,并且在一些特定事件触发的时候,Zookeeper服务端会将事件通知到感兴趣的客户端,该机制是Zookeeper实现分布式协调服务的重要特性

⑥ ACL
Zookeeper采用ACL(Access Control Lists)策略来进行权限控制,其定义了如下五种权限:
·CREATE:创建子节点的权限。
·READ:获取节点数据和子节点列表的权限。
·WRITE:更新节点数据的权限。
·DELETE:删除子节点的权限。
·ADMIN:设置节点ACL的权限。
其中需要注意的是,CREATE和DELETE这两种权限都是针对子节点的权限控制

环境搭建

Zookeeper的搭建方式

Zookeeper安装方式有三种,单机模式和集群模式以及伪集群模式。

  • 单机模式:Zookeeper只运行在一台服务器上,适合测试环境;
  • 集群模式:Zookeeper运行于一个集群上,适合生产环境,这个计算机集群被称为一个“集合体”
  • 伪集群模式:就是在一台服务器上运行多个Zookeeper实例;

单机模式搭建:
zookeeper安装以linux环境为例:

1、下载
首先我们下载稳定版本的zookeeper http://zookeeper.apache.org/releases.html

配置单节点

$ tar -zxf zookeeper-3.4.6.tar.gz
$ cd zookeeper-3.4.6
$ mkdir data

cd conf
mv zoo_sample.cfg zoo.cfg

vi conf/zoo.cfg
编辑文件设置 dataDir = /path/to/zookeeper/data

$ bin/zkServer.sh start

启动CLI

$ bin/zkCli.sh
$ bin/zkCli.sh  -server  需要连接的ip:需要连接的port

windows 下启用 zk

$ zkCli.cmd
$ zkCli.cmd  -server  需要连接的ip:需要连接的port

例如 zkCli.cmd 106.75.105.152, 不加端口,默认为 2181

若报错, 则检查是否防火墙拦截了.

Opening socket connection to server 192.168.153.12/192.168.153.12:2181.Will not attempt to authentic

停止ZooKeeper服务器
连接服务器并执行所有操作后,可以使用以下命令停止zookeeper服务器。

$ bin/zkServer.sh stop

在 Zookeeper中,每一个数据节点都是一个 Znode,上图根目录下有两个节点,分别是:app1和app2,其中app1下面又有三个子节点所有 Znode!按层次化进行组织,形成这么一颗树, Znodel的节点路径标识方式和 Unix文件系统路径非常相似,都是由一系列使用斜杠(/)进行分割的路径表示,开发人员可以向这个节点写入数据,也可以在这个节点下面创建子节点。

默认端口为 2181

配置伪集群模式

创建 data 文件夹 和 logs 文件夹

clientPort=2181
# 配置快照文件存放的目录
dataDir=/zkcluster/zookeeper01/data 
# 配置日志文件存放的目录
dataLogDir=/zkcluster/zookeeper01/data/logs

clientPort=2182 
dataDir=/zkcluster/zookeeper02/data 
dataLogDir=/zkcluster/zookeeper02/data/logs

clientPort=2183 
dataDir=/zkcluster/zookeeper03/data 
dataLogDir=/zkcluster/zookeeper03/data/logs

data 下创建 myid 文件, 内容分别为 1, 2, 3 (数字可以依次累增), 这个文件的作用就是记录zk的id

server.服务器ID=服务器IP地址:服务器之间通信端口:服务器之间投票选举端口

server.1=10.211.55.4:2881:3881 
server.2=10.211.55.4:2882:3882 
server.3=10.211.55.4:2883:3883 
touch myid

分别向三台服务器写入数字 1 2 和 3.

启动集群
分别启动这三台服务器

$ bin/zkServer.sh start

查看状态

./zkServer.sh status

Zookeeper基本使用

ZooKeeper系统模型

ZooKeeper数据模型Znode在ZooKeeper中,数据信息被保存在一个个数据节点上,这些节点被称为 ZNode。ZNode是Zookeeper中最小数据单位,在ZNode下面又可以再挂 ZNode,这样一层层下去就形成了一个层次化命名空间ZNode树,我们称为ZNode Tree,它采用了类似文件系统的层级树状结构进行管理。见下图示例:

四种节点类型

  • 持久节点
  • 持久顺序节点
  • 临时节点
  • 临时顺序节点

事务ID

ZNode 状态信息

ACL
我们可以从三个方面来理解ACL机制:权限模式( Scheme)、授权对象(ID)、权限( Permission),通常使用" scheme:id: permission"来标识一个有效的ACL信息。

权限模式: Scheme
权限模式用来确定权限验证过程中使用的检验策略

授权对象:ID
授权对象指的是权限赋予的用户或一个指定实体,例如IP地址或是机器等。在不同的权限模式下,授权对象是不同的,表中列出了各个权限模式和授权对象之间的对应关系

权限
权限就是指那些通过权限检査后可以被允许执行的操作。在 Zookeeper中,所有对数据的操作权限分为以下五大类

  • CREATE(C):数据节点的创建权限,允许授权对象在该数据节点下创建子节点。
  • DELETE(D子节点的删除权限,允许授权对象删除该数据节点的子节点。・
  • READ(R):数据节点的读取权限,允许授权对象访问该数据节点并读取其数据内容或子节点列表等。
  • WRTE(W):数据节点的更新权限,允许授权对象对该数据节点进行更新操作。
  • ADMIN(A):数据节点的管理权限,允许授权对象对该数据节点进行ACL相关的设置操作。

创建节点
使用 creates命令,可以创建一个 Zookeeper节点,如

create [-s][-e] path data acl

其中,-s-e 分别指定节点特性,顺序或临时节点,若不指定,则创建持久节点;ac1用来进行权限控制。

  1. 创建永久(持久)节点
    使用 create /zk-permanent 123命令创建zk- permanent永久节点
[zk: Loca thost: 2181(CONNECTED) 1] create/zk-permanent 123
Created/Zk-permanent
[zk: localhost: 2181(CONNECTED) 2] Ls/
[zk-permanent, zookeeper, zk-test00000000041
  1. 创建持久顺序节点
    使用 create -s /zk-test 123 命令创建zk-test顺序节点
lzk: Localhost: 2181(CONNECTED)0] create-s/zk-test 123
Created/Zk-testo000000004

执行完后,就在根节点下创建了一个叫做 / zk-test 的节点,该节点内容就是123,同时可以看到创建的
zk-test 节点后面添加了一串数字以示区别

  1. 创建临时节点
    使用 create -e /zk-temp 123命令创建zk-temp临时节点

  2. 创建临时顺序节点
    create -e -s /zk-temp 123

可以看到永久节点不同于顺序节点, 不会自动在后面添加一串数字

quit 退出客户端

读取节点
与读取相关的命令有 ls 命令和 get 命令

ls 命令可以列出 Zookeeper指定节点下的所有子节点,但只能查看指定节点下的第一级的所有子节点;

ls path

其中,path表示的是指定数据节点的节点路径
get命令可以获取 Zookeeper:指定节点的数据内容和属性信息。

get path

若获取根节点下面的所有子节点,使用 ls 命令即可

若想获取/zk-permanente的数据内容和属性,可使用如下命令:get /zk-permanent

更新节点
使用set命令,可以更新指定节点的数据内容,用法如下

set path data [version]

其中,data就是要更新的新内容, version表示数据版本,在 zookeeper中,节点的数据是有版本概
念的,这个参数用于指定本次更新操作是基于 Inode的哪一个数据版本进行的,如将/zk- permanent节
点的数据更新为455,可以使用如下命令:set /zk-permanent 456

zk: Loca Lhost: 2181( CONNECTED)3] set /zk-permanent 456
Iczxid 0X12
Ctime Sat Mar 07 18: 11: 14 CST 2020
Imzxid =0x13
Mtime =Sat Mar 07 18: 13: 48 CST 2020
Zxid = 0x12
cversion =O
ldataversion 1
laclversion =0
ephemera lowner 0x0
ldatalength =3
numchildren =0

刪除节点
使用 delete 命令可以删除 Zookeeper上的指定节点,用法如下
delete path [version]
其中 version也是表示数据版本,使用 delete /zk-permanent 命令即可删除 zk-permanent节点

zk 的 Java 客户端工具

zk 的 Java 客户端工具 curator

创建节点

获取数据

// 普通查询 
client.getData().forPath(path); 
// 包含状态查询 
Stat stat = new Stat(); 
client.getData().storingStatIn(stat).forPath(path);

更新数据

// 普通更新
client.setData().forPath(path,"新内容".getBytes());
// 指定版本更新 
client.setData().withVersion(1).forPath(path);

删除数据

配置存储

命名服务

如果加了排它锁则只对一个事务可见, 若加上共享锁,则对所有事务可见.

作业

编程题一:
在基于 Netty 的自定义RPC的案例基础上,进行改造。基于 Zookeeper 实现简易版服务的注册与发现机制

要求完成改造版本:

  1. 启动 2 个服务端,可以将IP及端口信息自动注册到 Zookeeper
  2. 客户端启动时,从Zookeeper中获取所有服务提供端节点信息,客户端与每一个服务端都建立连接
  3. 某个服务端下线后,Zookeeper注册列表会自动剔除下线的服务端节点,客户端与下线的服务端断开连接
  4. 服务端重新上线,客户端能感知到,并且与重新上线的服务端重新建立连接

编程题二:
基于作业一的基础上,实现基于 Zookeeper 的简易版负载均衡策略

要求完成改造版本:

  1. Zookeeper 记录每个服务端的最后一次响应时间,有效时间为 5秒,5s内如果该服务端没有新的请求,响应时间清零或失效
  2. 当客户端发起调用,每次都选择最后一次响应时间短的服务端进行服务调用,如果时间一致,随机选取一个服务端进行调用,从而实现负载均衡

编程题三:
基于Zookeeper实现简易版配置中心

要求实现以下功能:

  1. 创建一个 Web 项目,将数据库连接信息交给Zookeeper配置中心管理,即:当项目Web项目启动时,从 Zookeeper 进行 MySQL 配置参数的拉取
  2. 要求项目通过数据库连接池访问MySQL(连接池可以自由选择熟悉的)
  3. 当 Zookeeper 配置信息变化后Web项目自动感知,正确释放之前连接池,创建新的连接池

作业资料说明:
1、提供资料:3个代码工程、验证及讲解视频。(仓库中只有本次作业内容)

2、讲解内容包含:题目分析、实现思路、代码讲解。

3、效果视频验证:
3.1 作业1:服务端的上线与下线,客户端能动态感知,并能重新构成负载均衡。
3.2 作业2:作业完成情况下,选择性能好的服务器处理(响应时间短的服务器即为性能好)。Zookeeper记录客户端响应有效时间为5s,超时判定该客户端失效。
3.3 作业3:Zookeeper配置中心,web访问数据库需要从Zookeeper获取连接资源。当Zookeeper配置发生改变,web自动切换到新的连接资源,保持正常访问。

作业1
新增 NodeChangeListener 类

public interface NodeChangeListener {

    /**
     *
     * @param serviceName 服务名称
     * @param serviceList  服务名称对应节点下的所有子节点, 目前没有用到
     * @param pathChildrenCacheEvent
     */
    void notify(String serviceName, List<String> serviceList, PathChildrenCacheEvent pathChildrenCacheEvent);

}

将 zk 的行为抽象成接口

public interface RpcRegistryHandler extends NodeChangeListener {

    /**
     * 服务端进行调用
     *
     * @param service
     * @param ip
     * @param port
     * @return
     */
    boolean registry(final String service, final String ip, final int port);

    /**
     * 客户端进行调用
     *
     * @param service
     * @return
     */
    List<String> discovery(final String service);

    void addListener(NodeChangeListener service);

    void destroy();
}

ConfigKeeper 配置类

public class ConfigKeeper {

    /**
     * netty 的端口号
     */
    private int nettyPort;

    /**
     * zk 地址: ip + 端口
     */
    private String zkAddr;

    /**
     * 主动上报时间,单位 秒
     */
    private int interval;

    /**
     * 区分是客户端 还是 server 端, true 是服务端, false 是客户端
     */
    private boolean providerSide;

    // 单例类,setter 和 getter 方法
}

新增 RpcResponse 类

package com.lagou;

public class RpcResponse  {

    /**
     * 响应ID
     */
    private String requestId;
    /**
     * 错误信息
     */
    private String error;
    /**
     * 返回的结果
     */
    private Object result;

    // setter 和 getter 方法, toString方法 
}

RpcServerHandler 类,这次主要对 channelRead 的方法内容作了调整

    /**
     * 服务端将数据 写入 客户端, 继续传递下去
     * @throws Exception
     */
    @Override
    public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
        // 222222
        RpcRequest request = (RpcRequest) msg;
        final RpcResponse rpcResponse = new RpcResponse();
        rpcResponse.setRequestId(request.getRequestId());
        System.out.println("111 接收到" + request.getRequestId());
        rpcResponse.setResult(handler(request));
        // 3333333
        ctx.writeAndFlush(rpcResponse);
    }

服务端 rpc -server 用到的配置类 RpcServerConfig

public class RpcServerConfig {

    private String nettyHost;

    private int nettyPort;

    private int delay;

    /**
     * 是否是服务端
     */
    private boolean providerSide;

    /**
     * 应用的名称
     */
    private String applicationName;

    private Map<String, Class> services;
   
    // setter 和 getter 方法
}

RpcServer 类
自身 implements InitializingBean, DisposableBean 接口
Autowired 了 RpcRegistryFactory 对象
主要关注 afterPropertiesSet 和 destroy 方法即可

@Override
    public void afterPropertiesSet() throws Exception {
        this.initRpcServerConfig();
        this.startServer();
    }

    @Override
    public void destroy() {
        bossGroup.shutdownGracefully();
        workGroup.shutdownGracefully();
    }

这里牵涉到了RpcRegistryFactory

/**
 * 注册中心工厂类
 */
@Component
public class RpcRegistryFactory implements FactoryBean<RpcRegistryHandler>, DisposableBean {

    private RpcRegistryHandler rpcRegistryHandler;

    @Override
    public RpcRegistryHandler getObject()  {

        if (this.rpcRegistryHandler == null) {
            rpcRegistryHandler = new ZkRegistryHandler(ConfigKeeper.getInstance().getZkAddr());
        }
        return rpcRegistryHandler;
    }

    @Override
    public Class<?> getObjectType() {
        System.out.println("RpcRegistryFactory ### getObjectType.....");
        return RpcRegistryHandler.class;
    }

    @Override
    public void destroy() {
        System.out.println("RpcRegistryFactory ### destroy.....");
        rpcRegistryHandler.destroy();
    }
}

用于设计参数的 ProviderLoader

public class ProviderLoader {

    private ProviderLoader() {
    }

    /**
     * 返回类的全路径名 -> 该类的 class
     * @return
     */
    public static Map<String, Class> getInstanceCacheMap() {
        Map<String, Class> services = new HashMap<>();
        services.put(IUserService.class.getName(), IUserService.class);
        return services;
    }
}

ZkRegistryHandler 是对 RpcRegistryHandler 接口的 zk 实现。

public class ZkRegistryHandler implements RpcRegistryHandler {

    private static final String ZK_PATH_SPLITER = "/";
    private static final String LAGOU_EDU_RPC_ZK_ROOT = ZK_PATH_SPLITER + "lg-rpc-provider" + ZK_PATH_SPLITER;
    private List<NodeChangeListener> listenerList = new ArrayList<>();
    private final String url;

    private final CuratorFramework client;
    private volatile boolean closed;
    /**
     * 子节点列表
     */
    private List<String> serviceList;
    private static final ScheduledExecutorService REPORT_WORKER = Executors.newScheduledThreadPool(5);

    public ZkRegistryHandler(final String zkPath) {
        url = zkPath;
        this.client = CuratorFrameworkFactory.builder()
                .connectString(zkPath)
                .retryPolicy(new ExponentialBackoffRetry(1000, 3))
                .build();

        client.getConnectionStateListenable().addListener(new ConnectionStateListener() {
            @Override
            public void stateChanged(CuratorFramework curatorFramework, ConnectionState connectionState) {
                if (ConnectionState.CONNECTED.equals(connectionState)) {
                    System.out.println("注册中心连接成功");
                }
            }
        });
        client.start();

        // 定时上报
        final ConfigKeeper configKeeper = ConfigKeeper.getInstance();
        final boolean providerSide = configKeeper.isProviderSide();
        final int interval = configKeeper.getInterval();
        if (!providerSide && interval > 0) {
            REPORT_WORKER.scheduleWithFixedDelay(new Runnable() {
                @Override
                public void run() {
                    System.out.println("我是一个定时任务");
                    // RequestMetri
                }
            }, interval, interval, TimeUnit.SECONDS);
        }
    }

    /**
     * 服务端注册用到的方法
     * @param serviceName
     * @param nettyHost
     * @param nettyPort
     * @return
     */
    @Override
    public boolean registry(final String serviceName, final String nettyHost, final int nettyPort) {
        String zkPath = providePath(serviceName);
        if (!exists(zkPath)) {
            create(zkPath, false);
        }

        // /lg-rpc-provider/com.lagou.server.IUserService/provider/localhost:8999
        String instancePath = zkPath + ZK_PATH_SPLITER + nettyHost + ":" + nettyPort;
        create(instancePath, true);
        return true;
    }

    /**
     * 客户端查找服务的方法
     *
     * @param serviceName
     * @return
     */
    @Override
    public List<String> discovery(final String serviceName) {
        final String path = providePath(serviceName);
        if (serviceList == null || serviceList.isEmpty()) {
            System.out.println("首次查找地址");
            try {
                serviceList = client.getChildren().forPath(path);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        this.registryWatch(serviceName, path);
        return serviceList;
    }

    @Override
    public void addListener(NodeChangeListener listener) {
        listenerList.add(listener);
    }

    @Override
    public void destroy() {
        client.close();
    }

    @Override
    public void notify(String children, List<String> serviceList, PathChildrenCacheEvent pathChildrenCacheEvent) {
        for (NodeChangeListener nodeChangeListener : listenerList) {
            nodeChangeListener.notify(children, serviceList, pathChildrenCacheEvent);
        }
    }
    private void create(final String path, final boolean ephemeral) {
        CreateMode createMode;
        if (ephemeral) {
            createMode = CreateMode.EPHEMERAL;
        } else {
            createMode = CreateMode.PERSISTENT;
        }
        try {
            client.create().creatingParentsIfNeeded().withMode(createMode).forPath(path);
        } catch (KeeperException.NodeExistsException e) {
            // do nothing
            System.out.println("该路径已存在" + path);
        }
        catch (Exception e) {
            e.printStackTrace();
            throw new RuntimeException(e);
        }
    }

    private boolean exists(final String path) {
        try {
            if (client.checkExists().forPath(path) != null) {
                return true;
            }
        } catch (KeeperException.NoNodeException e) {
            // do nothing
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
        return false;
    }

    /**
     * 设置监听的方法
     *
     * @param serviceName
     * @param path
     */
    private void registryWatch(final String serviceName, final String path) {
        PathChildrenCache nodeCache = new PathChildrenCache(client, path, true);
        try {
            nodeCache.getListenable().addListener((client, pathChildrenCacheEvent) -> {
                // 更新本地緩存
                serviceList = client.getChildren().forPath(path);

                listenerList.forEach(nodeChangeListener -> {
                    System.out.println("节点变化");
                    nodeChangeListener.notify(serviceName, serviceList, pathChildrenCacheEvent);
                });
            });

            nodeCache.start(PathChildrenCache.StartMode.BUILD_INITIAL_CACHE);
        } catch (Exception e) {
        }
    }

    /**
     * 返回 /lg-rpc-provider/com.lagou.server.IUserService/provider
     * @param serviceName
     * @return
     */
    private String providePath(String serviceName) {
        return LAGOU_EDU_RPC_ZK_ROOT + serviceName + ZK_PATH_SPLITER + "provider";
    }

    private String metricsPath() {
        return LAGOU_EDU_RPC_ZK_ROOT + "metrics";
    }
}

rpc-server 的启动类

@SpringBootApplication
public class MyApplication {

    public static void main(String[] args) {
        // ["localhost:2181", "8999"]
        // ["localhost:2181", "9000"]
        final String zkPath = args[0];
        final int nettyPort = Integer.parseInt(args[1]);
        // 将IP及端口信息自动注册到 Zookeeper

        ConfigKeeper configKeeper = ConfigKeeper.getInstance();
        configKeeper.setProviderSide(true);
        configKeeper.setInterval(5);
        configKeeper.setNettyPort(nettyPort);
        configKeeper.setZkAddr(zkPath);
        System.out.println(configKeeper);

        SpringApplication.run(MyApplication.class, args);

        // 可以通过 ls /lg-rpc-provider/com.lagou.server.IUserService/provider 查看节点信息
    }
}

分别为 netty 启动 8888 和 8900 端口


接下来讲解 rpc-client

UserClientHandler 类可以复用

新增 RpcClient
主要对外暴露了 initClient 和 send 方法

// 2. 初始化netty客户端(创建连接池 bootstrap, 设置 BootStrap 连接服务器)
    public void initClient(String serviceClassName) throws InterruptedException {
        // 创建连接池
        this.group = new NioEventLoopGroup();
        // 创建客户端启动类
        Bootstrap bootstrap = new Bootstrap();
        // 配置启动引导类
        bootstrap.group(group)
                // 通道类型为 NIO
                .channel(NioSocketChannel.class)
                // 设置请求协议为 tcp
                .option(ChannelOption.TCP_NODELAY, true)
                .option(ChannelOption.SO_KEEPALIVE, true)
                .option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 3000)
                // 监听 channel 并初始化
                .handler(new ChannelInitializer<SocketChannel>() {
                    protected void initChannel(SocketChannel ch) throws Exception {
                        // 获取管道对象
                        ChannelPipeline pipeline = ch.pipeline();
                        pipeline.addLast(new RpcEncoder(RpcRequest.class, new JSONSerializer()));
                        pipeline.addLast(new RpcDecoder(RpcResponse.class, new JSONSerializer()));
                        // 自定义事件处理器
                        pipeline.addLast(new UserClientHandler());
                    }
                });
        this.channel = bootstrap.connect(this.nettyIp, this.nettyPort).sync().channel();

        if (!isValidate()) {
            close();
            return;
        }
        System.out.println("启动客户端" + serviceClassName + ", ip = " + this.nettyIp + ", port = " + nettyPort);
    }

    public Object send(RpcRequest request) throws InterruptedException, ExecutionException {
        // 统计请求时间
        RequestMetrics.getInstance().put(nettyIp, this.nettyPort, request.getRequestId());
        return this.channel.writeAndFlush(request).sync().get();
    }

新增 RpcConsumer 类

主要有用的方法有构造方法,createProxy的方法,notify为类自身 实现 NodeChangeListener 接口的方法(因为构造时会this.rpcRegistryHandler.addListener(this);)。

public class RpcConsumer implements NodeChangeListener {

    private final RpcRegistryHandler rpcRegistryHandler;

    private final Map<String, Class> serviceMap;

    /**
     * 服务名 -> List<RpcClient>
     */
    private final Map<String, List<RpcClient>> CLIENT_POOL = new HashMap<>();
    private LoadBalanceStrategy balanceStrategy = new RandomLoadBalance();

    /**
     * 初始化
     * @param rpcRegistryHandler
     * @param instanceCacheMap
     */
    public RpcConsumer(final RpcRegistryHandler rpcRegistryHandler, final Map<String, Class> instanceCacheMap) {
        this.rpcRegistryHandler = rpcRegistryHandler;
        this.serviceMap = instanceCacheMap;

        // 开始自动注册消费者逻辑: accept 方法
        serviceMap.forEach((className, clazz) -> {
            List<RpcClient> rpcClients = CLIENT_POOL.get(className);
            if (rpcClients == null) {
                rpcClients = new ArrayList<>();
            }
            // 127.0.0.1:8999 127.0.0.1:9000
            final List<String> discovery = this.rpcRegistryHandler.discovery(className);
            for (String s : discovery) {
                // s -> rpcClient
                final String[] split = s.split(":");
                String nettyIp = split[0];
                int nettyPort = Integer.parseInt(split[1]);
                final RpcClient rpcClient = new RpcClient(nettyIp, nettyPort);

                try {
                    rpcClient.initClient(className);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                rpcClients.add(rpcClient);
                CLIENT_POOL.put(className, rpcClients);
            }
        });
        this.rpcRegistryHandler.addListener(this);
    }

    // 4. 编写一个方法,使用 jdk 动态代理对象
    @SuppressWarnings("unchecked")
    public <T> T createProxyEnhance(final Class<T> serverClass) {
        return (T) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),
                new Class[]{serverClass}, (proxy, method, args) -> {

                    final String serverClassName = serverClass.getName();
                    // 封装
                    final RpcRequest request = new RpcRequest();
                    final String requestId = UUID.randomUUID().toString().substring(0, 7);

                    String className = method.getDeclaringClass().getName();
                    String methodName = method.getName();

                    Class<?>[] parameterTypes = method.getParameterTypes();

                    request.setRequestId(requestId);
                    request.setClassName(className);
                    request.setMethodName(methodName);
                    request.setParameterTypes(parameterTypes);
                    request.setParameters(args);

                    System.out.println("******************************\n请求id = " + requestId + ", 请求方法名 = " + methodName + ", 请求参数 = " + Arrays.toString(args));
                    RpcClient rpcClient = balanceStrategy.route(CLIENT_POOL, serverClassName);
                    if (rpcClient == null) {
                        System.out.println("没找到对应服务端,返 NULL");
                        return null;
                    }
                    System.out.println(request);
                    // request 最终会客户端发送给服务端进行消费
                    return rpcClient.send(request);
                });
    }

    /**
     * 监听临时节点的变化
     *
     * @param service 服务名称
     * @param serviceList  服务名称对应节点下的所有子节点
     * @param pathChildrenCacheEvent
     */
    @Override
    public void notify(final String service, final List<String> serviceList,
                       final PathChildrenCacheEvent pathChildrenCacheEvent) {
        // 取出变化的节点名称, 例如为 /lg-rpc-provider/com.lagou.server.IUserService/provider/localhost:9000
        final String path = pathChildrenCacheEvent.getData().getPath();
        System.out.println("变化节点的路径: " + path + ", 变化的类型: " + pathChildrenCacheEvent.getType());
        // 分离出 ip:port 的组合。
        final String instanceConfig = path.substring(path.lastIndexOf("/") + 1);
        System.out.println("instanceConfig: " + instanceConfig);
        final String[] address = instanceConfig.split(":");
        System.out.println("address: " + address);
        final String nettyIp = address[0];
        final int nettyPort = Integer.parseInt(address[1]);

        List<RpcClient> rpcClients = CLIENT_POOL.get(service);
        switch (pathChildrenCacheEvent.getType()) {
            // 增加节点
            case CHILD_ADDED:
            case CONNECTION_RECONNECTED:
                if (rpcClients == null) {
                    rpcClients = new ArrayList<>();
                }

                final RpcClient rpcClient = new RpcClient(nettyIp, nettyPort);
                try {
                    rpcClient.initClient(service);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                rpcClients.add(rpcClient);
                // 节点耗时统计
                RequestMetrics.getInstance().addNode(nettyIp, nettyPort);
                System.out.println("新增节点" + instanceConfig);
                break;
            // 增加节点
            case CHILD_REMOVED:
            case CONNECTION_SUSPENDED:
            case CONNECTION_LOST:
                if (rpcClients != null) {
                    for (RpcClient client : rpcClients) {
                        if (client.getNettyIp().equals(nettyIp) && client.getNettyPort() == nettyPort) {
                            rpcClients.remove(client);
                            // 节点耗时统计
                            RequestMetrics.getInstance().remoteNode(nettyIp, nettyPort);
                            System.out.println("移除节点" + instanceConfig);
                            break;
                        }
                    }
                }
                break;
        }
    }
}

最后再讲讲 ConsumerBootStrap, 该类基本没啥改动

public class ConsumerBootStrap {

    public static void main(final String[] args) throws Exception {
        final ConfigKeeper configKeeper = ConfigKeeper.getInstance();
        configKeeper.setZkAddr(args[0]);
        // 之后会启动一个定时的线程池,每 5s 上传到注册中心
        configKeeper.setInterval(5);
        configKeeper.setProviderSide(false);

        final RpcRegistryHandler rpcRegistryHandler = new ZkRegistryHandler(configKeeper.getZkAddr());
        System.out.println("客户端 Zookeeper session established.");

        // 最后一步
        final RpcConsumer consumer = new RpcConsumer(rpcRegistryHandler, ProviderLoader.getInstanceCacheMap());
        final IUserService userService = consumer.createProxyEnhance(IUserService.class);
        while (true) {
            Thread.sleep(4900);
            final String result = userService.sayHello("are you ok?");
            // 恒为 null
            System.out.println("返回 = " + result);
        }
    }
}

作业2

1.消费者每次请求完成时更新最后一次请求耗时和系统时间
这部分工作主要在客户端做。

首先介绍一下这次用到的两个类

package com.lagou.boot;

public class Metrics {

    private String nettyIp;
    private int nettyPort;
    private long start;
    private Long cost;

    public Metrics(String nettyIp, int nettyPort, long start, Long cost) {
        this.nettyIp = nettyIp;
        this.nettyPort = nettyPort;
        this.start = start;
        this.cost = cost;
    }

    public Metrics(String nettyIp, int nettyPort, long start) {
        this(nettyIp, nettyPort, start, null);
    }

    public String getNettyIp() {
        return nettyIp;
    }

    public void setNettyIp(String nettyIp) {
        this.nettyIp = nettyIp;
    }

    public int getNettyPort() {
        return nettyPort;
    }

    public void setNettyPort(int nettyPort) {
        this.nettyPort = nettyPort;
    }

    public long getStart() {
        return start;
    }

    public void setStart(long start) {
        this.start = start;
    }

    public Long getCost() {
        return cost;
    }

    public void setCost(Long cost) {
        this.cost = cost;
    }
}

RequestMetrics 类

COST_TIME_MAP变量 ip:端口 -》 耗时

REQUEST_ID_MAP变量 requestId -> ip + 端口 + 起始时间戳 + 耗时

calculate 方法用于 根据requestId 进行耗时统计

统计请求时间 在RpcClient的 send 方法中进行

public class RequestMetrics {

    /**
     * ip:端口 -》 耗时
     */
    private static final ConcurrentHashMap<String, Long> COST_TIME_MAP = new ConcurrentHashMap<>();

    /**
     * requestId -> ip + 端口 + 起始时间戳 + 耗时
     * 每个 requestId 用完一次后就会被销毁
     */
    private static final ConcurrentHashMap<String, Metrics> REQUEST_ID_MAP = new ConcurrentHashMap<>();

    private static final RequestMetrics requestMetrics = new RequestMetrics();

    public ConcurrentHashMap<String, Long> getMetricMap() {
        return COST_TIME_MAP;
    }

    private RequestMetrics() {
    }

    public static RequestMetrics getInstance() {
        return requestMetrics;
    }

    public void addNode(String nettyIp, int nettyPort) {
        COST_TIME_MAP.put(nettyIp + ":" + nettyPort, 0L);
    }

    public void remoteNode(String nettyIp, int nettyPort) {
        COST_TIME_MAP.remove(nettyIp + ":" + nettyPort);
    }

    /**
     * 响应时放入, 根据requestId 进行耗时统计
     * @param requestId
     */
    public void calculate(String requestId) {
        final Metrics metrics = REQUEST_ID_MAP.get(requestId);
        Long cost = System.currentTimeMillis() - metrics.getStart();
        COST_TIME_MAP.put(metrics.getNettyIp() + ":" + metrics.getNettyPort(), cost);
        REQUEST_ID_MAP.remove(requestId);
    }

    /**
     * 获取所有节点耗时统计
     */
    public List<Metrics> getAllInstances() {
        List<Metrics> result = new ArrayList<>();
        COST_TIME_MAP.forEach((url, aLong) -> {
            String[] split = url.split(":");
            result.add(new Metrics(split[0], Integer.parseInt(split[1]), aLong));
        });
        return result;
    }

    /**
     * 请求时放入
     * @param nettyIp
     * @param nettyPort
     * @param requestId
     */
    public void put(String nettyIp, int nettyPort, String requestId) {
        REQUEST_ID_MAP.put(requestId, new Metrics(nettyIp, nettyPort, System.currentTimeMillis(), null));
    }
}

2.消费者定时在启动时创建定时线程池,每隔5s自动上报,更新Zookeeper临时节点的值

ConsumerBootStrap 入口有一个参数配置

        // 之后会启动一个定时的线程池,每 5s 上传到注册中心
        configKeeper.setInterval(5);

ZkRegistryHandler 会开启一个 ScheduledExecutorService 线程池服务

RequestMetrics 的

       // 定时上报
        final ConfigKeeper configKeeper = ConfigKeeper.getInstance();
        final boolean providerSide = configKeeper.isProviderSide();
        final int interval = configKeeper.getInterval();
        if (!providerSide && interval > 0) {
            REPORT_WORKER.scheduleWithFixedDelay(new Runnable() {
                @Override
                public void run() {
                    System.out.println("我是一个定时任务");
                    // ...
                }
            }, interval, interval, TimeUnit.SECONDS);
        }
  1. 每次上报时判断当前时间距离最后一次请求是否超过5s,超过5s则需要删除Zookeeper上面的内容

这里介绍下 RequestMetrics 的 getAllInstances() 方法, 如果 5 秒内没有响应清空请求时间。

/**
     * 获取所有节点耗时统计
     */
    public List<Metrics> getAllInstances() {
        List<Metrics> result = new ArrayList<>();
        COST_TIME_MAP.forEach((url, aLong) -> {
            String[] split = url.split(":");
            result.add(new Metrics(split[0], Integer.parseInt(split[1]), aLong));
        });
        return result;
    }

接下来简单说一下负载均衡策略,这里主要涉及到了使用那个客户端进行服务的请求。

public abstract class AbstractLoadBalanceStrategy implements LoadBalanceStrategy{

    @Override
    public RpcClient route(Map<String, List<RpcClient>> clientPool, String serverClassName) {

        List<RpcClient> rpcClients = clientPool.get(serverClassName);
        if (null == rpcClients) return null;
        return doSelect(rpcClients);
    }

    protected abstract RpcClient doSelect(List<RpcClient> rpcClients);
}

MinCostLoadBalance (该类未经验证)

public class MinCostLoadBalance extends AbstractLoadBalanceStrategy {

    @Override
    protected RpcClient doSelect(final List<RpcClient> rpcClients) {
        ConcurrentHashMap<String, Long> metricMap = RequestMetrics.getInstance().getMetricMap();

        RpcClient minCostRpcClient = rpcClients.get(0);
        final Long minLong = metricMap.get(minCostRpcClient.getNettyIp() + minCostRpcClient.getNettyPort());

        for (int i = 1; i < rpcClients.size(); i++) {
            RpcClient rpcClient = rpcClients.get(i);

            String nettyIp = rpcClient.getNettyIp();
            int nettyPort = rpcClient.getNettyPort();

            // 取出最小响应时间的客户端,并进行调用
            Long aLong = metricMap.get(nettyIp + nettyPort);
            if (aLong != null && aLong < minLong) {
                minCostRpcClient = rpcClient;
            }
        }
        return minCostRpcClient;
    }
}

RandomLoadBalance (该类未经验证)

public class RandomLoadBalance extends AbstractLoadBalanceStrategy {
    private final Random random = new Random();

    @Override
    protected RpcClient doSelect(List<RpcClient> rpcClients) {
        int size = rpcClients.size();
        int index = random.nextInt(size);
        return rpcClients.get(index);
    }
}

作业3
以下项目主要使用了 commons-dbcp + fastjson + apache.curator 技术进行实现。

这里会通过create [-s][-e] path data acl命令创建节点:

建立所需节点

我会将数据库配置的用户名和密码等信息写入/dbConfig/lagou.config.DbConfig 节点中。

先建立父节点

create  /dbConfig  ""

然后若不存在临时节点则重新创建

# 向 /dbConfig/lagou.config.DbConfig 中写入配置信息
create -e /dbConfig/lagou.config.DbConfig {"username":"root","password":"123456","url":"jdbc:mysql://localhost:3306/aaaa?serverTimezone=UTC"}

# 查看是否能正常获取节点信息
get /dbConfig/lagou.config.DbConfig 

更改数据

# 更改数据库为 aaaa
set /dbConfig/lagou.config.DbConfig {"username":"root","password":"123456","url":"jdbc:mysql://localhost:3306/aaaa?serverTimezone=UTC"}

# 更改数据库为 bbbb
set /dbConfig/lagou.config.DbConfig {"username":"root","password":"123456","url":"jdbc:mysql://localhost:3306/bbbb?serverTimezone=UTC"}

创建Java类

  1. 创建工具类 RuntimeContext
@Component
public class RuntimeContext implements ApplicationContextAware {

    private static ApplicationContext applicationContext = null;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        if (RuntimeContext.applicationContext == null) {
            RuntimeContext.applicationContext = applicationContext;
        }
    }

    //获取applicationContext
    private static ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    //通过name获取Bean
    public static Object getBean(String name) {
        return getApplicationContext().getBean(name);
    }

    // ...
}
  1. 实体类
package lagou.config;

public class DbConfig {

    private String url;

    private String username;

    private String password;

    // setter getter 方法
}
  1. 创建 MyDataSource 自定义数据源,我们重写的dbcp中的类BasicDataSource,我们将其全部拷贝了出来,然后重命名为MyDataSource类,然后在其中修改了以下内容

3.1将 UNKNOWN_TRANSACTIONISOLATION 的值改为 -1. 否则这个内部变量会找不到

/**
     * The default TransactionIsolation state of connections created by this pool.
     */
    protected volatile int defaultTransactionIsolation =            PoolableConnectionFactory.UNKNOWN_TRANSACTIONISOLATION;

3.2jdk 1.7 之后需要实现该方法 getParentLogger()

@Override
    public Logger getParentLogger() throws SQLFeatureNotSupportedException {
        return null;
    }

3.3 修改已有的 createDataSource() 方法,删除这几行代码


3.4 新增 changeDataSource 方法

public static void changeDataSource() {
        MyDataSource dataSource = (MyDataSource) RuntimeContext.getBean("dataSource");
        try {
            dataSource.close();
            dataSource.createDataSource();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
  1. 新建 InitListener 类,该类实现了ServletContextListener 来对Zookeeper节点 /db/url 的监听
public class InitListener implements ServletContextListener {

    private static final String CONNENT_ADDR = "localhost:2181";
    private static final String PATH = "/dbConfig";
    private static final String SUB_PATH = PATH + "/" + DbConfig.class.getName();

    @Override
    public void contextInitialized(ServletContextEvent sce) {
        CuratorFramework curatorFramework = CuratorFrameworkFactory.builder()
                .connectString(CONNENT_ADDR)
                .retryPolicy(new ExponentialBackoffRetry(1000, 3))
                .build();
        curatorFramework.start();

        PathChildrenCache nodeCache = new PathChildrenCache(curatorFramework, "/dbConfig", true);
        try {
            nodeCache.getListenable().addListener((client, pathChildrenCacheEvent) -> {
                System.out.println(pathChildrenCacheEvent.getType());

                if (PathChildrenCacheEvent.Type.CHILD_UPDATED.equals(pathChildrenCacheEvent.getType())) {
                    final ChildData data = pathChildrenCacheEvent.getData();
                    if (data != null) {
                        final String path = data.getPath();
                        System.out.println(path);
                        System.out.println(SUB_PATH);
                        if (path.equals(SUB_PATH)) {
                            MyDataSource datasource = (MyDataSource) RuntimeContext.getBean("dataSource");

                            final DbConfig dbConfig = JSON.parseObject(new String(data.getData()), DbConfig.class);
                            System.out.println(dbConfig.toString());
                            datasource.setUrl(dbConfig.getUrl());
                            datasource.setUsername(dbConfig.getUsername());
                            datasource.setPassword(dbConfig.getPassword());

                            MyDataSource.changeDataSource();
                        }
                    }
                }
            });
            nodeCache.start(PathChildrenCache.StartMode.BUILD_INITIAL_CACHE);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Override
    public void contextDestroyed(ServletContextEvent sce) {
    }
}
  1. 修改spring boot 启动类,注册 InitListener ,配置 我们自定义的 DataSource,建立一个query方法(可通过/query 进行访问)暴露出去。
@SpringBootApplication
@RestController
public class MyApplication {

    public static void main(String[] args) {
        SpringApplication.run(MyApplication.class, args);
    }

    @RequestMapping("/query")
    public String query() {
        String sql = "select id from `info` limit 1";
        return jdbcTemplate.queryForObject(sql, String.class);
    }

    @Bean
    public ServletListenerRegistrationBean servletListenerRegistrationBean() {
        ServletListenerRegistrationBean servletListenerRegistrationBean = new ServletListenerRegistrationBean();
        servletListenerRegistrationBean.setListener(new InitListener());
        return servletListenerRegistrationBean;
    }

    @Bean
    public DataSource dataSource(@Value("${spring.datasource.url}") String url,
                                 @Value("${spring.datasource.username}") String username,
                                 @Value("${spring.datasource.password}") String password) {
        MyDataSource dataSource = new MyDataSource();
        dataSource.setUrl(url);
        dataSource.setUsername(username);
        dataSource.setPassword(password);
        return dataSource;
    }

    @Autowired
    private JdbcTemplate jdbcTemplate;
}

6.application.properties 配置

server.port=80

spring.datasource.url=jdbc:mysql://localhost:3306/aaaa?serverTimezone=GMT%2B8
spring.datasource.username=root
spring.datasource.password=123456

验证
http://localhost/query

参考

基于Zookeeper动态切换数据源_BXS_0107的博客-CSDN博客
https://blog.csdn.net/newbie0107/article/details/105500579

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