2019-07-24 Motan RPC 框架分析

Sina Motan RPC 框架分析 原文写于2年前

1. 场景

  • motan是微博开源的一套轻量级、方便使用的RPC框架
  • 项目地址:https://github.com/weibocom/motan
  • Motan 是基于 Java 的高性能轻量级 RPC 框架,
    • 其具备实用的服务治理功能和 RPC 协议扩展能力。
    • 服务发现灵活支持多种配置管理组件,基于高并发高负载场景的高可用策略优化,
    • 良好的 SPI(Service Provider Interface)扩展,详细的调用统计,灵活支持多种 RPC 传输协议,在使用上,
    • 无缝支持Spring 配置方式,通过简单灵活的配置即可快速接入使用

2 架构图 (图片来自网上)

  • 整体结构

RPC
  • 分层结构

Motan

3 RPC框架要重点核心点

推荐从motan看RPC框架实现

核心内容

  1. 如何与Spring做集成?
  2. SPI机制的实现原理是什么?为什么要用这个?
  3. 这么多的配置,究竟是如何在类间传递和保存的?
  4. 服务降级是如何做的?
  5. motan协议究竟是什么样子的?
  6. 服务发现和注册究竟是怎么做的?或者说客户端是如何找到服务端的?
  7. 为什么要用动态代理,到底proxy了什么?
  8. cluster的负载均衡和ha究竟是如何实现的?
  9. transport层是怎么进行抽象的?
  10. client对server的心跳检测是怎么实现的呢?
  11. client端的连接池是如何管理的?
  12. server端的连接是如何共享的?

源代码分析关注点下几点

  1. SPI机制的实现原理是什么?为什么要用这个?
  2. 这么多的配置,究竟是如何在类间传递和保存的?
  3. motan协议究竟是什么样子的?
  4. 为什么要用动态代理,到底proxy了什么?
  5. transport层是怎么进行抽象的?
  6. transport层是怎么进行抽象的?
  7. client对server的心跳检测是怎么实现的呢?
  8. client端的连接池是如何管理的?
  9. server端的连接是如何共享的?

我将从以上几点对motan的源代码进行分析debug
由于spring集成与解决涉及的内容比较多与杂,暂时先不作分析

RPC本质的理解

  1. 对于软件工程师来讲,形如object.method()的方法调用实在是太过熟悉,
    当我们在同一个JVM进程内执行方法调用的时候,一切都显得顺其自然。
    然而如果我们将上述的调用过程拆分成两个部分—-方法的调用端和方法的实现端,
    然后将他们分别放置到不同的进程中,使得调用端和实现端能够做到跨操作系统,跨网络。这便是RPC的本质。

  2. 从功能角度来讲,RPC框架可以分为服务治理型和多语言型。motan显然属于前者,
    因此对motan框架可以简单的理解为:分离方法的调用和实现,并具双端服务治理功能。

  1. 在同一jvm中就是 method.invoke(),而跨vm与网络时,则涉及到对象与数据的交换,而client方只用到了一个接口而已。
  2. 如果打电话,通过电话号找到远程的你call you ,而你我分属不同的vm进程

Quick Start

入门例子

  1. server demo 服务
     import com.weibo.api.motan.common.MotanConstants;.
     import ...
     public class MotanApiExportDemo {
         public static void main(String[] args) throws InterruptedException {
             ServiceConfig<MotanDemoService> motanDemoService = new ServiceConfig<MotanDemoService>();
     
             // 设置接口及实现类
             motanDemoService.setInterface(MotanDemoService.class);
             motanDemoService.setRef(new MotanDemoServiceImpl());
     
             // 配置服务的group以及版本号
             motanDemoService.setGroup("motan-demo-rpc");
             motanDemoService.setVersion("1.0");
     
             // 配置注册中心直连调用
             // RegistryConfig directRegistry = new RegistryConfig();
             // directRegistry.setRegProtocol("local");
             // directRegistry.setCheck("false"); //不检查是否注册成功
             // motanDemoService.setRegistry(directRegistry);
     
             // 同志我本机没有安装 zk,因此将zk的地上注释 ,此时motan将默认使用local来注册
             RegistryConfig zookeeperRegistry = new RegistryConfig();
             // 配置ZooKeeper注册中心
            //   zookeeperRegistry.setRegProtocol("zookeeper");
            //   zookeeperRegistry.setAddress("127.0.0.1:2181");
             motanDemoService.setRegistry(zookeeperRegistry);
     
             // 配置RPC协议
             ProtocolConfig protocol = new ProtocolConfig();
             protocol.setId("motan");
             protocol.setName("motan");
             motanDemoService.setProtocol(protocol);
     
             motanDemoService.setExport("motan:8002");
             motanDemoService.export();
     
             MotanSwitcherUtil.setSwitcherValue(MotanConstants.REGISTRY_HEARTBEAT_SWITCHER, true);
     
             System.out.println("server start...");
         }
     }
    
  • ServiceConfig 显露外部服务的公用类,如果是用spring集成,将会使用其子类 ServiceConfigBean,暂时不讨论
    1. ServiceConfig 有两个属性 interface ,ref 分别代码接口与实现类。然后指名服务 的分组与版本.
    2. RegistryConfig 代码服务生成后,注册地上,暂时不用zk,因此注释设置相关属性的地方。
    3. ProtocolConfig 服务是用保种协议进行传输与调用的 motan
    4. ServiceConfig 设置~ setRegistryConfig,setProtocolConfig~ ,然后调用export就实现了
  1. client调用
          
     public class MotanApiClientDemo {
     
         public static void main(String[] args) throws InterruptedException {
             RefererConfig<MotanDemoService> motanDemoServiceReferer = new RefererConfig<MotanDemoService>();
     
             // 设置接口及实现类
             motanDemoServiceReferer.setInterface(MotanDemoService.class);
     
             // 配置服务的group以及版本号
             motanDemoServiceReferer.setGroup("motan-demo-rpc");
             motanDemoServiceReferer.setVersion("1.0");
             motanDemoServiceReferer.setRequestTimeout(300);
     
             // 配置注册中心直连调用
             // RegistryConfig directRegistry = new RegistryConfig();
             // directRegistry.setRegProtocol("local");
             // motanDemoServiceReferer.setRegistry(directRegistry);
     
             // 配置ZooKeeper注册中心
             RegistryConfig zookeeperRegistry = new RegistryConfig();
     //        zookeeperRegistry.setRegProtocol("zookeeper");
     //        zookeeperRegistry.setAddress("127.0.0.1:2181");
             motanDemoServiceReferer.setRegistry(zookeeperRegistry);
     
             // 配置RPC协议
             ProtocolConfig protocol = new ProtocolConfig();
             protocol.setId("motan");
             protocol.setName("motan");
             motanDemoServiceReferer.setProtocol(protocol);
             motanDemoServiceReferer.setDirectUrl("localhost:8002");  // 注册中心直连调用需添加此配置
     
             // 使用服务
             MotanDemoService service = motanDemoServiceReferer.getRef();
             for (int i=0;i<1000;i++) {
                 System.out.println(service.hello("motan " +i));
                 Thread.sleep(1000);
             }
     
             System.exit(0);
         }
     }
    
    
  • RefererConfig 客户端对motan服务的一个封装,如果是用spring集成,将会使用其子类 RefererConfigBean,暂时不讨论
    1. RefererConfig 有两个属性 interface ,ref 分别代码接口与实现类。然后指名服务 的分组与版本.
    2. RegistryConfig 代码服务生成后,注册地上,暂时不用zk,因此注释设置相关属性的地方。
    3. ProtocolConfig 服务是用保种协议进行传输与调用的 motan
    4. RefererConfig#gerRef().hello() 通过远程调用返回 hello motan

    这里使用setDirectUrl localhost:8002 主要是因为client,server在同一台机器上,但是端口仍然监听


公共组件部分

  1. SPI:Service Provider Interface,主要通过ExtensionLoader提供扩展点功能,用来动态装载接口具体实现,以提供客户端扩展能力。
  2. Logger:使用slf4j,提供整个框架统一的日志记录工具。
  3. Statistic:使用定时回调方式,收集和记录框架监控信息。
  4. URL:非Jdk里的URL。他对协议,路径,参数的统一抽象,系统也是使用URL来保存和读取配置信息的。
  5. Switcher:提供开关控制服务,能够控制框架关键路径的升级和降级。
  6. Exception:统一异常处理,分为业务异常,框架异常,服务异常等。

层的作用:

  1. Config层:主要提供了配置读取,解析,实体生成。同时他也是整个框架的入口,
  2. Proxy层:服务端无proxy,客户端具有代理功能,他通过InvocationHandler来拦截方法调用。目前只使用了jdk原生动态代理工具。
  3. Registry层:用来进行服务发现和注册,server端进行服务的注册,client进行服务发现。目前有zk,consul的实现,还有对directUrl(就是p2p,不借助中心)的特殊处理。
  4. Cluster层:集群功能,他提供了集群的服务暴露、服务引用、负载均衡,HA等。
  5. Protocol层:协议层,目前只支持injvm和motan协议,主要就是提供给下层协议编码和解析,并且使用filter/pipe模式进行访问日志,最大并发量控制等功能。
  6. Transport层:主要处理网络连接,他主要抽象了Client和Server接口。使用netty框架。

