IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 网络协议 -> DIY一个属于自己的微服务框架 -> 正文阅读

[网络协议]DIY一个属于自己的微服务框架

从Vertx提供的通信方式开始

第一个集群节点启动

package org.example;

import com.hazelcast.config.Config;
import com.hazelcast.config.FileSystemXmlConfig;
import io.vertx.core.Vertx;
import io.vertx.core.VertxOptions;
import io.vertx.core.eventbus.EventBusOptions;
import io.vertx.spi.cluster.hazelcast.HazelcastClusterManager;

import java.net.InetAddress;
import java.net.UnknownHostException;

public class AppCluster {

    public static void main(String[] args) throws  UnknownHostException {
        final VertxOptions vertxOptions = new VertxOptions();
        EventBusOptions eventBusOptions = new EventBusOptions();
        // 本机局域网Ip
        String hostAddress = InetAddress.getLocalHost().getHostAddress();
        vertxOptions.setEventBusOptions(eventBusOptions).getEventBusOptions().setHost(hostAddress);
        //集群方式启动
        HazelcastClusterManager clusterManager = new HazelcastClusterManager();

        vertxOptions.setClusterManager(clusterManager);

        Vertx.clusteredVertx(vertxOptions, res -> {
            Vertx result = res.result();
            result.deployVerticle(new MainClusterVerticle(), r -> {
                if (r.succeeded()) {
                    System.out.println(MainClusterVerticle.class.getName() + " --> 部署成功");
                } else {
                    r.cause().printStackTrace();
                    System.err.println(MainClusterVerticle.class.getName() + " --> 部署失败, " + r.cause().getMessage());
                }
            });
        });
    }
}

发布verticle服务

package org.example;

import io.vertx.core.AbstractVerticle;
import io.vertx.core.json.JsonObject;

public class MainClusterVerticle extends AbstractVerticle {

  public void start() {
    System.out.println("start thread" + Thread.currentThread().getName());
    //发布eventbus服务
    vertx.eventBus().consumer("com.xiaoniu.bus", msg -> {
      System.out.println("read thread" + Thread.currentThread().getName());
      System.out.println("收到消息");
      System.out.println(msg != null ? ((JsonObject) msg.body()).encodePrettily() : "没有消息");
      JsonObject j = new JsonObject();
      j.put("info", "我是Main");
      msg.reply(j);
    });
  }
}

第二个集群节点启动

package org.example;

import com.hazelcast.config.Config;
import com.hazelcast.config.FileSystemXmlConfig;
import io.vertx.core.Vertx;
import io.vertx.core.VertxOptions;
import io.vertx.core.eventbus.EventBusOptions;
import io.vertx.spi.cluster.hazelcast.HazelcastClusterManager;

import java.io.FileNotFoundException;
import java.net.InetAddress;
import java.net.UnknownHostException;

/**
 * @Author: Administrator
 * @Description:
 */
public class AppCluster2 {

    public static void main(String[] args) throws  UnknownHostException {
        final VertxOptions vertxOptions = new VertxOptions();
        EventBusOptions eventBusOptions = new EventBusOptions();
        // 本机Ip
        String hostAddress = InetAddress.getLocalHost().getHostAddress();
        vertxOptions.setEventBusOptions(eventBusOptions).getEventBusOptions().setHost(hostAddress);

        HazelcastClusterManager clusterManager = new HazelcastClusterManager();

        vertxOptions.setClusterManager(clusterManager);
        Vertx.clusteredVertx(vertxOptions, res -> {
            Vertx result = res.result();
            result.deployVerticle(new MainClusterVerticle2(), r -> {
                if (r.succeeded()) {
                    System.out.println(MainClusterVerticle2.class.getName() + " --> 部署成功");
                } else {
                    r.cause().printStackTrace();
                    System.err.println(MainClusterVerticle2.class.getName() + " --> 部署失败, " + r.cause().getMessage());
                }
            });
        });
    }
}

发布verticle服务

package org.example;

import io.vertx.core.AbstractVerticle;
import io.vertx.core.http.HttpServer;
import io.vertx.core.json.JsonObject;
import io.vertx.ext.web.Router;

