最美情侣中文字幕电影,在线麻豆精品传媒,在线网站高清黄,久久黄色视频

歡迎光臨散文網(wǎng) 會(huì)員登陸 & 注冊(cè)

如何實(shí)現(xiàn)四層負(fù)載均衡

2023-06-30 21:41 作者:Cpp程序員  | 我要投稿

四層負(fù)載均衡和七層負(fù)載均衡

七層負(fù)載均衡

首先,我們來(lái)看下七層負(fù)載均衡,它一般是針對(duì)應(yīng)用層請(qǐng)求協(xié)議做請(qǐng)求轉(zhuǎn)發(fā),拿http請(qǐng)求舉例,有A,B兩臺(tái)服務(wù)器,如果采用輪詢的負(fù)載均衡策略,負(fù)載均衡器將第一個(gè)請(qǐng)求轉(zhuǎn)發(fā)給了A服務(wù)器,那么第二個(gè)請(qǐng)求到達(dá)時(shí),負(fù)載均衡器就會(huì)把請(qǐng)求轉(zhuǎn)發(fā)到B服務(wù)器。

在轉(zhuǎn)發(fā)時(shí),能夠在應(yīng)用協(xié)議層對(duì)請(qǐng)求做一些變動(dòng),拿http請(qǐng)求來(lái)說(shuō),可以對(duì)http的請(qǐng)求頭,http路徑做相應(yīng)的變動(dòng)。

四層負(fù)載均衡

再來(lái)看看四層負(fù)載均衡,它一般是指針對(duì)連接做的負(fù)載均衡,舉例說(shuō)明下,有A,B兩臺(tái)服務(wù)器,同樣采取輪詢的策略,某個(gè)客戶端發(fā)起一個(gè)新的連接,經(jīng)過(guò)均衡器連接到了A服務(wù)器,現(xiàn)在又來(lái)一個(gè)客戶端同樣發(fā)起連接,經(jīng)過(guò)均衡器后,此時(shí)就該和B服務(wù)器建立連接了。而在同一個(gè)連接里是能夠發(fā)送多個(gè)請(qǐng)求的,這也是和七層負(fù)載均衡最本質(zhì)的區(qū)別,它是針對(duì)連接做的負(fù)載均衡。

實(shí)現(xiàn)四層負(fù)載均衡器

實(shí)現(xiàn)四層負(fù)載均衡策略的方式有很多,比較著名的四層負(fù)載均衡軟件就有l(wèi)vs,它是通過(guò)修改數(shù)據(jù)包的ip地址或者mac地址實(shí)現(xiàn)四層負(fù)載均衡,性能較好,工作模式有好幾種,具體的就不在本文展開(kāi)了。

本文實(shí)現(xiàn)的四層負(fù)載均衡的原理和nginx四層負(fù)載類似?,通過(guò)均衡器在客戶端和服務(wù)端之前都維護(hù)一個(gè)連接來(lái)達(dá)到讓 客戶端在同一個(gè)連接里發(fā)送的請(qǐng)求都會(huì)被服務(wù)端同一個(gè)連接所接收的目的。如下圖所示:

以后client1 通過(guò)連接A發(fā)的請(qǐng)求都會(huì)由連接B發(fā)往服務(wù)器,而client2通過(guò)連接C發(fā)送的請(qǐng)求,都將經(jīng)過(guò)連接D發(fā)往另一臺(tái)服務(wù)器。

實(shí)現(xiàn)邏輯

現(xiàn)在讓我們來(lái)實(shí)現(xiàn)下這部分的邏輯,我將會(huì)以輪詢的策略實(shí)現(xiàn)連接的負(fù)載均衡。

并且這里還要考慮下實(shí)現(xiàn)數(shù)據(jù)復(fù)制的邏輯,我們需要在均衡器分別建立對(duì)客戶端和服務(wù)端的socket連接,并且將其中一個(gè)socket的數(shù)據(jù)轉(zhuǎn)移到另一個(gè)socket,如果每次都將某一個(gè)socket數(shù)據(jù)讀到用戶層,再寫(xiě)到另一個(gè)socket就會(huì)導(dǎo)致一些沒(méi)有必要的拷貝。偽代碼如下:

