远程协助
当前位置:重庆拓磊计算机运营维护中心 > 文档中心 > 服务器 >
标题:golang实现简易的分布式系统方法    日期:2018-12-25

本文介绍了golang实现简易的分布式系统方法,分享给大家,具体如下:

功能

  • 能够发送/接收请求和响应
  • 能够连接到集群
  • 如果无法连接到群集(如果它是第一个节点),则可以作为主节点启动节点
  • 每个节点有唯一的标识
  • 能够在节点之间交换json数据包
  • 接受命令行参数中的所有信息(将来在我们系统升级时将会很有用)
  • 源码

    package main
    
    import (
      "fmt"
      "strconv"
      "time"
      "math/rand"
      "net"
      "flag"
      "strings"
      "encoding/json"
    )
    
    // 节点数据信息
    type NodeInfo struct {
    
      // 节点ID,通过随机数生成
      NodeId int `json:"nodeId"`
      // 节点IP地址
      NodeIpAddr string `json:"nodeIpAddr"`
      // 节点端口
      Port string `json: "port"`
    }
    
    // 将节点数据信息格式化输出
    //NodeInfo:{nodeId: 89423,nodeIpAddr: 127.0.0.1/8,port: 8001}
    func (node *NodeInfo) String() string {
    
      return "NodeInfo:{ nodeId:" + strconv.Itoa(node.NodeId) + ",nodeIpAddr:" + node.NodeIpAddr + ",port:" + node.Port + "}"
    }
    
    /* 添加一个节点到集群的一个请求或者响应的标准格式 */
    type AddToClusterMessage struct {
      // 源节点
      Source NodeInfo `json:"source"`
      // 目的节点
      Dest NodeInfo `json:"dest"`
      // 两个节点连接时发送的消息
      Message string `json:"message"`
    }
    
    /* Request/Response 信息格式化输出 */
    func (req AddToClusterMessage) String() string {
      return "AddToClusterMessage:{\n source:" + req.Source.String() + ",\n dest: " + req.Dest.String() + ",\n message:" + req.Message + " }"
    }
    
    // cat vi go
    // rm
    
    func main() {
    
      // 解析命令行参数
      makeMasterOnError := flag.Bool("makeMasterOnError", false, "如果IP地址没有连接到集群中,重庆网络维护,我们将其作为Master节点.")
      clusterip := flag.String("clusterip", "127.0.0.1:8001", "任何的节点连接都连接这个IP")
      myport := flag.String("myport", "8001", "ip address to run this node on. default is 8001.")
      flag.Parse() //解析
    
      fmt.Println(*makeMasterOnError)
      fmt.Println(*clusterip)
      fmt.Println(*myport)
    
      /* 为节点生成ID */
      rand.Seed(time.Now().UTC().UnixNano()) //种子
      myid := rand.Intn(99999999) // 随机
    
      //fmt.Println(myid)
    
      // 获取IP地址
      myIp,_ := net.InterfaceAddrs()
      fmt.Println(myIp[0])
    
      // 创建NodeInfo结构体对象
      me := NodeInfo{NodeId: myid, NodeIpAddr: myIp[0].String(), Port: *myport}
      // 输出结构体数据信息
      fmt.Println(me.String())
      dest := NodeInfo{ NodeId: -1, NodeIpAddr: strings.Split(*clusterip, ":")[0], Port: strings.Split(*clusterip, ":")[1]}
    
      /* 尝试连接到集群,在已连接的情况下并且向集群发送请求 */
      ableToConnect := connectToCluster(me, dest)
    
      /*
       * 监听其他节点将要加入到集群的请求
       */
      if ableToConnect || (!ableToConnect && *makeMasterOnError) {
        if *makeMasterOnError {fmt.Println("Will start this node as master.")}
        listenOnPort(me)
      } else {
        fmt.Println("Quitting system. Set makeMasterOnError flag to make the node master.", myid)
      }
    
    }
    
    /*
     * 这是发送请求时格式化json包有用的工具
     * 这是非常重要的,如果不经过数据格式化,你最终发送的将是空白消息
     */
    func getAddToClusterMessage(source NodeInfo, dest NodeInfo, message string) (AddToClusterMessage){
      return AddToClusterMessage{
        Source: NodeInfo{
          NodeId: source.NodeId,
          NodeIpAddr: source.NodeIpAddr,
          Port: source.Port,
        },
        Dest: NodeInfo{
          NodeId: dest.NodeId,
          NodeIpAddr: dest.NodeIpAddr,
          Port: dest.Port,
        },
        Message: message,
      }
    }
    
    func connectToCluster(me NodeInfo, dest NodeInfo) (bool){
      /* 连接到socket的相关细节信息 */
      connOut, err := net.DialTimeout("tcp", dest.NodeIpAddr + ":" + dest.Port, time.Duration(10) * time.Second)
      if err != nil {
        if _, ok := err.(net.Error); ok {
          fmt.Println("未连接到集群.", me.NodeId)
          return false
        }
      } else {
        fmt.Println("连接到集群. 发送消息到节点.")
        text := "Hi nody.. 请添加我到集群.."
        requestMessage := getAddToClusterMessage(me, dest, text)
        json.NewEncoder(connOut).Encode(&requestMessage)
    
        decoder := json.NewDecoder(connOut)
        var responseMessage AddToClusterMessage
        decoder.Decode(&responseMessage)
        fmt.Println("得到数据响应:\n" + responseMessage.String())
    
        return true
      }
      return false
    }
    
    func listenOnPort(me NodeInfo){
      /* 监听即将到来的消息 */
      ln, _ := net.Listen("tcp", fmt.Sprint(":" + me.Port))
      /* 接受连接 */
      for {
        connIn, err := ln.Accept()
        if err != nil {
          if _, ok := err.(net.Error); ok {
            fmt.Println("Error received while listening.", me.NodeId)
          }
        } else {
          var requestMessage AddToClusterMessage
          json.NewDecoder(connIn).Decode(&requestMessage)
          fmt.Println("Got request:\n" + requestMessage.String())
    
          text := "Sure buddy.. too easy.."
          responseMessage := getAddToClusterMessage(me, requestMessage.Source, text)
          json.NewEncoder(connIn).Encode(&responseMessage)
          connIn.Close()
        }
      }
    }
    下一篇:详解nginx代理socket.io服务踩坑
    
    联系我们
    • 客服热线:023-63522929(7 x 24h)
    • 在线客服:
    • 微信公众号 官方微博