Server端

要关注的问题

  1. 配置解析与传递
    2。服务URL与注册URL
  2. SPI 机制
  3. transport 封装
  4. 服务暴露
  5. 类图

从ServiceConfig.export()开始。

类图

[图片上传失败...(image-b3f4e3-1563956674836)]

ServiceConfig 主要逻辑:


  
public class ServiceConfig<T> extends AbstractServiceConfig {
    private static ConcurrentHashSet<String> existingServices = new ConcurrentHashSet<String>();
    // 具体到方法的配置
    protected List<MethodConfig> methods;
    // 接口实现类引用
    private T ref;

    // service 对应的exporters,用于管理service服务的生命周期
    private List<Exporter<T>> exporters = new CopyOnWriteArrayList<Exporter<T>>();
    private Class<T> interfaceClass;
    private BasicServiceInterfaceConfig basicService;
    private AtomicBoolean exported = new AtomicBoolean(false);
    // service的用于注册的url,用于管理service注册的生命周期,url为regitry url,内部嵌套service url。
    private ConcurrentHashSet<URL> registereUrls = new ConcurrentHashSet<URL>();

    protected boolean serviceExists(URL url) {
        return existingServices.contains(url.getIdentity());
    }

    public synchronized void export() {
        if (exported.get()) {
            return;
        }
        // [1]
        checkInterfaceAndMethods(interfaceClass, methods);
        // [2]  AbstractInterfaceConfig#loadRegistryUrls  父类的方法来收集
        List<URL> registryUrls = loadRegistryUrls();
        if (registryUrls == null || registryUrls.size() == 0) {
            throw new IllegalStateException("Should set registry config for service:" + interfaceClass.getName());
        }

        Map<String, Integer> protocolPorts = getProtocolAndPort();
        for (ProtocolConfig protocolConfig : protocols) {
            Integer port = protocolPorts.get(protocolConfig.getId());
            if (port == null) {
                throw new MotanServiceException();
            }
            // [3]
            doExport(protocolConfig, port, registryUrls);
        }
        afterExport();
    }

    @SuppressWarnings("unchecked")
    private void doExport(ProtocolConfig protocolConfig, int port, List<URL> registryURLs) {
        String protocolName = protocolConfig.getName();
        if (protocolName == null || protocolName.length() == 0) {
            protocolName = URLParamType.protocol.getValue();
        }

        String hostAddress = host;
        if (StringUtils.isBlank(hostAddress) && basicService != null) {
            hostAddress = basicService.getHost();
        }
        if (NetUtils.isInvalidLocalHost(hostAddress)) {
            hostAddress = getLocalHostAddress(registryURLs);
        }

        Map<String, String> map = new HashMap<String, String>();

        map.put(URLParamType.nodeType.getName(), MotanConstants.NODE_TYPE_SERVICE);
        map.put(URLParamType.refreshTimestamp.getName(), String.valueOf(System.currentTimeMillis()));

        collectConfigParams(map, protocolConfig, basicService, extConfig, this);
        collectMethodConfigParams(map, this.getMethods());

        // {id=motan, export=motan:8002, protocol=motan, refreshTimestamp=1505801631928, group=motan-demo-rpc, nodeType=service, version=1.0}
        System.out.println(" 7 service param ==>" +map);
        URL serviceUrl = new URL(protocolName, hostAddress, port, interfaceClass.getName(), map);
        // [4]
        if (serviceExists(serviceUrl)) {
            throw new MotanFrameworkException();
        }

        List<URL> urls = new ArrayList<URL>();

        // injvm 协议只支持注册到本地,其他协议可以注册到local、remote
        if (MotanConstants.PROTOCOL_INJVM.equals(protocolConfig.getId())) {
            URL localRegistryUrl = null;
            for (URL ru : registryURLs) {
                if (MotanConstants.REGISTRY_PROTOCOL_LOCAL.equals(ru.getProtocol())) {
                    localRegistryUrl = ru.createCopy();
                    break;
                }
            }
            if (localRegistryUrl == null) {
                localRegistryUrl =
                        new URL(MotanConstants.REGISTRY_PROTOCOL_LOCAL, hostAddress, MotanConstants.DEFAULT_INT_VALUE,
                                RegistryService.class.getName());
            }
            urls.add(localRegistryUrl);
        } else {
            // TODO 我使用的是 local,因此会走到此处
            for (URL ru : registryURLs) {
                System.out.println(" ru "+ ru);
                urls.add(ru.createCopy());
            }
        }

        for (URL u : urls) {
            u.addParameter(URLParamType.embed.getName(), StringTools.urlEncode(serviceUrl.toFullStr()));
            registereUrls.add(u.createCopy());
        }
        // [local://127.0.0.1:0/com.weibo.api.motan.registry.RegistryService?group=default_rpc]
        System.out.println(" 8 registereUrls ==>" +registereUrls);
        ConfigHandler configHandler = ExtensionLoader.getExtensionLoader(ConfigHandler.class).getExtension(MotanConstants.DEFAULT_VALUE);
        // ~ [5]  ConfigHandler 如何加载 ~
        exporters.add(configHandler.export(interfaceClass, ref, urls));
        initLocalAppInfo(serviceUrl);
    }

    private void afterExport() {
        exported.set(true);
        for (Exporter<T> ep : exporters) {
            existingServices.add(ep.getProvider().getUrl().getIdentity());
        }
    }

    private void afterUnexport() {
        exported.set(false);
        for (Exporter<T> ep : exporters) {
            existingServices.remove(ep.getProvider().getUrl().getIdentity());
            exporters.remove(ep);
        }
        exporters.clear();
        registereUrls.clear();
    }

    @ConfigDesc(excluded = true)
    public BasicServiceInterfaceConfig getBasicService() {
        return basicService;
    }

    public Map<String, Integer> getProtocolAndPort() {
        return ConfigUtil.parseExport(this.export);
    }

    @ConfigDesc(excluded = true)
    public String getHost() {
    }

}
 
  1. 无论是spring方式,还是api的方式,最终会调用export()方法

  1. loadRegistryUrls():返回注册中心地址,是一个list,打印出来后的结果如下:

  1. 执行 ServiceConfig#doExport() 方法
    1. 根据一系统 参数构造一个服务 service URL,请注意(motan中有register url ,servcieurl)

      URL serviceUrl = new URL(protocolName, hostAddress, port, interfaceClass.getName(), map);

    2. 此URL对象模仿了 java 标准api的方法 : protocol url param,后面再分析这个URL,这个是整个motan协议结构的中心
    3. 生成了 service url,然后,通过sip的方式,找到ConfigHandler的实现类 SimpleConfigHandler
      • ConfigHandler configHandler = ExtensionLoader.getExtensionLoader(ConfigHandler.class).getExtension(MotanConstants.DEFAULT_VALUE);
      • 通过configHandler个哦执行真正的export exporters.add(configHandler.export(interfaceClass, ref, urls))
        • configHandler.export(interfaceClass, ref, urls)

          将 interfaceClass, ref,向多个注册中心,注册

SimpleConfigHandler 执行服务暴露

[图片上传失败...(image-af38f-1563956674836)]


@SpiMeta(name = MotanConstants.DEFAULT_VALUE)
public class SimpleConfigHandler implements ConfigHandler {

    @Override
    public <T> ClusterSupport<T> buildClusterSupport(Class<T> interfaceClass, List<URL> registryUrls) {
        ClusterSupport<T> clusterSupport = new ClusterSupport<T>(interfaceClass, registryUrls);
        clusterSupport.init();
        return clusterSupport;
    }

    @Override
    public <T> T refer(Class<T> interfaceClass, List<Cluster<T>> clusters, String proxyType) {
        ProxyFactory proxyFactory = ExtensionLoader.getExtensionLoader(ProxyFactory.class).getExtension(proxyType);
        return proxyFactory.getProxy(interfaceClass, new RefererInvocationHandler<T>(interfaceClass, clusters));
    }

    @Override
    public <T> Exporter<T> export(Class<T> interfaceClass, T ref, List<URL> registryUrls) {
        // motan://10.14.162.137:8002/com.weibo.motan.demo.service.MotanDemoService?id=motan&export=motan:8002&protocol=motan&refreshTimestamp=1505801631928&group=motan-demo-rpc&nodeType=service&version=1.0&
        String serviceStr = StringTools.urlDecode(registryUrls.get(0).getParameter(URLParamType.embed.getName()));
        System.out.println("5 SimpleConfigHandler export serviceStr ==>" +serviceStr);

        URL serviceUrl = URL.valueOf(serviceStr);
        // export service
        // 利用protocol decorator来增加filter特性
        String protocolName = serviceUrl.getParameter(URLParamType.protocol.getName(), URLParamType.protocol.getValue());
        Protocol ss = ExtensionLoader.getExtensionLoader(Protocol.class).getExtension(protocolName);
        Protocol protocol = new ProtocolFilterDecorator(ss);

        //    com.weibo.api.motan.protocol.rpc.DefaultRpcProtocol@33f724c7
        System.out.println(" 11 default Protocol ==> " + ss);
        Provider<T> provider = new DefaultProvider<T>(ref, serviceUrl, interfaceClass);
        Exporter<T> exporter = protocol.export(provider, serviceUrl);
        // register service
        register(registryUrls, serviceUrl);

        return exporter;
    }