var ( src net.Conn ?// 一個(gè)socket 連接dst net.Conn ?// 一個(gè)socket連接)// ...buf = make([]byte, size) ? ? nr, er := src.Read(buf) nw, ew := dst.Write(buf[0:nr])

有沒(méi)有什么技術(shù)讓內(nèi)核自動(dòng)將某個(gè)socket的數(shù)據(jù)轉(zhuǎn)移到另一個(gè)socket,不用將數(shù)據(jù)拷貝到應(yīng)用層來(lái),這正是零拷貝相關(guān)的技術(shù),關(guān)于零拷貝的技術(shù)原理我在之前這篇文章?有很詳細(xì)的介紹,內(nèi)核提供了一個(gè)splice的系統(tǒng)調(diào)用,專門用于socket連接間拷貝數(shù)據(jù),只需要調(diào)用時(shí)傳入對(duì)應(yīng)socket連接的文件描述符即可讓內(nèi)核自動(dòng)完成拷貝過(guò)程。

func?Splice(rfd?int,?roff?*int64,?wfd?int,?woff?*int64,?len?int,?flags?int)?(n?int64,?err?error)?

這個(gè)系統(tǒng)調(diào)用已經(jīng)被golang更深層次的封裝到了一個(gè)比較常用的方法io.Copy里,這個(gè)方法會(huì)自動(dòng)判斷reader和writer底層的類型,如果都是socket連接則會(huì)調(diào)用splice系統(tǒng)調(diào)用實(shí)現(xiàn)零拷貝。

func Copy(dst Writer, src Reader) (written int64, err error) { ? ? return copyBuffer(dst, src, nil) ? }

接著我們看下均衡的代碼邏輯,運(yùn)行邏輯如下:

1, 監(jiān)聽(tīng)到新連接,啟動(dòng)一個(gè)協(xié)程去處理連接。

2 , 在新協(xié)程里與通過(guò)輪詢的策略,選擇一個(gè)后端服務(wù)器并與之建立連接。

3, 啟動(dòng)兩個(gè)協(xié)程分別進(jìn)行io.Copy ,將客戶端的socket寫(xiě)到服務(wù)端socket,將服務(wù)端socket返回的信息寫(xiě)到客戶端socket。代碼如下:

type Server struct { ? ? Li ? ? ?net.Listener ? ? Balance balancepolicy.Policy ? } ? ?func (s *Server) Run() { ? ? for { ? ? ? ?c, err := s.Li.Accept() ? ? ? ?if err != nil { ? ? ? ? ? log.Fatal(err) ? ? ? ?} ? ? ? ?go func(c net.Conn) { ? ? ? ? ? remoteAddr := c.RemoteAddr() ? ? ? ? ? backendIp := s.Balance.PickNode(remoteAddr.String()) ? ? ? ? ? serverConn, err := net.Dial("tcp", backendIp) ? ? ? ? ? if err != nil { ? ? ? ? ? ? ?log.Fatal(err) ? ? ? ? ? ? ?c.Close() ? ? ? ? ? ? ?return ? ? ? ? ? } ? ? ? ? ? fmt.Println("獲取到了新連接", remoteAddr, backendIp) ? ? ? ? ? go func() { ? ? ? ? ? ? ?_, err := io.Copy(serverConn, c) ? ? ? ? ? ? ?if err != nil { ? ? ? ? ? ? ? ? fmt.Println(err, 1) ? ? ? ? ? ? ?} ? ? ? ? ? ? ?c.Close() ? ? ? ? ? ? ?serverConn.Close() ? ? ? ? ? ? ?fmt.Println("結(jié)束1", err) ? ? ? ? ? }() ? ? ? ? ? go func() { ? ? ? ? ? ? ?_, err := io.Copy(c, serverConn) ? ? ? ? ? ? ?if err != nil { ? ? ? ? ? ? ? ? fmt.Println(err, 2) ? ? ? ? ? ? ?} ? ? ? ? ? ? ?c.Close() ? ? ? ? ? ? ?serverConn.Close() ? ? ? ? ? ? ?fmt.Println("結(jié)束2", err) ? ? ? ? ? }() ? ? ? ?}(c) ? ? } ? ? }

io.Copy 會(huì)不斷的拷貝源socket的數(shù)據(jù)到目的socket,直到連接關(guān)閉。

更好的方案

可以看到上述方案中維護(hù)一個(gè)客戶端的連接將會(huì)啟動(dòng)3個(gè)協(xié)程,當(dāng)連接量上去后,均衡器很可能成為瓶頸,有沒(méi)有辦法減少下協(xié)程的數(shù)量,可以直接采用epoll的方式監(jiān)聽(tīng)連接的讀寫(xiě),以及關(guān)閉事件(這樣能在一個(gè)協(xié)程里處理多個(gè)連接),當(dāng)連接可讀時(shí),直接使用splice系統(tǒng)調(diào)用對(duì)數(shù)據(jù)進(jìn)行拷貝直到返回syscall.EAGAIN 就停止,因?yàn)榉祷豷yscall.EAGAIN 說(shuō)明連接緩沖區(qū)內(nèi)的數(shù)據(jù)暫時(shí)被讀取完了,繼續(xù)下一次epoll wait的監(jiān)聽(tīng)循環(huán)。這樣能極大的減少協(xié)程數(shù)量。不過(guò)實(shí)現(xiàn)我就不準(zhǔn)備再繼續(xù)展開(kāi)了,后續(xù)有空再補(bǔ)充下這部分。對(duì)epoll的使用有興趣的同學(xué)也可以看看我之前一篇用epoll實(shí)現(xiàn)類似redis的網(wǎng)絡(luò)模型框架這篇文章

