源码已经上传,有需要可在资源下载。在分布式day01的基础上,本节的新增内容主要是注册服务并独立运行, data:image/s3,"s3://crabby-images/6da95/6da951c3d49462ed24ce59ea2e9586a8899741fd" alt="在这里插入图片描述" 首先在distributed文件夹下创建registry文件夹并在其下分别创建client.go,registration.go和server.go文件。 1、client.go
package registry
import (
"bytes"
"encoding/json"
"fmt"
"net/http"
)
func RegisterService(r Registration)error {
buf := new(bytes.Buffer)
enc := json.NewEncoder(buf)
err :=enc.Encode(r)
if err!=nil{
return err
}
res,err := http.Post(ServiceURL,"application/json",buf)
if err !=nil{
return err
}
if res.StatusCode!=http.StatusOK{
return fmt.Errorf("Failed to register")
}
return nil
}
主要用于客户端的注册服务,其中创建Post请求向注册地址即ServiceURL请求注册服务 2、registration.go
package registry
type Registration struct {
ServiceName ServiceName
ServiceURL string
}
type ServiceName string
const(
LogService = ServiceName("LogService")
)
创建注册信息结构体包括服务名服务地址 3、server.go
package registry
import (
"encoding/json"
"log"
"net/http"
"sync"
)
const ServerPort =":3000"
const ServiceURL = "http://localhost"+ServerPort+"/services"
type registry struct {
registrations []Registration
mutex *sync.Mutex
}
func (r *registry)add(reg Registration)error {
r.mutex.Lock()
r.registrations = append(r.registrations,reg)
r.mutex.Unlock()
return nil
}
var reg = registry{
registrations: make([]Registration,0),
mutex: new(sync.Mutex),
}
type RegistryService struct {}
func (s RegistryService)ServeHTTP(w http.ResponseWriter,r *http.Request) {
log.Println("Request received")
switch r.Method {
case http.MethodPost:
dec := json.NewDecoder(r.Body)
var r Registration
err := dec.Decode(&r)
if err!=nil{
log.Println(err)
w.WriteHeader(http.StatusBadRequest)
return
}
log.Printf("Add service:%v with URL %v\n",r.ServiceName,r.ServiceURL)
err = reg.add(r)
if err!=nil{
log.Println(err)
w.WriteHeader(http.StatusBadRequest)
return
}
default:
w.WriteHeader(http.StatusMethodNotAllowed)
}
}
用于将请求注册的服务加入注册服务集群中,此时的RegistryService因为实现了ServeHTTP接口,故成为了一个Handle,当请求为 //访问该地址后,启动RegistryService使其成为一个handle,如下,当请求为/services时,RegistryService便会将服务信息添加到注册集群中
http.Handle("/services",®istry.RegistryService{})
在cmd包下创建如下目录 data:image/s3,"s3://crabby-images/0fa1b/0fa1b3d6ab30021a612a6ece56fe2483b3d8de18" alt="在这里插入图片描述" main.go的内容如下
package main
import (
"context"
"distributed/registry"
"fmt"
"log"
"net/http"
)
func main() {
http.Handle("/services",®istry.RegistryService{})
ctx,cancle :=context.WithCancel(context.Background())
var srv http.Server
srv.Addr = registry.ServerPort
go func() {
log.Println(srv.ListenAndServe())
cancle()
}()
go func(){
fmt.Println("Registry service started.Press any key to stop.")
var s string
fmt.Scanln(&s)
srv.Shutdown(ctx)
cancle()
}()
<-ctx.Done()
fmt.Println("shutting down registy service")
}
其中ctx的作用,个人认为是退出服务的条件,即让<-ctx.Done不再阻塞,即可退出。 参数做如下修改 data:image/s3,"s3://crabby-images/a0aad/a0aad650c2c173e22f6de32134899e26444a3fc6" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/62c02/62c02532337bc068885fcd4be139cc284aee3719" alt="在这里插入图片描述" 其中先执行注册服务,再执行日志服务 data:image/s3,"s3://crabby-images/b210f/b210f8319ab9b74e3dc889a0c1e45a0a52f7a673" alt="在这里插入图片描述" 测试工具Postman data:image/s3,"s3://crabby-images/c72ca/c72ca9d20ae400609b8de0f8c185dc8f4700f0d6" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/c549d/c549dd3664eca5d1f01970e77530330bb3e758a9" alt="在这里插入图片描述"
|