    private void register(List<URL> registryUrls, URL serviceUrl) {
        for (URL url : registryUrls) {
            // 根据check参数的设置,register失败可能会抛异常,上层应该知晓
            RegistryFactory registryFactory = ExtensionLoader.getExtensionLoader(RegistryFactory.class).getExtension(url.getProtocol());
            if (registryFactory == null) {
                throw  new  Excepiton(..);
            }
            Registry registry = registryFactory.getRegistry(url);
            registry.register(serviceUrl);
        }
    }

    private void unRegister(Collection<URL> registryUrls) {
        for (URL url : registryUrls) {
                 // 不管check的设置如何,做完所有unregistry,做好清理工作
                String serviceStr = StringTools.urlDecode(url.getParameter(URLParamType.embed.getName()));
                URL serviceUrl = URL.valueOf(serviceStr);
                RegistryFactory registryFactory = ExtensionLoader.getExtensionLoader(RegistryFactory.class).getExtension(url.getProtocol());
                Registry registry = registryFactory.getRegistry(url);
                registry.unregister(serviceUrl);
            
        }
    }

}

SimpleConfigHandler#export()方法执行服务暴露(refer方法是客户端使用,暂时不讨论)
  1. motan服务协议的是怎么样的?

  2. 服务协议初步分析:类似于java 标准api URL类的方式

    • motan 表示用motan协议
    • 192.168.1.1:8002: 当前服务的ip:port
    • com.weibo.motan.demo.service.MotanDemoService :接口名
    • ?号后面是参数,包括timestamp id version,group 等信息
  3. 将 Service url字符串转换成 motan URL对象

    • URL serviceUrl = URL.valueOf(serviceStr);
  4. 根据第3的URL对象。通过SPI机制找到对应的 Protocol 实现类

    • String protocolName = serviceUrl.getParameter(URLParamType.protocol.getName(), URLParamType.protocol.getValue());
    • Protocol ss = ExtensionLoader.getExtensionLoader(Protocol.class).getExtension(protocolName);

      com.weibo.api.motan.protocol.rpc.DefaultRpcProtocol@33f724c7

    • 使用decorate模拟,增强Protocol功能,添加过滤器,后面我在分析

      Protocol protocol = new ProtocolFilterDecorator(ss);

    • 使用默认Provider实现来暴露服务 DefaultProvider ,此为将是service impl的调用就是一个invoke而已
      1. Provider<T> provider = new DefaultProvider<T>(ref, serviceUrl, interfaceClass);
      2. DefaultProvider 本质上是通过 invoke(request) 根据reqeust 来找到对应的method,反射调用方法,仅此而已

    -- 如何将provider暴露 · Exporter<T> exporter = protocol.export(provider, serviceUrl);·

    服务暴露成功,则将服务添加 到对应注册中心地址。先export 再register

DefaultProvider 如何封装服务端调用:原理
  1. 源代码

@SpiMeta(name = "motan")
public class DefaultProvider<T> extends AbstractProvider<T> {
   protected T proxyImpl;

   // TODO SIimpleConfigHandler 中调用引构造器
   public DefaultProvider(T proxyImpl, URL url, Class<T> clz) {
       super(url, clz);
       this.proxyImpl = proxyImpl;
   }

   @Override
   public T getImpl(){   return proxyImpl;  }

   @Override
   public Response invoke(Request request) {
       DefaultResponse response = new DefaultResponse();
       // TODO netty 接收客户端请求,从中解析出方法名
       Method method = lookup(request);
       if (method == null) {
           // ... return ;
       }
       try {
           // TODO 因为defaultprovider中已经知道是这个接口实现类了,因此 只需要invoke即可完成
           Object value = method.invoke(proxyImpl, request.getArguments());
           response.setValue(value);
       } catch (Exception e) {/服务发生错误时,显示详细日志
       } catch (Throwable t) {
           // 如果服务发生Error,将Error转化为Exception,防止拖垮调用方         //对于Throwable,也记录日志
       }
       // 传递rpc版本和attachment信息方便不同rpc版本的codec使用。
       response.setRpcProtocolVersion(request.getRpcProtocolVersion());
       response.setAttachments(request.getAttachments());
       return response;
   }
}

[图片上传失败...(image-50c6d7-1563956674836)]

DefaultRpcProtocol 如何暴露 DefaultProvider
  1. 翠花上码

public class ProtocolFilterDecorator implements Protocol {
       public ProtocolFilterDecorator(Protocol protocol) {
            this.protocol = protocol; ///  TODO  默认使用  DefaultRpcProtocol 
        }
        @Override
        public <T> Exporter<T> export(Provider<T> provider, URL url) {
           // TODO 调用父类  AbstractProtocol#export方法
            return protocol.export(decorateWithFilter(provider, url), url);
        }
}

AbstractProtocol.java

public abstract class AbstractProtocol implements Protocol {
    protected ConcurrentHashMap<String, Exporter<?>> exporterMap = new ConcurrentHashMap<String, Exporter<?>>();

    @SuppressWarnings("unchecked")
    @Override
    public <T> Exporter<T> export(Provider<T> provider, URL url) {
        String protocolKey = MotanFrameworkUtil.getProtocolKey(url);
        synchronized (exporterMap) {
            Exporter<T> exporter = (Exporter<T>) exporterMap.get(protocolKey);
            // TODO  此处会回调子类    DefaultRpcProtocol#createExporter
            exporter = createExporter(provider, url);
            // TODO 回调 AbstractNode#init()
            exporter.init();
            exporterMap.put(protocolKey, exporter);
            return exporter;
        }
    }

    public <T> Referer<T> refer(Class<T> clz, URL url) {  }

    @Override
    public <T> Referer<T> refer(Class<T> clz, URL url, URL serviceUrl) {   // 略
    }

    protected abstract <T> Exporter<T> createExporter(Provider<T> provider, URL url);
    protected abstract <T> Referer<T> createReferer(Class<T> clz, URL url, URL serviceUrl);
}

DefaultRpcProtocol.java


@SpiMeta(name = "motan")
public class DefaultRpcProtocol extends AbstractProtocol {

    // 多个service可能在相同端口进行服务暴露,因此来自同个端口的请求需要进行路由以找到相应的服务,同时不在该端口暴露的服务不应该被找到
    private Map<String, ProviderMessageRouter> ipPort2RequestRouter = new HashMap<String, ProviderMessageRouter>();

    @Override
    protected <T> Exporter<T> createExporter(Provider<T> provider, URL url) {
        return new DefaultRpcExporter<T>(provider, url);
    }

    @Override // 实现父类 
    protected <T> Referer<T> createReferer(Class<T> clz, URL url, URL serviceUrl) {
        return new DefaultRpcReferer<T>(clz, url, serviceUrl);
    }

    class DefaultRpcExporter<T> extends AbstractExporter<T> {
        private Server server;
        private EndpointFactory endpointFactory;
        // TODO SPI 查找 endpointFactory ,用此创建 server  
        public DefaultRpcExporter(Provider<T> provider, URL url) {
            super(provider, url);
            ProviderMessageRouter requestRouter = initRequestRouter(url);
            endpointFactory =  ExtensionLoader.getExtensionLoader(EndpointFactory.class).getExtension(
                            url.getParameter(URLParamType.endpointFactory.getName(), URLParamType.endpointFactory.getValue()));
            server = endpointFactory.createServer(url, requestRouter);
        }

        @Override
        public void unexport() {
            String protocolKey = MotanFrameworkUtil.getProtocolKey(url);
            String ipPort = url.getServerPortStr();
            Exporter<T> exporter = (Exporter<T>) exporterMap.remove(protocolKey);
            if (exporter != null) {
                exporter.destroy();
            }
            synchronized (ipPort2RequestRouter) {
                ProviderMessageRouter requestRouter = ipPort2RequestRouter.get(ipPort);
                if (requestRouter != null) {
                    requestRouter.removeProvider(provider);
                }
            }
        }

        @Override
        protected boolean doInit() {   server.open();  }

        private ProviderMessageRouter initRequestRouter(URL url) {
            ProviderMessageRouter requestRouter = null;
            String ipPort = url.getServerPortStr();
            synchronized (ipPort2RequestRouter) { //  TODO此处后面再分析 
                requestRouter = ipPort2RequestRouter.get(ipPort);
                if (requestRouter == null) {
                    requestRouter = new ProviderProtectedMessageRouter(provider);
                    ipPort2RequestRouter.put(ipPort, requestRouter);
                } else {
                    requestRouter.addProvider(provider);
                }
            }
            return requestRouter;
        }
    }

    class DefaultRpcReferer<T> extends AbstractReferer<T> {
        // TODO  先略去 
    }
}


  1. ProtocolFilterDecorator(DefaultRpcProtocol proto)
  2. 调用 AbstractProtocol.export();
  3. export()回调子类 createExporter();
  4. 创建内部类 DefaultRpcExporter
  5. 完成exporter实现

[图片上传失败...(image-a3794b-1563956674836)]