public class MainClusterVerticle2 extends AbstractVerticle {
  public void start() {
    System.out.println("start thread" + Thread.currentThread().getName());
    //发布web服务

    // 创建HttpServer
    HttpServer server = vertx.createHttpServer();
    // 创建路由对象
    Router router = Router.router(vertx);
    // 监听/index地址
    router.route("/index").handler(request -> {
      System.out.println("haha" + Thread.currentThread().getName());
      System.out.println("时间到了,发送消息");
      JsonObject json = new JsonObject().put("info", "我是另主");
      //通过eventbus发送请求
      vertx.eventBus().request("com.xiaoniu.bus", json, msg -> {
        System.out.println("read thread" + Thread.currentThread().getName());
        if (msg.succeeded()) {
          System.out.println(msg.result() != null
                  ? ((JsonObject)msg.result().body()).encodePrettily()
                  : "没有信息");
        } else {
          System.err.println(msg.cause().getMessage());
          msg.cause().printStackTrace();
        }
      });
      request.response().end("INDEX SUCCESS");
    });

    // 把请求交给路由处理--------------------(1)
    server.requestHandler(router);
    server.listen(7777);


  }
}

上面提供了两种通信方式,一种是通过HttpServer对外提供http服务,一种式集群间的基于EventBus的通信,后面我将基于这两种通信方式来构造一个小型的微服务框架。

集群启动及原理分析

在这里插入图片描述

  • 该进程三个重要的TCP端口。
  1. 58331端口用于集群中别的vertx服务通过eventbus向它发数据。
  2. 5700端口,实验中观察发现集群中启动n个服务,每个服务都会与其他n-1个服务建立连接,并且通过心跳保持连接,这样就可以在一个服务挂掉以后,别的服务会有断开连接提示进而知道集群中某节点挂掉。
  • 该进程一个重要UDP端口,组播端口,用于互相告知连接端口,如果上面的5700,进而建立连接形成集群。

第一个vertx服务启动界面
在这里插入图片描述

第二个vertx服务启动界面
在这里插入图片描述

第三个vertx服务启动界面
在这里插入图片描述
可以看出每个服务都保存一份集群节点

从第一个进程(4980)的套接字看
在这里插入图片描述
可以看出第一个进程通过它的集群端口(5700端口)和其他两个集群中的服务保持了TCP连接从而建立集群

从第二个进程的套接字看(3376)
在这里插入图片描述
可以看出第二个进程通过50674端口和第一个进程集群端口(5700端口)保持TCP连接建立集群,通过50787端口连接第三个进程的集群端口(5702端口)保持TCP连接建立集群

看图可能会直观一点

在这里插入图片描述

效果展示

定义controller对外提供服务

package com.koala.controller;

import java.util.Date;

import com.koala.common.dto.User;
import com.koala.common.exception.DemoException;
import com.koala.core.annotion.*;
import com.koala.service.DemoService;

@BusController
@BusMapping("demo")
public class DemoController {

    @Autowired
    private DemoService demoService;

    @BusMapping(value = "resource",method = RequestMethod.GET)
    public User test1(@RequestParam("user") User msg,@RequestParam("num") long num){
        demoService.serviceMethod();
        //获取传参
        System.out.println(msg.getA());
        //返回结果
        User user = new User();
        user.setA("adf32");
        user.setB(2);
        user.setDate(new Date());
        return user;
    }

    @BusMapping(value = "resource1",method = RequestMethod.GET)
    public User test2(@RequestParam("user") User msg,@RequestParam("num") long num) throws DemoException {
        throw new DemoException("您报错了");
    }

}

在这里插入图片描述

在这里插入图片描述
支持Bean配置类:

package com.koala.config;

import com.koala.core.annotion.Value;
import com.koala.core.annotion.VertxBean;
import com.koala.core.annotion.VertxConfiguration;
import com.koala.pojo.A;
import com.koala.pojo.B;

/**
 * @Author: chendi
 * @Description: @VertxConfiguration下的@VertxBean可以创建一些基础公共组件,在@VertxComponent下可以使用@Autowired注入
 */
@VertxConfiguration
public class TestConfiguration {

    //配置公共组件需要一些配置参数时,目前先在这儿写死,后续会支持@VertxValue("${my.value}")从application.properties读取
    @Value("${my.value}")
    private String value;

    @VertxBean
    public B getB() {
        B b = new B();
        b.setA(getA());
        return b;
    }

    @VertxBean
    public A getA() {
        A a = new A();
        a.setValue(value);
        return a;
    }
}

支持IOC及服务启动时执行组件自定义初始化方法

package com.koala.service;