測(cè)試負(fù)載均衡代碼

現(xiàn)在讓我們來(lái)測(cè)試下負(fù)載均衡的代碼,我會(huì)用docker-compose去啟動(dòng)兩個(gè)mysql,然后本地啟動(dòng)我們負(fù)載均衡器的代碼,之后用兩個(gè)mysql客戶端去連接負(fù)載均衡器,看下是不是mysql客戶端連接到了不同的mysql服務(wù)器。

docker-compose的配置文件如下:

version: '3' ? services: ? ?mysql1: ? ? ?restart: always ? ? ?image: amd64/mysql:latest ? ? ?container_name: mysql1 ? ? ?environment: ? ? ? ?- "MYSQL_ROOT_PASSWORD=1234567" ? ? ? ?- "MYSQL_DATABASE=test" ? ? ?ports: ? ? ? ?- "3306:3306" ? ? ?mysql2: ? ? ?restart: always ? ? ?image: amd64/mysql:latest ? ? ?container_name: mysql2 ? ? ?environment: ? ? ? ?- "MYSQL_ROOT_PASSWORD=1234567" ? ? ? ?- "MYSQL_DATABASE=test2" ? ? ?ports: ? ? ? ?- "3307:3306"

為了能驗(yàn)證不同客戶端的確連上了不同的mysql服務(wù)器,我在mysql1上創(chuàng)建了test數(shù)據(jù)庫(kù),在mysql2上創(chuàng)建了test2數(shù)據(jù)庫(kù)。到時(shí)候連上不同服務(wù)器數(shù)據(jù)庫(kù)是不一樣的。

均衡服務(wù)器監(jiān)聽(tīng)5555端口啟動(dòng)

s := &proxy.Server{} ? li, err := net.Listen("tcp", ":5555") ?if err != nil { ? ? log.Fatal(err) ? } ? s.Li = li ? s.Balance = balancepolicy.NewRoundRobin() ? s.Balance.AddNode("127.0.0.1:3306", "mysql1") ? s.Balance.AddNode("127.0.0.1:3307", "mysql2") ? s.Run()

之后用mysql客戶端去連接均衡服務(wù)器

## client1mysql -h 127.0.0.1 -u root ?-P 5555 ?-D test ?-p1234567## client2mysql -h 127.0.0.1 -u root ?-P 5555 ?-D test2 ?-p1234567

發(fā)現(xiàn)兩個(gè)mysql客戶端的確連接到了不同服務(wù)器,并且能正常執(zhí)行命令,over。


如何實(shí)現(xiàn)四層負(fù)載均衡的評(píng)論 (共 條)

分享到微博請(qǐng)遵守國(guó)家法律
巴东县| 灵川县| 兴城市| 镇康县| 盐山县| 铁岭市| 商都县| 彭山县| 随州市| 衢州市| 荆门市| 玛沁县| 金阳县| 新疆| 衡东县| 孝昌县| 祥云县| 河北省| 鄯善县| 临安市| 新昌县| 九江市| 商丘市| 老河口市| 苗栗县| 阿克| 河北省| 牡丹江市| 乐清市| 镇赉县| 景泰县| 中卫市| 宁河县| 阿巴嘎旗| 安龙县| 陆丰市| 寻乌县| 牙克石市| 东方市| 扬州市| 中超|