DefaultRpcExporter 是如何 暴露的
DefaultRpcExporter.java
class DefaultRpcExporter<T> extends AbstractExporter<T> {
    private Server server;
    private EndpointFactory endpointFactory;

    public DefaultRpcExporter(Provider<T> provider, URL url) {
        // TODO 1 调用父类  AbstractExporter
        super(provider, url);
        // TODO 2 requestRouter 其作用是将 request 映射到provider上
        ProviderMessageRouter requestRouter = initRequestRouter(url);
       // TODO  3 通过SPI找到 endpoint ,相当于底层传输层实现,
        endpointFactory =   ExtensionLoader.getExtensionLoader(EndpointFactory.class).getExtension(
                        url.getParameter(URLParamType.endpointFactory.getName(), URLParamType.endpointFactory.getValue()));
        // TODO 4 根据endpointfactory创建服务器,Server是一个接口,默认实现是NettyServer.java后面专门会分析
        server = endpointFactory.createServer(url, requestRouter);
    }
}
public abstract class AbstractNode implements Node {
    @Override
    public synchronized void init() {
        if (init) {    return;    }
        boolean result = doInit();
    }
    // TODO DefaultRpcExporter会实现此方法
    protected abstract boolean doInit();
}
  1. 整体调用链
  • --> ProtocolFilterDecorator#export()
    • --> AbstractProtocol#export()
      • --> 1. exporter = abstract createExporter(provider, url)
        • --> DefaultRpcProtocol#createExporter() 回调子类的createExporter
      • <-- 2. exporter.init() 父类继续调用 exporter 的init方法
        • --> AbstractNode#init();
          • --> DefaultRpcExporter#doInit(); 做真实的doInit打开
          • --> 打开服务端server server.open()
      • <-- server 的初始化完成
  1. endpointFactory.createServer(url, requestRouter)是什么?
  1. 实现类是 NettyEndpointFactory
  2. 父类会回调 此类的 innerCreateClient()方法实例化NettyServer对象
@SpiMeta(name = "motan")
public class NettyEndpointFactory extends AbstractEndpointFactory {
    @Override
    protected Server innerCreateServer(URL url, MessageHandler messageHandler) {
        return new NettyServer(url, messageHandler);
    }

   //  TODO 此处会被你类回调 
    @Override
    protected Client innerCreateClient(URL url) {
        return new NettyClient(url);
    }
}


public abstract class AbstractEndpointFactory implements EndpointFactory {

    /** 维持share channel 的service列表 **/
    protected Map<String, Server> ipPort2ServerShareChannel = new HashMap<String, Server>();
    protected ConcurrentMap<Server, Set<String>> server2UrlsShareChannel = new ConcurrentHashMap<Server, Set<String>>();
    private EndpointManager heartbeatClientEndpointManager = null;

    public AbstractEndpointFactory() {
        // TODO  服务端Hheartbeat 
        heartbeatClientEndpointManager = new HeartbeatClientEndpointManager();
        heartbeatClientEndpointManager.init();
    }

    @Override
    public Server createServer(URL url, MessageHandler messageHandler) {
        HeartbeatFactory heartbeatFactory = getHeartbeatFactory(url);
        messageHandler = heartbeatFactory.wrapMessageHandler(messageHandler);
        synchronized (ipPort2ServerShareChannel) {
            String ipPort = url.getServerPortStr();
            System.out.println("createServer "+ipPort);
            String protocolKey = MotanFrameworkUtil.getProtocolKey(url);
            boolean shareChannel = url.getBooleanParameter(URLParamType.shareChannel.getName(), URLParamType.shareChannel.getBooleanValue());
            if (!shareChannel) { // 独享一个端口
                // 如果端口已经被使用了,使用该server bind 会有异常
                return innerCreateServer(url, messageHandler);
            }
            Server server = ipPort2ServerShareChannel.get(ipPort);
            if (server != null) {
                // can't share service channel
                if (!MotanFrameworkUtil.checkIfCanShallServiceChannel(server.getUrl(), url)) {
                }
                saveEndpoint2Urls(server2UrlsShareChannel, server, protocolKey);
                return server;
            }

            url = url.createCopy();
            url.setPath(""); // 共享server端口,由于有多个interfaces存在,所以把path设置为空
            // TODO  回调 NettyEndpointFactory# innerCreateServer
            server = innerCreateServer(url, messageHandler);
            ipPort2ServerShareChannel.put(ipPort, server);
            saveEndpoint2Urls(server2UrlsShareChannel, server, protocolKey);
            return server;
        }
    }

    @Override
    public Client createClient(URL url) {
        return createClient(url, heartbeatClientEndpointManager);
    }

    @Override
    public void safeReleaseResource(Server server, URL url) {
        safeReleaseResource(server, url, ipPort2ServerShareChannel, server2UrlsShareChannel);
    }

    @Override
    public void safeReleaseResource(Client client, URL url) { }

    private <T extends Endpoint> void safeReleaseResource(T endpoint, URL url, Map<String, T> ipPort2Endpoint,ConcurrentMap<T, Set<String>> endpoint2Urls) {
    }

    private <T> void saveEndpoint2Urls(ConcurrentMap<T, Set<String>> map, T endpoint, String namespace) { }

    private HeartbeatFactory getHeartbeatFactory(URL url) {}

    private Client createClient(URL url, EndpointManager endpointManager) {
        Client client = innerCreateClient(url);
        endpointManager.addEndpoint(client);
        return client;
    }
    protected abstract Server innerCreateServer(URL url, MessageHandler messageHandler);
    protected abstract Client innerCreateClient(URL url);

}

[图片上传失败...(image-c034a0-1563956674836)]

  1. createServer最终会返回一个new NettyServer(url, messageHandler);对象

public class NettyServer extends AbstractServer implements StatisticCallback {
    // default io thread is Runtime.getRuntime().availableProcessors() * 2
    private final static ChannelFactory channelFactory = new NioServerSocketChannelFactory(
            Executors.newCachedThreadPool(new DefaultThreadFactory("nettyServerBoss", true)),
            Executors.newCachedThreadPool(new DefaultThreadFactory("nettyServerWorker", true)));

    // 单端口需要对应单executor 1) 为了更好的隔离性 2) 为了防止被动releaseExternalResources:
    private StandardThreadExecutor standardThreadExecutor = null;
    
    protected NettyServerChannelManage channelManage = null;
    private org.jboss.netty.channel.Channel serverChannel;
    private ServerBootstrap bootstrap;
    private MessageHandler messageHandler;

    public NettyServer(URL url, MessageHandler messageHandler) {
        // TODO  调用父类构造器,初始化 encode ,decode 
        super(url);
        /** T
            public AbstractServer(URL url) {
                this.url = url;
                this.codec =
                        ExtensionLoader.getExtensionLoader(Codec.class).getExtension(
                                url.getParameter(URLParamType.codec.getName(), URLParamType.codec.getValue()));
            }
        */
        this.messageHandler = messageHandler;
    }

    @Override
    public Response request(Request request) throws TransportException {
        throw new MotanFrameworkException("NettyServer request(Request request) method unsupport: url: " + url);
    }

    @Override
    public synchronized boolean open() {
        if (isAvailable()) {
            return true;
        }

        initServerBootstrap();

        serverChannel = bootstrap.bind(new InetSocketAddress(url.getPort()));
        state = ChannelState.ALIVE;

        StatsUtil.registryStatisticCallback(this);
        LoggerUtil.info("NettyServer ServerChannel finish Open: url=" + url);

        return state.isAliveState();
    }

    private synchronized void initServerBootstrap() {
        boolean shareChannel = url.getBooleanParameter(URLParamType.shareChannel.getName(), URLParamType.shareChannel.getBooleanValue());
        final int maxContentLength = url.getIntParameter(URLParamType.maxContentLength.getName(),   URLParamType.maxContentLength.getIntValue());
        int maxServerConnection = url.getIntParameter(URLParamType.maxServerConnection.getName(),           URLParamType.maxServerConnection.getIntValue());
        int workerQueueSize = url.getIntParameter(URLParamType.workerQueueSize.getName(),URLParamType.workerQueueSize.getIntValue());
        int minWorkerThread = 0, maxWorkerThread = 0;
        // TODO  .... 略 
        // TODO  StandardThreadExecutor 的目的 ?
        standardThreadExecutor = (standardThreadExecutor != null && !standardThreadExecutor.isShutdown()) ? standardThreadExecutor
                : new StandardThreadExecutor(minWorkerThread, maxWorkerThread, workerQueueSize,
                        new DefaultThreadFactory("NettyServer-" + url.getServerPortStr(), true));
        standardThreadExecutor.prestartAllCoreThreads();
        // 连接数的管理,进行最大连接数的限制 
        channelManage = new NettyServerChannelManage(maxServerConnection);
        bootstrap = new ServerBootstrap(channelFactory);
        bootstrap.setOption("child.tcpNoDelay", true);
        bootstrap.setOption("child.keepAlive", true);

        final NettyChannelHandler handler = new NettyChannelHandler(NettyServer.this, messageHandler,
                standardThreadExecutor);

        bootstrap.setPipelineFactory(new ChannelPipelineFactory() {
            // FrameDecoder非线程安全,每个连接一个 Pipeline
            public ChannelPipeline getPipeline() {
                ChannelPipeline pipeline = Channels.pipeline();
                pipeline.addLast("channel_manage", channelManage);
                pipeline.addLast("decoder", new NettyDecoder(codec, NettyServer.this, maxContentLength));
                pipeline.addLast("encoder", new NettyEncoder(codec, NettyServer.this));
                pipeline.addLast("handler", handler);
                return pipeline;
            }
        });
    }