import com.koala.core.annotion.Autowired;
import com.koala.core.annotion.Value;
import com.koala.core.annotion.VertxComponent;
import com.koala.core.ioc.VertxInitializer;
import com.koala.core.ioc.aware.VertxAware;
import com.koala.core.log.LogUtil;
import com.koala.pojo.A;
import com.koala.pojo.B;

import io.vertx.core.Vertx;

/**
 * @Author: chen_di
 * @Description: @VertxComponent一般用于业务层组件或者数据层组件,可以使用@Autowired注入公共组件或者别的组件或者远程调用组件
 */
@VertxComponent
//实现VertxInitializer接口,init方法中的业务代码会在服务启动时执行
package com.koala.service;

import com.koala.core.annotion.Autowired;
import com.koala.core.annotion.Value;
import com.koala.core.annotion.VertxComponent;
import com.koala.core.ioc.VertxInitializer;
import com.koala.core.ioc.aware.VertxAware;
import com.koala.core.log.LogUtil;
import com.koala.pojo.A;
import com.koala.pojo.B;

import io.vertx.core.Vertx;

/**
 * @Author: chen_di
 * @Description: @VertxComponent一般用于业务层组件或者数据层组件,可以使用@Autowired注入公共组件或者别的组件或者远程调用组件
 */
@VertxComponent
//实现VertxInitializer接口,init方法中的业务代码会在服务启动时执行
public class DemoService implements VertxInitializer, VertxAware {

    @Value("${my.url}")
    private String url;

    @Autowired
    private A a;

    @Autowired
    private B b;

    private Vertx vertx;

    public void serviceMethod(){
        System.out.println(a);
        System.out.println(b);
    }


    @Override
    public void init() {

        System.out.println(a);
        System.out.println(b.getA());

        System.out.println(a.getValue());
        System.out.println(url);
    }

    @Override
    public void setVertx(Vertx vertx) {
        this.vertx = vertx;
    }


}

支持接口配置式的远程调用并可注入

package com.koala.client;

import com.koala.common.dto.User;
import com.koala.common.exception.DemoException;
import com.koala.core.annotion.BusMapping;
import com.koala.core.annotion.RequestMethod;
import com.koala.core.annotion.RequestParam;
import com.koala.core.annotion.VertxClient;

/**
 * @Author: chen_di
 * @Description: @VertxClient标识远程调用组件
 */
@VertxClient
@BusMapping("koala.demo")//koala为第一个应用的名字
public interface MyClient {

    @BusMapping(value = "resource", method = RequestMethod.GET)
    public User test1(@RequestParam("user") User msg, @RequestParam("num") long num);

    @BusMapping(value = "resource1", method = RequestMethod.GET)
    public User test2(@RequestParam("user") User msg, @RequestParam("num") long num) throws DemoException;

}
package com.koala.controller;


import com.koala.client.MyClient;
import com.koala.common.dto.User;
import com.koala.common.exception.DemoException;
import com.koala.core.annotion.*;

@BusController
@BusMapping("demo")
public class DemoController {

    @Autowired
    private MyClient myClient;

    @BusMapping(value = "resource",method = RequestMethod.GET)
    public User test1(@RequestParam("user") User msg,@RequestParam("num") long num) {
        try {
            //远程调用得到结果
            System.out.println(myClient.test2(msg,num));
        }catch (DemoException e){
            e.printStackTrace();
        }
        return new User();

    }

    @BusMapping(value = "resource1",method = RequestMethod.GET)
    public User test2(@RequestParam("user") User msg,@RequestParam("num") long num) {
        try {
        //远程调用捕获异常
            myClient.test2(msg,num);
        }catch (DemoException e){
            e.printStackTrace();
        }
        return new User();

    }

}

实现概览

在这里插入图片描述

源码解析

。。。

适用场景

服务之间传输轻量,单Reactor多线程模型,使用于IO密集型应用。

  网络协议 最新文章
使用Easyswoole 搭建简单的Websoket服务
常见的数据通信方式有哪些?
Openssl 1024bit RSA算法---公私钥获取和处
HTTPS协议的密钥交换流程
《小白WEB安全入门》03. 漏洞篇
HttpRunner4.x 安装与使用
2021-07-04
手写RPC学习笔记
K8S高可用版本部署
mySQL计算IP地址范围
上一篇文章      下一篇文章      查看所有文章
加:2021-08-18 13:03:32  更:2021-08-18 13:05:20 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 20:19:59-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码