golang实现简易的分布式系统方法
来源:脚本之家
时间:2022-12-31 13:53:41 335浏览 收藏
IT行业相对于一般传统行业,发展更新速度更快,一旦停止了学习,很快就会被行业所淘汰。所以我们需要踏踏实实的不断学习,精进自己的技术,尤其是初学者。今天golang学习网给大家整理了《golang实现简易的分布式系统方法》,聊聊分布式,我们一起来看看吧!
本文介绍了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实现简易的分布式系统方法》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!
声明:本文转载于:脚本之家 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
-
220 收藏
-
117 收藏
-
384 收藏
-
113 收藏
-
269 收藏
最新阅读
更多>
-
193 收藏
-
271 收藏
-
414 收藏
-
179 收藏
-
194 收藏
-
441 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习
评论列表
-
- 有魅力的砖头
- 这篇博文太及时了,太细致了,很棒,码住,关注作者大大了!希望作者大大能多写Golang相关的文章。
- 2023-02-28 06:25:42
-
- 英勇的机器猫
- 很详细,mark,感谢楼主的这篇技术贴,我会继续支持!
- 2023-02-19 04:00:28
-
- 体贴的白开水
- 感谢大佬分享,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢师傅分享文章内容!
- 2023-02-04 08:55:42
-
- 苗条的汽车
- 这篇技术贴太及时了,很详细,赞 👍👍,已加入收藏夹了,关注楼主了!希望楼主能多写Golang相关的文章。
- 2023-02-04 02:53:25
-
- 俊逸的凉面
- 这篇文章真及时,up主加油!
- 2023-01-10 14:43:10
-
- 彩色的枕头
- 很有用,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢老哥分享技术贴!
- 2023-01-03 09:43:06
-
- 单身的故事
- 赞 👍👍,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢博主分享技术文章!
- 2023-01-02 08:20:38
-
- 饱满的战斗机
- 这篇技术贴真及时,太全面了,真优秀,码起来,关注作者大大了!希望作者大大能多写Golang相关的文章。
- 2023-01-01 23:37:42
-
- 强健的芹菜
- 这篇技术文章真是及时雨啊,太全面了,很棒,码起来,关注作者大大了!希望作者大大能多写Golang相关的文章。
- 2022-12-31 21:25:38