    @Override
    public synchronized void close() {
        close(0);
    }

    @Override
    public synchronized void close(int timeout) {
        if (state.isCloseState()) {
            return;
        }
        if (state.isUnInitState()) {
            return;
        }
        try {
            // close listen socket
            serverChannel.close();
            // close all clients's channel
            channelManage.close();
            // shutdown the threadPool
            standardThreadExecutor.shutdownNow();
            // 设置close状态
            state = ChannelState.CLOSE;
            // 取消统计回调的注册
            StatsUtil.unRegistryStatisticCallback(this);
        } catch (Exception e) {
        }
    }

    /**
     * 统计回调接口
     */
    @Override
    public String statisticCallback() {
    }

    /**
     * 是否已经绑定端口
     */
    @Override
    public boolean isBound() {
        return serverChannel != null && serverChannel.isBound();
    }
}

  1. NettyServer.open()做了什么? 创建了一个netty server ,监听 指定端口。

    • 设置 ip port
    • 在pipeline中增加channel factory
    • 增加编码,解码器 codes :后面分析
    • 注册业务handler : NettyChannelHandler

      pipeline.addLast("handler", handler);

  2. NettyChannelHandler 的逻辑是什么?

[图片上传失败...(image-cb6175-1563956674836)]

继承了Netty包中的 SimpleChannelHandler 复写父类的三个方法

  • channelConnected

  • channelDisconnected

  • messageReceived(ChannelHandlerContext ctx, MessageEvent e) 主要处理逻辑

  • 主要逻辑:

    • 判断e事件类型motan定义的 Request ==> processRequest(ctx, e);
    • 判断e事件类型motan定义的 Response ==> processResponse(ctx, e);
    • 对于 server来说 Request 主要来自于client的请求
  • processRequest 逻辑如下:

    • 每一个request会被封装成一个Runnable作为task 供线程池处理
      • threadPoolExecutor.execute(new Runnable() {
        processRequest(ChannelHandlerContext ctx, Request request, long processStartTime) {
        }
      • processRequest(ChannelHandlerContext ctx, Request request, long processStartTime)实现关键内容

        Object result = messageHandler.handle(serverChannel, request); ***
        - 获取 service imp的方法调用结果,将结果写入netty 通道。

  • messageHandler 是什么?作用

    • 对于每一个request请求,服务端需要路由找到对应的接口实现方法,然后invoke得到结果
    • messageHandler从如果来的 DefaultRpcExporter 的构造函数创建,通过encpoint factory方法传递进来的
public class NettyChannelHandler extends SimpleChannelHandler {
    private ThreadPoolExecutor threadPoolExecutor;
    private MessageHandler messageHandler;
    private Channel serverChannel;

    public NettyChannelHandler(Channel serverChannel, MessageHandler messageHandler,
            ThreadPoolExecutor threadPoolExecutor) {
        this.serverChannel = serverChannel;
        this.messageHandler = messageHandler;
        this.threadPoolExecutor = threadPoolExecutor;
    }

    @Override
    public void channelConnected(ChannelHandlerContext ctx, ChannelStateEvent e) throws Exception {
    }

    @Override
    public void channelDisconnected(ChannelHandlerContext ctx, ChannelStateEvent e) throws Exception {
    }

    @Override
    public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) throws Exception {
        Object message = e.getMessage();

        if (message instanceof Request) {
            processRequest(ctx, e);
        } else if (message instanceof Response) {
            processResponse(ctx, e);
        } else {
            throw new  Exception();
        }
    }

    /**
     *  request process: 主要来自于client的请求,需要使用threadPoolExecutor进行处理,避免service message处理比较慢导致iothread被阻塞
     */
    private void processRequest(final ChannelHandlerContext ctx, MessageEvent e) {
        final Request request = (Request) e.getMessage();
        request.setAttachment(URLParamType.host.getName(), NetUtils.getHostName(ctx.getChannel().getRemoteAddress()));
        final long processStartTime = System.currentTimeMillis();
        // 使用线程池方式处理
        try {
            threadPoolExecutor.execute(new Runnable() {
                @Override
                public void run() {
                    try{
                        RpcContext.init(request);
                        System.out.println("ctx instance ==>" +ctx.toString());
                        processRequest(ctx, request, processStartTime);
                    }finally{
                        RpcContext.destroy();
                    }
                }
            });
        } catch (RejectedExecutionException rejectException) {
        }
    }

    private void processRequest(ChannelHandlerContext ctx, Request request, long processStartTime) {
        // TODO 根据 路由判断,找到对应的rpvoder#invoke 其实最终找到的是DefautlProvider类 
        Object result = messageHandler.handle(serverChannel, request);
        DefaultResponse response = null;
        if (!(result instanceof DefaultResponse)) {
            response = new DefaultResponse(result);
        } else {
            response = (DefaultResponse) result;
        }
        response.setRequestId(request.getRequestId());
        response.setProcessTime(System.currentTimeMillis() - processStartTime);
        if (ctx.getChannel().isConnected()) {
            // TODO 回写结果到客户端
            ctx.getChannel().write(response);
        }
    }

    private void processResponse(ChannelHandlerContext ctx, MessageEvent e) {
        messageHandler.handle(serverChannel, e.getMessage());
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, ExceptionEvent e) throws Exception {
        ctx.getChannel().close();
    }
}

稍微总结一下

ServiceConfig #export())

  • -> doExport()
    • -> ConfigHandler的实现 SimpleConfigHandler#export
      • -> 构造service url
      • -> 构造s Protocol 默认实现类 DefaultProtocol
      • -> 构造s Provider 默认实现类 DefaultProvider
      • -> Protocol.exporter(provider,serviceUrl ) 构造s Exporter 默认实现类
        • -> DefaultRpcProtocol#createExporter : new DefaultRpcExporter() 实例后 init
          • -> DefaultRpcExporter构造器
            • -> 实例化 server:生成一个nettyserver对象,使用netty api构造一个bootstrap监听 指定端口
            • -> 实例化 ProviderMessageRouter 路由
          • -> DefaultRpcExporter 实例 doInit()
            • -> server.open(); 完成服务端端口监听 服务启动。

服务端如何解析Request请求

{ client send r equest } --> { network } --> { server listen port }

服务端server.open后,监听 指定端口,获取网络数据流

netty server 如何解析这个网络请求数据,并将其反序列化成 Request对象


class NettyServer{
   public void  initServerBootstrap() {
       // TODO 获取网络流后,使用 NettyDecoder对数据流进行解码
        pipeline.addLast("decoder", new NettyDecoder(codec, NettyServer.this, maxContentLength));
        pipeline.addLast("encoder", new NettyEncoder(codec, NettyServer.this));
   }
}

NettyEncoder NettyDecoder最终调用的是 DefaultRpcCodec

  1. 对于client发起的request请求
    • NettyClient : encode
    • ↓ Request ↑
      
    • NettServer : decode
  2. 对于server 发起的 response 请求
    • NettyClient : decode
    • ↓ response ↑
      
    • NettServer : encode

DefaultRpcCodec逻辑:

  1. NettyClient encode逻辑
  • writye body : interfacename,methodname ,method paramter
  • writye head : 16个字节
  • head + body (对象使用hessian序列化)
  1. NettyServer decode 逻辑
  • writye head : 16个字节
  • writye body : interfacename,methodname ,method paramter
  • body (对象使用hessian序列化)
  • server 找到了接口方法,及参数值 ,下面接着如何找到对应的provider调用
public class DefaultRpcCodec extends AbstractCodec {
    private static final short MAGIC = (short) 0xF0F0;
    private static final byte MASK = 0x07;

    @Override
    public byte[] encode(Channel channel, Object message) throws IOException {
        if (message instanceof Request) {
            return encodeRequest(channel, (Request) message);
        } else if (message instanceof Response) {
            return encodeResponse(channel, (Response) message);
        }
    }
    /**request body 数据:
     */
    private byte[] encodeRequest(Channel channel, Request request) throws IOException {
        ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
        ObjectOutput output = createOutput(outputStream);
        //  TODO 生成协议body 
        //  TODO 1 接口名
        output.writeUTF(request.getInterfaceName());
        //  TODO 1 方法名    
        output.writeUTF(request.getMethodName());
        //  TODO 3 方法参数类型
        output.writeUTF(request.getParamtersDesc());
        // TODO 4 使用hessian序列化类
        Serialization serialization =
                ExtensionLoader.getExtensionLoader(Serialization.class).getExtension(
                        channel.getUrl().getParameter(URLParamType.serialize.getName(), URLParamType.serialize.getValue()));

        if (request.getArguments() != null && request.getArguments().length > 0) {
            for (Object obj : request.getArguments()) {
                // TODO 使用hhessain序列化object
                serialize(output, obj, serialization);
            }
        }
        output.writeInt(request.getAttachments().size());
        for (Map.Entry<String, String> entry : request.getAttachments().entrySet()) {
            output.writeUTF(entry.getKey());
            output.writeUTF(entry.getValue());
        }
        output.flush();
        byte[] body = outputStream.toByteArray();
        byte flag = MotanConstants.FLAG_REQUEST;
        output.close();
        // TODO 5 将motan 协议头与body内容合并
        return encode(body, flag, request.getRequestId());
    }
    
    /**
     * 数据协议:
     * header:  16个字节 
     * 0-15 bit     :  magic
     * 16-23 bit    :  version
     * 24-31 bit    :  extend flag , 其中: 29-30 bit: event 可支持4种event,比如normal, exception等,  31 bit : 0 is request , 1 is response 
     * 32-95 bit    :  request id
     * 96-127 bit   :  body content length
     */
    private byte[] encode(byte[] body, byte flag, long requestId) throws IOException {
        byte[] header = new byte[RpcProtocolVersion.VERSION_1.getHeaderLength()];
        int offset = 0;

        // 0 - 15 bit : magic  标识这是一个motan协议
        ByteUtil.short2bytes(MAGIC, header, offset);
        offset += 2;

        // 16 - 23 bit : version  标识协议版本
        header[offset++] = RpcProtocolVersion.VERSION_1.getVersion();

        // 24 - 31 bit : extend flag  
        header[offset++] = flag;

        // 32 - 95 bit : requestId
        ByteUtil.long2bytes(requestId, header, offset);
        offset += 8;

        // 96 - 127 bit : body content length
        ByteUtil.int2bytes(body.length, header, offset);
        byte[] data = new byte[header.length + body.length];
        System.arraycopy(header, 0, data, 0, header.length);
        System.arraycopy(body, 0, data, header.length, body.length);
        return data;
    }  
     
  /**
     * decode data
     *      对于client端:主要是来自server端的response or exception
     *      对于server端: 主要是来自client端的request
     */
    @Override
    public Object decode(Channel channel, String remoteIp, byte[] data) throws IOException {
        if (data.length <= RpcProtocolVersion.VERSION_1.getHeaderLength()) {
        }
        short type = ByteUtil.bytes2short(data, 0);
        if (data[2] != RpcProtocolVersion.VERSION_1.getVersion()) { //TODO 
        }
        // TODO 解码头码数据
        int bodyLength = ByteUtil.bytes2int(data, 12);
        byte flag = data[3];
        byte dataType = (byte) (flag & MASK);
        boolean isResponse = (dataType != MotanConstants.FLAG_REQUEST);
        byte[] body = new byte[bodyLength];
        // TODO 解码body内容
        System.arraycopy(data, RpcProtocolVersion.VERSION_1.getHeaderLength(), body, 0, bodyLength);

        long requestId = ByteUtil.bytes2long(data, 4);
        Serialization serialization =
                ExtensionLoader.getExtensionLoader(Serialization.class).getExtension(
                        channel.getUrl().getParameter(URLParamType.serialize.getName(), URLParamType.serialize.getValue()));
        if (isResponse) { // response
            return decodeResponse(body, dataType, requestId, serialization);
        } else {
            return decodeRequest(body, requestId, serialization);
        }
         
    }
    

    private Object decodeRequest(byte[] body, long requestId, Serialization serialization) throws IOException, ClassNotFoundException {
        ByteArrayInputStream inputStream = new ByteArrayInputStream(body);
        ObjectInput input = createInput(inputStream);
       // TODO  body内容解码 与encode是个逆向操作
        String interfaceName = input.readUTF();
        String methodName = input.readUTF();
        String paramtersDesc = input.readUTF();

        DefaultRequest rpcRequest = new DefaultRequest();
        rpcRequest.setRequestId(requestId);
        rpcRequest.setInterfaceName(interfaceName);
        rpcRequest.setMethodName(methodName);
        rpcRequest.setParamtersDesc(paramtersDesc);
        rpcRequest.setArguments(decodeRequestParameter(input, paramtersDesc, serialization));
        rpcRequest.setAttachments(decodeRequestAttachments(input));
        input.close();
        return rpcRequest;
    }

    private Object decodeResponse(byte[] body, byte dataType, long requestId, Serialization serialization) throws IOException,
            ClassNotFoundException {
        // ......
        return response;
    }
}



于是又回到了 NettyChannelHandler处理业务请求转发

class NettyChannelHandler{
    public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) throws Exception {
            Object message = e.getMessage();
            if (message instanceof Request) {
                processRequest(ctx, e);
             }
    }
        
    private void processRequest(ChannelHandlerContext ctx, Request request, long processStartTime) {
        Object result = messageHandler.handle(serverChannel, request);
    }
}


  1. messageHandler实现方法路由?如何找到指定接口的实现类
     class DefaultRpcExporter<T> extends AbstractExporter<T> {   
       public DefaultRpcExporter (){
          requestRouter = new ProviderProtectedMessageRouter(provider);  
       }
      }
    
  1. 还记得DefaultRpcExporter类么
  • ProviderMessageRouter#handler
    • -> ProviderMessageRouter#call方法
      • -> super.lookup():找到method
    • -> method.invoke() 生成结果

