用Go語言編寫一個簡單的分布式系統(tǒng)
分布式
- 注冊服務:
RegistryService - 日志服務:
LogService - 其他服務:
GradingService、portal
RegistryService
RegistryService 提供的服務:
- 提供
/services接口,用于其他服務在啟動或者停止時告知POST:告訴RegistryService,我啟動了一個服務,調用add方法DELETE:告訴RegistryService,我停止了一個服務,調用remove方法
- 通過
add函數(shù)將服務添加到registrations列表中r.registrations = append(r.registrations, reg)
- 通過
remove函數(shù)將服務從registrations列表中移除r.registrations = append(r.registrations[:i], r.registrations[i+1:]...)
- 這里需要注意的是:要保證線程安全,也就是在
append時,需要使用到鎖
mutex.Lock() append(xxx, xxx) mutex.UnLock()
- 服務發(fā)現(xiàn):
比如說
GradingService依賴LogService,那么GradingService就需要知道LogService的地址這個時候
RegistryService就可以通過registrations列表來通知GradingService,LogService的地址RegistryService是通過ServiceUpdateURL來通知的,GradingService,LogService的地址 - 服務發(fā)現(xiàn)需要分兩步進行
如果
GradingService啟動時,如果LogService已經(jīng)啟動了,那么RegistryService就可以直接通知GradingService,LogService的地址(r.sendRequiredServices(reg)方法)如果
GradingService啟動時,如果LogService還沒有啟動,那么RegistryService就不會通知GradingService,LogService的地址,等到LogService啟動后,RegistryService才會通知GradingService,LogService的地址(notify方法)
RegistryService 對外只需要提供 RegisterService 方法,其他服務調用這個函數(shù),就能夠獲取 RegistryService 提供的服務
- 調用
RegisterService提供的接口/services,將服務注冊到RegistryService中 - 為注冊的服務添加路由:
ServiceUpdateURL - 為注冊的服務添加
ServeHTTP方法,用于處理ServiceUpdateURL的請求,這個請求在方法sendRequiredServices調用時相應,更新providers中的service - 為每個注冊的服務提供健康檢查
最后在提供一個 ShutdownService 用于像 /services 接口發(fā)送 delete 請求,告知 RegistryService,我停止了一個服務
LogService
LogService 服務是對日志進行管理,將其他服務的日志進行收集、存儲,提供 /log 接口,用于其他服務將日志發(fā)送給 LogService
GradingService 和 Portal
這兩個是業(yè)務服務
- 在啟動服務時調用方法
RegistryService,將自己注冊到RegistryService中 - 在停止服務時調用方法
ShutdownService,將自己從RegistryService中移除
api
os.OpenFile
用于指定模式打開文件,并返回文件的指針
func OpenFile(name string, flag int, perm FileMode) (*File, error)
flag 參數(shù):
os.O_RDONLY:只讀模式打開文件os.O_WRONLY:只寫模式打開文件os.O_RDWR:讀寫模式打開文件os.O_APPEND:追加模式,寫入內容時將數(shù)據(jù)附加到文件尾部os.O_CREATE:如果文件不存在,則創(chuàng)建一個新文件
perm 參數(shù):
- 0:無權限
- 1:執(zhí)行權限
- 2:寫權限
- 3:寫和執(zhí)行權限
- 4:讀權限
- 5:讀和執(zhí)行權限
- 6:讀和寫權限
- 7:讀、寫和執(zhí)行權限
0644:表示文件的所有者可以讀取和寫入文件,文件所屬組和其他用戶只能讀取文件。這是比較常見的設置0600:表示文件的所有者可以讀取和寫入文件,但是文件所屬組和其他用戶不能訪問該文件。這種權限安全性較高
ioutil.ReadAll
可以將整個文件內容讀取到內存中,可以將請求體的內容讀取到內存中
ps:將整個文件的內容或者請求體一次性讀取到內存中,對于非常大的文件或者請求體,內存占用過高
fmt.Scanln
會阻塞程序的執(zhí)行,直到用戶在終端輸入一行內容并按下回車鍵,然后它會將用戶輸入的值存儲到傳入的參數(shù)中
它主要用于讀取并解析簡單的基本類型數(shù)據(jù)
func main(){
var name string
var age int
fmt.Print("Enter your name: ")
fmt.Scanln(&name)
fmt.Print("Enter your age: ")
fmt.Scanln(&age)
fmt.Printf("Hello, %s! You are %d years old.\n", name, age)
}http
http.Server
ListenAndServe:啟動服務,并監(jiān)聽指定的地址和端口,會阻塞Shutdown:優(yōu)雅地關閉服務,可以保證正在處理的服務不會被中斷
var srv htto.Server
go func(){
srv.ListenAndServe()
}()
go func(){
srv.Shutdown()
}()ServeHTTP
當一個結構體實現(xiàn)了 ServeHTTP 方法后,那么這個結構體就實現(xiàn)了 http.Handler 接口
實現(xiàn)了 http.Handler 接口的結構體,就可以作為 http.Handle 方法的第二個參數(shù)
然后調用 http.ListenAndServe 方法就可以啟動一個服務,會自動調用 ServeHTTP 方法來處理請求
func main() {
http.Handle("/ping", &MyHandler{})
http.ListenAndServe(":8080", nil)
}
type MyHandler struct{}
func (mh MyHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
switch r.Method {
case http.MethodGet:
w.WriteHeader(http.StatusOK)
w.Write([]byte("pong"))
default:
w.WriteHeader(http.StatusMethodNotAllowed)
}
}將結構體序列化
buf := new(bytes.Buffer)創(chuàng)建了一個新的bytes.Buffer對象,用于存儲編碼后的JSON數(shù)據(jù)enc := json.NewEncoder(buf)創(chuàng)建了一個新的JSON編碼器enc,并將其關聯(lián)到buf對象。這意味著編碼后的JSON數(shù)據(jù)將被寫入到buf中err := enc.Encode(r)使用JSON編碼器enc將結構體r編碼為JSON數(shù)據(jù),并將結果寫入到buf中。Encode方法返回一個可能的錯誤err
type Registration struct {
ServiceName string
ServiceURL string
}
r := Registration{
ServiceName: "LogService",
ServiceURL: "http://localhost:3000/services",
}
buf := new(bytes.Buffer)
enc := json.NewEncoder(buf)
err := enc.Encode(r)
res, err := http.Post(ServicesURL, "application/json", buf)使用 http 默認請求
http.DefaultClient是標準庫中提供的默認HTTP請求。它已經(jīng)預先配置好了一些默認的設置,例如超時時間、重試機制等Do(req)是http.Client類型的方法,用于執(zhí)行一個HTTP請求并返回響應- 它接受一個
http.Request對象作為參數(shù),表示要發(fā)送的請求
- 它接受一個
req, _ := http.NewRequest(http.MethodDelete, "http://localhost:3000/services", bytes.NewBuffer([]byte("http://localhost:4000/log")))
req.Header.Add("Content-Type", "text/plain")
res, err := http.DefaultClient.Do(req)log
log.New
log.New 用于創(chuàng)建一個新的日志記錄器實例,用于將日志消息寫入指定的輸出地,并可選擇性地添加前綴字符串
- 以文件的形式記錄日志,用
log.New創(chuàng)建一個新的log實例,然后調用log.Printf方法將日志寫入文件
它接收 io.Writer 類型的參數(shù),os.OpenFile 返回的文件指針類型 *os.File 實現(xiàn)了 io.Writer 接口,所以可以將文件指針傳入 log.New 方法中
代碼參考如下:
import (
"fmt"
stlog "log"
"os"
)
func main() {
file, err := os.OpenFile("./logs", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0600)
if err != nil {
fmt.Println(err)
}
defer file.Close()
log := stlog.New(file, "[go] - ", stlog.LstdFlags)
log.Println("hello world")
}- 重寫
log的Write方法,也能實現(xiàn)將日志寫入文件
在重寫 Write 方法時,需要定義一個類型別名,然后在類型別名上實現(xiàn) Write 方法,那么這個類型別名就能夠傳入 log.New 方法中
代碼參考如下:
import (
stlog "log"
"os"
)
type filelog string
func (fl filelog) Write(data []byte) (int, error) {
file, err := os.OpenFile(string(fl), os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0600)
if err != nil {
return 0, err
}
defer file.Close()
file.Write(data)
return len(data), nil
}
func main() {
log := stlog.New(filelog("./logs"), "[go] - ", stlog.LstdFlags)
log.Println("hello world")
}以上就是用Go語言編寫一個簡單的分布式系統(tǒng)的詳細內容,更多關于Go語言分布式系統(tǒng)的資料請關注腳本之家其它相關文章!

