中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

golang實現簡易的分布式系統方法

發布時間:2020-10-24 23:05:47 來源:腳本之家 閱讀:201 作者:小馬哥_Magical 欄目:編程語言

本文介紹了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()
    }
  }
}

運行程序

/Users/liyuechun/go
liyuechun:go yuechunli$ go install main
liyuechun:go yuechunli$ main
My details: NodeInfo:{ nodeId:53163002, nodeIpAddr:127.0.0.1/8, port:8001 }
不能連接到集群. 53163002
Quitting system. Set makeMasterOnError flag to make the node master. 53163002
liyuechun:go yuechunli$

獲取相關幫助信息

$ ./bin/main -h
liyuechun:go yuechunli$ ./bin/main -h
Usage of ./bin/main:
 -clusterip string
    ip address of any node to connnect (default "127.0.0.1:8001")
 -makeMasterOnError
    make this node master if unable to connect to the cluster ip provided.
 -myport string
    ip address to run this node on. default is 8001. (default "8001")
liyuechun:go yuechunli$

啟動Node1主節點

$ ./bin/main --makeMasterOnError
liyuechun:go yuechunli$ ./bin/main --makeMasterOnError
My details: NodeInfo:{ nodeId:82381143, nodeIpAddr:127.0.0.1/8, port:8001 }
未連接到集群. 82381143
Will start this node as master.

添加節點Node2到集群

$ ./bin/main --myport 8002 --clusterip 127.0.0.1:8001

添加節點Node3到集群

main --myport 8004 --clusterip 127.0.0.1:8001

添加節點Node4到集群

$ main --myport 8003 --clusterip 127.0.0.1:8002

golang實現簡易的分布式系統方法

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

华池县| 金寨县| 景泰县| 盈江县| 长岭县| 仁布县| 长顺县| 丰都县| 定陶县| 玉环县| 卢龙县| 永平县| 邢台市| 龙里县| 安丘市| 新河县| 钦州市| 新乐市| 嘉祥县| 什邡市| 赤城县| 高密市| 隆德县| 大同市| 阳曲县| 磐安县| 巩义市| 渝北区| 北碚区| 旬阳县| 武功县| 罗江县| 长沙县| 舞钢市| 京山县| 嘉禾县| 芜湖县| 鹿泉市| 宾川县| 武陟县| 澎湖县|