[图片上传失败...(image-971f7f-1563956674836)]

3 DefaultRpcProvider#call

  • AbstractProvider#invoke

class DefaultRpcProvider {
    public Response invoke(Request request) {
       DefaultResponse response = new DefaultResponse();
       //  TODO AbstractProvider#lookup找到对应的方法
       Method method = lookup(request);
       if (method == null) {    return response;      }
       // TODO 反射回调,生成response对象,netty server 的pipeline对response进行编码,发送至客户端 
       Object value = method.invoke(proxyImpl, request.getArguments());
       response.setValue(value);
                   
       // 传递rpc版本和attachment信息方便不同rpc版本的codec使用。
       response.setRpcProtocolVersion(request.getRpcProtocolVersion());
       response.setAttachments(request.getAttachments());
       return response;
   }
}

class AbstractProvider {
    protected Method lookup(Request request) {
          String methodDesc = ReflectUtil.
              getMethodDesc(request.getMethodName(), request.getParamtersDesc());
          return methodMap.get(methodDesc);
      }
}

Client客户端启动过程

API方式

    // 设置接口及实现类
    RefererConfig<MotanDemoService> motanDemoServiceReferer = new RefererConfig<MotanDemoService>();
    motanDemoServiceReferer.setInterface(MotanDemoService.class);
    
    // 配置服务的group以及版本号
    // 配置注册中心直连调用
    // 配置ZooKeeper注册中心
    
    // 配置RPC协议
    ProtocolConfig protocol = new ProtocolConfig();
    protocol.setId("motan");
    protocol.setName("motan");
    motanDemoServiceReferer.setProtocol(protocol);
    motanDemoServiceReferer.setDirectUrl("localhost:8002");  // 注册中心直连调用需添加此配置
    // 使用服务
    MotanDemoService service = motanDemoServiceReferer.getRef();
    service.hello("motan " +i)

RefererConfig 设置接口类即可

public class RefererConfig<T> extends AbstractRefererConfig {
    private Class<T> interfaceClass;
    // 具体到方法的配置
    protected List<MethodConfig> methods;
    // 点对点直连服务提供地址
    private String directUrl;
    private AtomicBoolean initialized = new AtomicBoolean(false);

    private T ref;
    public T getRef() {
        if (ref == null) {
            initRef();
        }
        return ref;
    }

    public synchronized void initRef() {
        if (initialized.get()) {
            return;
        }
        interfaceClass = (Class) Class.forName(interfaceClass.getName(), true, Thread.currentThread().getContextClassLoader());
        checkInterfaceAndMethods(interfaceClass, methods);
        clusterSupports = new ArrayList<ClusterSupport<T>>(protocols.size());
        List<Cluster<T>> clusters = new ArrayList<Cluster<T>>(protocols.size());
        String proxy = null;

        ConfigHandler configHandler = ExtensionLoader.getExtensionLoader(ConfigHandler.class).getExtension(MotanConstants.DEFAULT_VALUE);

        List<URL> registryUrls = loadRegistryUrls();
        String localIp = getLocalHostAddress(registryUrls);
        for (ProtocolConfig protocol : protocols) {
            Map<String, String> params = new HashMap<String, String>();
            params.put(URLParamType.nodeType.getName(), MotanConstants.NODE_TYPE_REFERER);
            // TODO 生成ref url
            URL refUrl = new URL(protocol.getName(), localIp, MotanConstants.DEFAULT_INT_VALUE, interfaceClass.getName(), params);
            ClusterSupport<T> clusterSupport = createClusterSupport(refUrl, configHandler, registryUrls);
            clusterSupports.add(clusterSupport);
            clusters.add(clusterSupport.getCluster());
            
            proxy = (proxy == null) ? refUrl.getParameter(URLParamType.proxy.getName(), URLParamType.proxy.getValue()) : proxy;
        }
        // TODO 根据接口类生成代理类。完成底层 transport的传输
        ref = configHandler.refer(interfaceClass, clusters, proxy);
        initialized.set(true);
    }
}


关键点

   `ref = configHandler.refer(interfaceClass, clusters, proxy);`
  • 熟悉的 SimpleConfigHandler.java
public class SimpleConfigHandler implements ConfigHandler {

    @Override
    public <T> ClusterSupport<T> buildClusterSupport(Class<T> interfaceClass, List<URL> registryUrls) {
        ClusterSupport<T> clusterSupport = new ClusterSupport<T>(interfaceClass, registryUrls);
        clusterSupport.init();
        return clusterSupport;
    }

    @Override
    public <T> T refer(Class<T> interfaceClass, List<Cluster<T>> clusters, String proxyType) {
        // TODO 依旧是对过SPI,找到客情端代理工厂,生成代理类 
        ProxyFactory proxyFactory = ExtensionLoader.getExtensionLoader(ProxyFactory.class).getExtension(proxyType);
        return proxyFactory.getProxy(interfaceClass, new RefererInvocationHandler<T>(interfaceClass, clusters));
    }
}

代理是什么

  1. 静态代理(静态定义代理类,我们自己静态定义的代理类。比如我们自己定义一个明星的经纪人类)

         //  每一个接口类都要创建一个对应的代理类,增加了代码量。 其次一旦接口发生变化,代理也要修改。     
         public interface Star {
             void confer();//面谈
         }
         public class ProxyStar implements Star{
             private Star star;//真实对象的引用(明星)
             @Override
             public void confer() {
                 System.out.println("ProxyStar.confer() start ");
                 star.confer(); // TODO 此处才是真实对象的调用
                 System.out.println("ProxyStar.confer() end ");
             }
         }
         
         public class RealStar implements Star{
             @Override
             public void confer() {
                 System.out.println("RealStar.confer()");
             }
         }
         public static void main(String[] args) {
             Star real = new RealStar();
             Star proxy = new ProxyStar(real);
             proxy.sing();//真实对象的操作(明星唱歌)
         }
    
  2. 动态代理(通过程序动态生成代理类,该代理类不是我们自己定义的。运行期间由JVM根据反射等机制动态的生!

  3. 怎么样创建的代理 motan中使用 com.weibo.api.motan.proxy.spi.JdkProxyFactory

     public class JdkProxyFactory implements ProxyFactory {
         // 使用JDK 动态代理
         @SuppressWarnings("unchecked")
         public <T> T getProxy(Class<T> clz, InvocationHandler invocationHandler) {
             return (T) Proxy.newProxyInstance(this.getClass().getClassLoader(), new Class[] {clz}, invocationHandler);
         }
     }
    

public class RefererInvocationHandler<T> implements InvocationHandler {

    private List<Cluster<T>> clusters;
    private Class<T> clz;
    private SwitcherService switcherService = null;
    private String interfaceName;

    public RefererInvocationHandler(Class<T> clz, Cluster<T> cluster) {
        init();
    }

    public RefererInvocationHandler(Class<T> clz, List<Cluster<T>> clusters) {
        this.clz = clz;
        this.clusters = clusters;
        init();
    }

    private void init() {
        // clusters 不应该为空
        String switchName =
                this.clusters.get(0).getUrl().getParameter(URLParamType.switcherService.getName(), URLParamType.switcherService.getValue());
        switcherService = ExtensionLoader.getExtensionLoader(SwitcherService.class).getExtension(switchName);
        interfaceName = MotanFrameworkUtil.removeAsyncSuffix(clz.getName());
    }

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        if (isLocalMethod(method)) {
            throw new MotanServiceException("can not invoke local method:" + method.getName());
        }
        DefaultRequest request = new DefaultRequest();
        request.setRequestId(RequestIdGenerator.getRequestId());
        request.setArguments(args);
        String methodName = method.getName();
        boolean async = false;
        if (methodName.endsWith(MotanConstants.ASYNC_SUFFIX) && method.getReturnType().equals(ResponseFuture.class)) {
            methodName = MotanFrameworkUtil.removeAsyncSuffix(methodName);
            async = true;
        }
        RpcContext.getContext().putAttribute(MotanConstants.ASYNC_SUFFIX, async);
        //  TODO   主要是设置 request 中的方法包,参数等信息
        request.setMethodName(methodName);
        request.setParamtersDesc(ReflectUtil.getMethodParamDesc(method));
        request.setInterfaceName(interfaceName);
        request.setAttachment(URLParamType.requestIdFromClient.getName(), String.valueOf(RequestIdGenerator.getRequestIdFromClient()));

        // 当 referer配置多个protocol的时候,比如A,B,C,
        // 那么正常情况下只会使用A,如果A被开关降级,那么就会使用B,B也被降级,那么会使用C
        for (Cluster<T> cluster : clusters) {
            String protocolSwitcher = MotanConstants.PROTOCOL_SWITCHER_PREFIX + cluster.getUrl().getProtocol();
            Switcher switcher = switcherService.getSwitcher(protocolSwitcher);
            if (switcher != null && !switcher.isOn()) {
                continue;
            }
            request.setAttachment(URLParamType.version.getName(), cluster.getUrl().getVersion());
            request.setAttachment(URLParamType.clientGroup.getName(), cluster.getUrl().getGroup());
            // 带上client的application和module
            Response response = null;
            try {
                //  TODO 代码 cluster调用远程服务,本质上是将reqeust的相关信息发送到服务端
                response = cluster.call(request);
                if (async && response instanceof ResponseFuture) {
                    return response;
                } else {
                    return response.getValue();
                }
            } catch (RuntimeException e) {
            }
        }
    }
}

客户端如何选择一个服务地址呢--集群中选择

集群 cluster.call(request) ,这里cluster是什么?
  1. cluster是一系列[] interface,registryUrls] 的配置类的对象集合
    ClusterSupport<T> clusterSupport = new ClusterSupport<T>(interfaceClass, registryUrls);
    clusters.add(clusterSupport.getCluster());
  2. ClusterSupport 做了什么?
  • ClusterSupport 提供了对Cluster的一些辅助工作,cluster生成,参数初始化
class ClusterSupport {

    private void prepareCluster() {
           String clusterName = url.getParameter(URLParamType.cluster.getName(), URLParamType.cluster.getValue());
           String loadbalanceName = url.getParameter(URLParamType.loadbalance.getName(), URLParamType.loadbalance.getValue());
           String haStrategyName = url.getParameter(URLParamType.haStrategy.getName(), URLParamType.haStrategy.getValue());
           // TODO 通过SPI获取默认的  负载均衡实现  ,高可用策略实现 
           System.out.println(lusterName); // default
           System.out.println(" " +loadbalanceName); // activeWeight
           System.out.println(" " +haStrategyName);  // failover
           
           cluster = ExtensionLoader.getExtensionLoader(Cluster.class).getExtension(clusterName);
           LoadBalance<T> loadBalance = ExtensionLoader.getExtensionLoader(LoadBalance.class).getExtension(loadbalanceName);
           HaStrategy<T> ha = ExtensionLoader.getExtensionLoader(HaStrategy.class).getExtension(haStrategyName);
           cluster.setLoadBalance(loadBalance);
           cluster.setHaStrategy(ha);
           cluster.setUrl(url);
       }
}
  • cluster的 HaStrategy 与 LoadBalance

    • 负载均衡的实现 com.weibo.api.motan.cluster.loadbalance.ActiveWeightLoadBalance
    • 高可用策略实现 com.weibo.api.motan.cluster.ha.FailoverHaStrategy

    [图片上传失败...(image-b7a9d7-1563956674836)]

    [图片上传失败...(image-39b478-1563956674836)]

集群 cluster ,是如何将ha 策略与loadbalance结合起来的呢
什么是HA,高可用
  1. 在集群服务器架构中,当主服务器故障时,备份服务器能够自动接管主服务器的工作,
    并及时切换过去,以实现对用户的不间断服务。这里我感觉它跟故障转移(failover)是一个意思,
  2. 简单的理解:当调用制作时,是直接返回结果,还是重试一定的次数来得到最终结果
  3. motan支持的HaStrategy只有两种:FailfastHatrategy和FailoverStrategy
什么是 LoadBalance
  1. 如何处理高并发带来的系统性能问题,最终大家都会使用负载均衡机制。
    它是根据某种负载策略把请求分发到集群中的每一台服务器上,让整个服务器群来处理网站的请求。
  2. 简单的理解就是如何选择一个合理的请求服务端地址
HA与 LoadBalance的关系
  1. ha向 lB发出请求,lb通过一定机制找到负载全蓝的服务端地址
  2. ha向这个服务端url发出网络请求,根据ha的策略。对请求的次数,时间待作出一定的处理
    直接返回或者重试N次请求
HA源代码分析
public class FailfastHaStrategy<T> extends AbstractHaStrategy<T> {
    @Override
    public Response call(Request request, LoadBalance<T> loadBalance) {
        Referer<T> refer = loadBalance.select(request);
        // 找到一个refer call 
        return refer.call(request);
    }
}
// TODO 最多尝试N次
public class FailoverHaStrategy<T> extends AbstractHaStrategy<T> {

    protected ThreadLocal<List<Referer<T>>> referersHolder = new ThreadLocal<List<Referer<T>>>() {
        @Override
        protected java.util.List<com.weibo.api.motan.rpc.Referer<T>> initialValue() {
            return new ArrayList<Referer<T>>();
        }
    };

    @Override
    public Response call(Request request, LoadBalance<T> loadBalance) {
        List<Referer<T>> referers = selectReferers(request, loadBalance);
        if (referers.isEmpty()) {
            throw new MotanServiceException("xx");
        }
        URL refUrl = referers.get(0).getUrl();
        // 先使用method的配置
        int tryCount =   refUrl.getMethodParameter(request.getMethodName(), request.getParamtersDesc(), URLParamType.retries.getName(),    URLParamType.retries.getIntValue());
        // 如果有问题,则设置为不重试
        if (tryCount < 0) {
            tryCount = 0;
        }

        for (int i = 0; i <= tryCount; i++) {
            Referer<T> refer = referers.get(i % referers.size());
            try {
                request.setRetries(i);
                return refer.call(request);
            } catch (RuntimeException e) {
            }
        }
    }

    protected List<Referer<T>> selectReferers(Request request, LoadBalance<T> loadBalance) {
        List<Referer<T>> referers = referersHolder.get();
        referers.clear();
        loadBalance.selectToHolder(request, referers);
        return referers;
    }

}

LoadBalance 的实现分析
  1. ActivieWegithLoadBalance 最少并发优先,怎么计算并发量?很简单,通过原子计数器即可

@SpiMeta(name = "activeWeight")
public class ActiveWeightLoadBalance<T> extends AbstractLoadBalance<T> {

    @Override
    protected Referer<T> doSelect(Request request) {
        List<Referer<T>> referers = getReferers();
        int refererSize = referers.size();
        int startIndex = ThreadLocalRandom.current().nextInt(refererSize);
        int currentCursor = 0;
        int currentAvailableCursor = 0;
        Referer<T> referer = null;

        while (currentAvailableCursor < MAX_REFERER_COUNT && currentCursor < refererSize) {
            Referer<T> temp = referers.get((startIndex + currentCursor) % refererSize);
            currentCursor++;

            if (!temp.isAvailable()) {
                continue;
            }

            currentAvailableCursor++;

            if (referer == null) {
                referer = temp;
            } else {
                if (compare(referer, temp) > 0) {
                    referer = temp;
                }
            }
        }

        return referer;
    }

    @Override
    protected void doSelectToHolder(Request request, List<Referer<T>> refersHolder) {
        List<Referer<T>> referers = getReferers();

        int refererSize = referers.size();
        int startIndex = ThreadLocalRandom.current().nextInt(refererSize);
        int currentCursor = 0;
        int currentAvailableCursor = 0;

        while (currentAvailableCursor < MAX_REFERER_COUNT && currentCursor < refererSize) {
            Referer<T> temp = referers.get((startIndex + currentCursor) % refererSize);
            currentCursor++;

            if (!temp.isAvailable()) {
                continue;
            }

            currentAvailableCursor++;

            refersHolder.add(temp);
        }

        Collections.sort(refersHolder, new LowActivePriorityComparator<T>());
    }

    private int compare(Referer<T> referer1, Referer<T> referer2) {
        return referer1.activeRefererCount() - referer2.activeRefererCount();
    }

    static class LowActivePriorityComparator<T> implements Comparator<Referer<T>> {
        @Override
        public int compare(Referer<T> referer1, Referer<T> referer2) {
            return referer1.activeRefererCount() - referer2.activeRefererCount();
        }
    }
}
  1. 问题来了,这里只看到 loadbalance#doSelect并没有看到,哪里将referer的权重变化
    • 生成一个随机数,然后使用取余来获取一个referer ,然后将referer进行sort排序
    • LowActivePriorityComparator 中比较的是 activeRefererCount()的值,那么这个值在lb中工没有看到变更
    • 由此怀疑,只能是在referer.call的时候,作的改变
Refer如何控制使用次数
  1. DefaultRpcReferer

public abstract class AbstractReferer<T> extends AbstractNode implements Referer<T> {

    protected Class<T> clz;
    // todo 嘿嘿就在这里 
    protected AtomicInteger activeRefererCount = new AtomicInteger(0);
    protected URL serviceUrl;

    @Override
    public Response call(Request request) {
        if (!isAvailable()) {
        }
        // TODO  ha.call(request,loadbalance)
        // TODO 一旦被调用,则活动数量add ,实现则在具体类
        // TODO 默认策略会将计数器 +1
        incrActiveCount(request);
        Response response = null;
        try {
            response = doCall(request);
            return response;
        } finally {
            // TODO 释放activecount ,默认策略会将计数器-1
            decrActiveCount(request, response);
        }
    }

    @Override
    public int activeRefererCount() {
        return activeRefererCount.get();
    }

    protected void incrActiveCount(Request request) {
        activeRefererCount.incrementAndGet();
    }

    protected void decrActiveCount(Request request, Response response) {
        activeRefererCount.decrementAndGet();
    }

    protected abstract Response doCall(Request request);

    @Override
    public String desc() {
        return "[" + this.getClass().getSimpleName() + "] url=" + url;
    }

    @Override
    public URL getServiceUrl() {
        return serviceUrl;
    }

}

class DefaultRpcReferer<T> extends AbstractReferer<T> {
        private Client client;
        private EndpointFactory endpointFactory;

        public DefaultRpcReferer(Class<T> clz, URL url, URL serviceUrl) {
            super(clz, url, serviceUrl);
            endpointFactory =
                    ExtensionLoader.getExtensionLoader(EndpointFactory.class).getExtension(
                            url.getParameter(URLParamType.endpointFactory.getName(), URLParamType.endpointFactory.getValue()));
            client = endpointFactory.createClient(url);
        }

        @Override
        protected Response doCall(Request request) {
            // 为了能够实现跨group请求,需要使用server端的group。
            request.setAttachment(URLParamType.group.getName(), serviceUrl.getGroup());
            return client.request(request);
        }

        @Override
        protected void decrActiveCount(Request request, Response response) {
            if (response == null || !(response instanceof Future)) {
                activeRefererCount.decrementAndGet();
                return;
            }
            Future future = (Future) response;
            future.addListener(new FutureListener() {
                @Override
                public void operationComplete(Future future) throws Exception {
                    activeRefererCount.decrementAndGet();
                }
            });
        }

        @Override
        protected boolean doInit() {
            boolean result = client.open();
            return result;
        }
    }

协议位 意义
0-15 bit magic
16-23 bit version
24-28 bit extend flag
29-30 bit event 比如normal, exception等
31 bit 0 is request , 1 is response
32-95 bit request id
96-127 bit body content length

SPI 机制

类加载机制 :双亲选手

public class Test {
    public static void main(String[] args){
         ClassLoader loader = TestClassLoader.class.getClassLoader();  
        //获得加载ClassLoaderTest.class这个类的类加载器
        while(loader != null) {
            System.out.println(loader);
            loader = loader.getParent();    //获得父类加载器的引用
        }
        System.out.println(loader);
        // output >>>
        // sun.misc.Launcher$AppClassLoader@7b7035c6
        // sun.misc.Launcher$ExtClassLoader@3da997a
        // null
    }
  }
    
  
  1. 那么问题来了JDBC人驱动器接口定义是在java核心包中定义,而具体驱动实现是在mysql,oracle的第三方jar中提供
    是怎么做到的很。很明显这接口的与实现类是由不同的类加载器实现的,这似乎违背了前面所
    提到的parent委托机器。

2.为了解决这个矛盾 Java提供了一线程上下文的 thread classloader
Thread.currentThread().getContextClassLoader().

什么是SIP

SPI : Service Provider Interface

java spi就是提供这样的一个机制:为某个接口寻找服务实现的机制。有点类似IOC的思想,
就是将装配的控制权移到程序之外,在模块化设计中这个机制尤其重要。

SPI 规范

  1. java spi的具体约定为:当服务的提供者,

  2. 提供了服务接口的一种实现之后,

  3. 在jar包的META-INF/services/目录里同时创建一个以服务接口命名的文件。

  4. 该文件里就是实现该服务接口的具体实现类。而当外部程序装配这个模块的时候,

  5. 能通过该jar包META-INF/services/里的配置文件找到具体的实现类名,并装载实例化,完成模块的注入。

  6. 基于这样一个约定就能很好的找到服务接口的实现类,而不需要再代码里制定。
    jdk提供服务实现查找的一个工具类:java.util.ServiceLoader

    package my.xyz.spi;
    import java.util.List;
    public interface Search {
    public List serch(String keyword);
    }

    package com.xyz.factory;
    import java.util.Iterator;
    import java.util.ServiceLoader;
    import my.xyz.spi.Search;
    public class SearchFactory {
    private SearchFactory() {
    }
    public static Search newSearch() {
    Search search = null;
    ServiceLoader<Search> serviceLoader = ServiceLoader.load(Search.class);
    Iterator<Search> searchs = serviceLoader.iterator();
    if (searchs.hasNext()) {
    search = searchs.next();
    }
    return search;
    }
    }

 package my.xyz.test;  
 import java.util.Iterator;  
 import java.util.ServiceLoader;  
 import com.xyz.factory.SearchFactory;  
 import my.xyz.spi.Search;  
 public class SearchTest {  
     public static void main(String[] args) {  
         Search search = SearchFactory.newSearch();  
         search.serch("java spi test");  
     }  
 }  

http://www.mamicode.com/info-detail-1328167.html

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

推荐阅读更多精彩内容