明年6月才國產!全新5系值得等嗎?

0T直六發動機,搭配8AT變速箱。從數據上看,動力單元已經足夠能滿足大部分用戶。除了一貫的后驅,新5系還能選裝四驅系統,依然能保留寶馬的操控特性。而新5系最大的亮點是在車鑰匙加入了一系列的軟件功能,中控觸摸屏也加入了手勢功能。

今年寶馬5系迎來了代號為G30的換代車型。雖然現在只放出了540i一款車型的信息,但新5系的進化已經足以讓買家耐心去等待了。

新5系的造型已經改得很像小一號的7系,頭尾燈的造型也有了變化,加上雙腰線的處理,新5系的顏值還是有所提升。

而視頻里的試駕車540i,是造裝了M系列套件,新5系將搭載2.0T和3.0T直六發動機,搭配8AT變速箱。從數據上看,動力單元已經足夠能滿足大部分用戶。除了一貫的后驅,新5系還能選裝四驅系統,依然能保留寶馬的操控特性。

而新5系最大的亮點是在車鑰匙加入了一系列的軟件功能,中控觸摸屏也加入了手勢功能。這些改變雖然無足輕重,但對於車主的體驗來說一定會更上一個檔次。

據聞新5系明年就會國產,作為一款在國內知名度足夠高的豪華轎車,它的上市一定會吸引到不少的買家。本站聲明:網站內容來源於http://www.auto6s.com/,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

台北網頁設計公司這麼多該如何選擇?

※智慧手機時代的來臨,RWD網頁設計為架站首選

※評比南投搬家公司費用收費行情懶人包大公開

※回頭車貨運收費標準

網頁設計最專業,超強功能平台可客製化

※別再煩惱如何寫文案,掌握八大原則!

發動機不磨合?機油燒死你!

還說買了性能車,直接來一發暴力磨合。然而事實是,出廠前就磨合好的車只是佔少數,同時暴力磨合也是一種不合理的磨合方式。所以,作為普通車主,還是不要搞什麼幺蛾子了。另外,磨合的時候多跑高速也是有特定條件的。

大家常聽老司機說新車要磨合,有所謂的磨合期,那麼磨合到底是什麼鬼?

發動機是磨合過程中最核心的部分,為了讓內部的活塞和發動機缸體達到完美的活塞運動,這時候就要在它們之間放置活塞環和缸套。

而磨合的作用,是要讓活塞環與缸套的內壁完美地磨合密封。

但是因為活塞環有彈性,如果金屬內壁不夠平滑,那就很容易出現各種磨損,最明顯的一個傷害就是燒機油。所以,所謂磨合期,就是要慢慢的讓發動機進行活塞運動,然後讓零件間的接合面打磨得更加貼合,達到最佳狀態。

不過長期以來,各種鍵盤車神和老司機間都流傳着一些以訛傳訛的說法,比如說新車早就在廠里磨合好,買回來就儘管飆。還說買了性能車,直接來一發暴力磨合。

然而事實是,出廠前就磨合好的車只是佔少數,同時暴力磨合也是一種不合理的磨合方式。所以,作為普通車主,還是不要搞什麼幺蛾子了。

另外,磨合的時候多跑高速也是有特定條件的。在跑的過程中多切換不同的擋位和油門深淺,同是不要貪新鮮開個幾百米就算,這樣各類油溫沒起來,會對車輛造成一定損害。

不過說了那麼多,千言萬語合萬一句,就是要多看看車主手冊。看完這期節目,大家記得要等着看下一期,什麼是燒機油!本站聲明:網站內容來源於http://www.auto6s.com/,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

台北網頁設計公司這麼多該如何選擇?

※智慧手機時代的來臨,RWD網頁設計為架站首選

※評比南投搬家公司費用收費行情懶人包大公開

※回頭車貨運收費標準

網頁設計最專業,超強功能平台可客製化

※別再煩惱如何寫文案,掌握八大原則!

厲害word哥!上汽打算今年賣30萬輛車!

99萬)11月銷售1。2萬輛榮威 RX5搭載1。5T/2。0T發動機,MT/AT可選,參考價格(9。98萬至18。68萬)11月銷量2。1萬輛其中70%為互聯網版本乘着廣州車展的東風,11月上汽榮威品牌成績斐然,傳統熱銷車型榮威360和榮威350月銷售均破萬,累計銷量超1。

“國家政策添把火,國內廠商出爆款”來形容2016年的車市並不為過。近日,自主品牌一哥上汽傳來一則好消息,通過其爆款互聯網車型“RX5”的熱銷,已提前完成了2016年全年的銷售指標,劍指年銷售30萬量的目標(月2.5萬/輛),穩坐中國自主品牌領導者的地位。

榮威350

搭載1.5L發動機,MT/AT可選,

參考價格(7.87萬至9.97萬)

11月銷售1.2萬輛

榮威360

搭載1.5L/1.4T發動機,MT/AT可選,

參考價格(7.59萬至12.99萬)

11月銷售1.2萬輛

榮威 RX5

搭載1.5T/2.0T發動機,MT/AT可選,

參考價格(9.98萬至18.68萬)

11月銷量2.1萬輛

其中70%為互聯網版本

乘着廣州車展的東風,11月上汽榮威品牌成績斐然,傳統熱銷車型榮威360和榮威350月銷售均破萬,累計銷量超1.2萬輛。帶有馬雲爸爸光環的RX5更熱銷21344輛,三款熱銷車型帶動品牌增長超過173%,創上汽集團歷史新高,縱觀國內車市,消費者對自主品牌和互聯網汽車的認同感不斷提高,榮威品牌更獲得消費者的強烈認可。

自主品牌汽車與互聯網的結合,開始很多人並不看好,但上汽用一張漂亮的成績單告訴質疑者,在“陳虹爸爸”和“馬雲爸爸”的全力支持下,自主品牌也可以有越級的品質、各種黑科技,更重要的是,讓中國消費者買得起,用的爽,隨着榮威新一代產品Erx5和i6燈發布上市,必然會掀起一陣購車熱潮。

未來的車型發展中,上汽更深耕“互聯網概念汽車”和“新能源汽車”等技術,搭載更多貼合潮流和先進技術的汽車推出市面,如一發布就引起各方興趣的MG-ZS、榮威I6等,讓人拭目以待吧。

本站聲明:網站內容來源於http://www.auto6s.com/,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

網頁設計一頭霧水該從何著手呢? 台北網頁設計公司幫您輕鬆架站!

網頁設計公司推薦不同的風格,搶佔消費者視覺第一線

※Google地圖已可更新顯示潭子電動車充電站設置地點!!

※廣告預算用在刀口上,台北網頁設計公司幫您達到更多曝光效益

※別再煩惱如何寫文案,掌握八大原則!

網頁設計最專業,超強功能平台可客製化

※回頭車貨運收費標準

澳洲發現無齒恐龍 與迅猛龍是近親

摘錄自2020年5月19日自由時報報導

澳洲發現新種無齒恐龍,跟暴龍、迅猛龍是近親!澳洲古生物學家日前發布了新品種恐龍,此種新恐龍雖然與暴龍和迅猛龍同屬於一個亞目,但卻有著長脖子、沒有牙齒、飲食習性也不相同等特點。學界尚未給予正式名稱,研究人員暫時將其稱作「伊拉夫羅龍(Era the Elaphrosaur)」。

根據《BBC》報導,伊拉夫羅龍在維多利亞州奧特韋角出土,作為新品種的獸腳亞目,這種新恐龍具有許多古代捕食者的熟悉體型,牠採用雙足站立,加上兩隻些微笨拙退化的前爪(小胳膊),甚至可能披覆著一層羽毛,但從鼻子到尾巴的身長只有約2公尺左右。

伊拉夫羅龍與其他獸腳亞目的最大區別是,其脖子較大多數獸腳類要長得多,且似乎只有小時候才有牙齒。隨著年齡的增長,伊拉夫羅龍的牙齒會逐漸退化,並留下角質的喙。該物種可能小時候吃肉、長大後又轉向吃素。不過研究人員暫時還無法確定,因為目前暫缺頭骨部分的化石(只有頸骨部分)。

生活環境
國際新聞
澳洲
恐龍

本站聲明:網站內容來源環境資訊中心https://e-info.org.tw/,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

※自行創業缺乏曝光? 網頁設計幫您第一時間規劃公司的形象門面

※如何讓商品強力曝光呢? 網頁設計公司幫您建置最吸引人的網站,提高曝光率!

※綠能、環保無空污,成為電動車最新代名詞,目前市場使用率逐漸普及化

※廣告預算用在刀口上,台北網頁設計公司幫您達到更多曝光效益

※教你寫出一流的銷售文案?

※別再煩惱如何寫文案,掌握八大原則!

疫情沒遊客使盜獵激增 柬埔寨國鳥也遇難 3隻極危大䴉遭加保扶毒死

環境資訊中心綜合外電;黃鈺婷 翻譯;林大利 審校;稿源:Mongabay

本站聲明:網站內容來源環境資訊中心https://e-info.org.tw/,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

台北網頁設計公司這麼多該如何選擇?

※智慧手機時代的來臨,RWD網頁設計為架站首選

※評比南投搬家公司費用收費行情懶人包大公開

※回頭車貨運收費標準

網頁設計最專業,超強功能平台可客製化

※別再煩惱如何寫文案,掌握八大原則!

皮實耐用家轎王終於迎來新款 7.99萬起!

新車的動力系統為1。4L 90馬力+5擋手動、1。5L 110馬力+5擋手動/6擋自動、1。4T 131馬力+7擋雙離合。老款的1。6L發動機將被新款的1。5L所取代。1。5L發動機最大馬力和1。6L相同,只是最大扭矩由160牛·米變為150牛·米,在保持動力基本不變的前提下可以降低油耗,這也是消費者喜歡的。

捷達自從2013年上市以來,一直沒有經過什麼大的改變。這一點讓消費者不太滿意。不過這一情況如今得到了改善,目前新捷達已經上市,外觀改變較大,配置提升,1.5L發動機取代1.6L發動機。下面就一起看看改款后的捷達到底性價比如何。

一汽-大眾新捷達在12月7日晚上上市,新車售價為7.99-13.49萬,共推出9個車型供消費者選擇。全新捷達上市,老款捷達處於停產在售狀態,同時1.6L車型被1.5L車型取代。

既然是新款車型,最起碼外觀還是要做出一點改變的吧!要不然就太對不起觀眾了。捷達也不例外,雖然還是遵從着大眾的家族式設計,但是不管是外觀還是內飾,還是有所改變的。

首先是前臉的變化,這也是最明的變化。新車的前進氣格柵採用了網格狀造型,同時增加了鍍鉻元素,前大燈造型和保險杠也有較大的改變,整體來看新車看起來比老款要年輕時尚許多。

前臉看完了來看側面,側面的變化非常小,新車的車窗下方會有鍍鉻裝飾條,只是高配車型的輪轂造型有一些變化。

尾部造型變化也比較大,新款車型的尾部看起來“機靈”了許多,造型也向大哥寶來靠攏。

新款車型的顏色增加了胡桃棕、鈦光灰兩種顏色。同時還有深黑、甜蜜金、反射銀、糖果白、板岩灰五種顏色。新車一共有七種車身配色供消費者選擇。

同時新車的車尺寸為4501*1704*1469mm,軸距為2604mm,老款車型的車身尺寸4487*1706*1470mm,軸距為2603mm。尺寸略有增大。

內飾有所升級,整體質感比老款車型有所提升。整車的配置也有不小的升級,全系車型除了時尚版,都標配了車身穩定系統,要知道,這可是一款捷達,大眾能給它裝上ESp算是有點意外了,老款車型只有頂配才有。同時新車也會有座椅加熱、胎壓監測、真皮坐椅等,有了這些配置,捷達終於有點上檔次了。

新車的動力系統為1.4L 90馬力+5擋手動、1.5L 110馬力+5擋手動/6擋自動、1.4T 131馬力+7擋雙離合。老款的1.6L發動機將被新款的1.5L所取代。1.5L發動機最大馬力和1.6L相同,只是最大扭矩由160牛·米變為150牛·米,在保持動力基本不變的前提下可以降低油耗,這也是消費者喜歡的。

競爭對手:

可以看出新捷達的配置有所提升,外觀也變得更加年輕了,競爭力也會隨之增強。但是捷達所處的這個級別競爭真的是特別激烈。他面臨的競爭對手也不是吃素的,不少車型性價要比捷達高。

吉利汽車-帝豪

帝豪的指導價為6.98-24.98萬,作為最暢銷的自主品牌緊湊型轎車,同價位的帝豪要比捷達的配置高了幾個檔次。不管是做工還是用料都不輸捷達,差的只是“車標”吧!

奇瑞汽車-艾瑞澤5

艾5的指導價為5.89-9.79萬,作為自主品牌性價比最高的車型之一,艾瑞澤5的實力一點也不差,同等配置,選擇艾5,省下的幾萬塊錢可以買多少升汽油了,可以多跑多少公里了…

上汽通用雪佛蘭-科沃茲

科沃茲的指導價為7.99-10.99萬,就算你避過了自主品牌的鋒芒,同級別合資車這一關你也不過好過吧!科沃茲的配置比你好,車身也比你大,外觀也很時尚。

除了這些還有鋒范、標緻301、桑塔納等,所以捷達如果還想維持較高的銷量,就必須和老款一樣,大力降價促銷,畢竟薄利多銷也是一個很好的選擇么!本站聲明:網站內容來源於http://www.auto6s.com/,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

※自行創業缺乏曝光? 網頁設計幫您第一時間規劃公司的形象門面

※如何讓商品強力曝光呢? 網頁設計公司幫您建置最吸引人的網站,提高曝光率!

※綠能、環保無空污,成為電動車最新代名詞,目前市場使用率逐漸普及化

※廣告預算用在刀口上,台北網頁設計公司幫您達到更多曝光效益

※教你寫出一流的銷售文案?

※別再煩惱如何寫文案,掌握八大原則!

運動與豪華的完美平衡:全新保時捷Panamera亮相廣州車展

6秒,最高時速可達306 km/h。此外,首次配備的全新8速保時捷雙離合器變速箱(porsche Doppelkupplung II)可提供不間斷的動力傳輸,同時有效降低油耗。重新設計的底盤使全新panamera在擁有豪華轎車巡航舒適性的同時,兼具跑車的賽道性能。

上海/廣州。11月18日,保時捷攜旗下全系頂級跑車亮相第14屆廣州國際汽車展覽會,以享譽全球的運動基因和創新科技,詮釋超凡性能與日常實用性的完美平衡。與此同時,擁有保時捷純正跑車基因的豪華轎車全新panamera更是首次於公眾面前華麗亮相,展示其卓越運動性和豪華舒適性。此外,保時捷旗下中置發動機跑車最新力作全新718系列、中置跑車鼻祖——550 Spyder和Cayenne platinum Edition亦同台亮相。保時捷為廣大車友傾情呈現的這場跑車盛宴,不僅詮釋了賽道激情與跑車文化的無窮魅力,更體現出保時捷不懈追求巔峰性能、至臻設計和馭風樂趣的獨特價值以及對中國市場的高度重視。

繼2015年中國首次躍居保時捷全球最大單一市場,2016年保時捷中國繼續保持穩定增長的勢頭,於前十個月實現超過55,000台新車交付量。“保時捷中國之所以能保持如此良好的業績得益於我們全面的產品線以及不斷拓展的專業經銷商網絡。”保時捷中國總裁及首席執行官方智勇先生(Franz Jung)表示,“隨着中國豪華車細分市場結構的調整和消費者需求的成熟化,我們將繼續推出更具競爭力的產品,不斷積累和提升保時捷的品牌價值。此次亮相廣州車展的全新panamera即是保時捷對華承諾的最好印證。”

全新panamera再續輝煌

7年前,保時捷在上海全球首發panamera車型,並成功打入運動型四門豪華轎車的細分市場。迄今,已有超過150,000台panamera銷往世界各地,其中中國市場的總銷量超過46,000台,佔全球總銷量的30%以上,使得中國成為panamera目前為止全球最大的單一市場。新車亮相廣州車展,再次印證保時捷對中國市場的高度重視與長期承諾。

此次,亮相廣州車展的兩款全新panamera車型——panamera 4S 和panamera Turbo,兼備豪華舒適感和超凡運動性能,在低耗油及低排放,與高功率及高扭矩這兩組相互對立的特性之間實現了完美平衡,彰顯了無可比擬的性能和澎湃的動力。全新panamera 4S搭載2.9升V6雙渦輪增壓發動機,最大輸出功率為440 hp,峰值扭矩為550 Nm。百公里加速僅需4.4秒(搭配Sport Chrono組件僅需4.2秒),最高時速可達289 km/h。全新panamera Turbo搭載4.0升V8雙渦輪增壓發動機,最大輸出功率為550 hp,峰值扭矩為770 Nm。百公里加速為3.8秒;配備Sport Chrono組件后,百公里加速時間可縮短至3.6秒,最高時速可達306 km/h。此外,首次配備的全新8速保時捷雙離合器變速箱(porsche Doppelkupplung II)可提供不間斷的動力傳輸,同時有效降低油耗。

重新設計的底盤使全新panamera在擁有豪華轎車巡航舒適性的同時,兼具跑車的賽道性能。保時捷主動懸挂管理系統(pASM)和採用全新三腔室設計的自適應空氣懸架在華作為全新panamera 4S及Turbo車型的標準配置,帶來了出色的駕駛動態表現與駕乘舒適性。

除了性能升級之外,全新panamera於外觀方面也進行了更為鮮明的改動:修長而動感的車身比例、犀利的跑車輪廓、具有運動風格的側翼以及流暢的車頂線條,突顯了一款純正保時捷的特色。車尾堪稱全新 panamera 的一大亮點:帶一體式四點制動燈的三維LED尾燈賦予其獨一無二的夜間效果。

全新panamera不僅在外觀上平衡運動性與豪華性,其內飾設計也基於面向未來的理念,布局簡潔直觀且充滿科技感。保時捷先進駕駛艙(porsche Advanced Cockpit)使用觸摸液晶屏,傳統按鍵被觸摸式面板取代,彰顯了品牌純粹的未來跑車理念。得益於整車尺寸的變化,這款豪華四門轎車營造出同級領先的寬敞乘坐空間,同時也擁有豪華車型級別中最靈活可變的空間布局。行李廂容積可以更為靈活地從495升增至1,304升。車內標配的全景式天窗、帶觸摸式12.3英寸高分辨率显示屏、帶離子空氣凈化器的四區域自動恆溫空調和BOSE®環繞聲音響系統,以及Burmester®高端3D環繞聲音響系統、後座娛樂設施等一系列全新的選裝配置,進一步彰顯了panamera豪華、典雅的品質與格調。

面向未來,首推保時捷智慧互聯(porsche Connect)全新科技

為了踐行對智慧互聯的長期承諾,從全新panamera開始,保時捷將推出“保時捷智慧互聯”這一全新科技。該科技基於多點觸控和手寫輸入的全新保時捷通訊管理系統(pCM 4.1)以及智慧互聯模組升級版(Connect plus),提供在線信息娛樂服務、實時導航和禮賓服務。並通過Apple® Carplay,高效集成智能手機的諸多移動互聯應用程序。保時捷還將在2017年上半年推出保時捷ID、保時捷智慧互聯門戶、保時捷智慧互聯商店和保時捷智慧互聯支持。並將持續致力於構建一個獨一無二、高端的互聯生態體系,以進一步豐富保時捷車主的互聯駕乘體驗。

保時捷於11月16日洛杉磯國際車展首發了行政加長版車型以及搭載3.0升V6渦輪增壓發動機的panamera。至今,全新panamera車型在華已擴充至7款,並均已在中國接受預定。此次廣州車展發布的panamera 4S和panamera Turbo預計將於2017年初進入中國市場;其餘5款車型預計在明年下半年登陸全國各大保時捷中心。

更多全新panamera的衍生車型將會陸續進入中國市場。

全新panamera車型製造商建議零售價:

2016廣州車展呈現的保時捷車型:

本站聲明:網站內容來源於http://www.auto6s.com/,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

台北網頁設計公司這麼多該如何選擇?

※智慧手機時代的來臨,RWD網頁設計為架站首選

※評比南投搬家公司費用收費行情懶人包大公開

※回頭車貨運收費標準

網頁設計最專業,超強功能平台可客製化

※別再煩惱如何寫文案,掌握八大原則!

Java編程技術之淺析JVM內存

JVM

JVM->Java Virtual Machine:Java虛擬機,是一種用於計算設備的規範,它是一個虛構出來的計算機,是通過在實際的計算機上仿真模擬各種計算機功能來實現的。

基本認知:

  • 1.JVM是用於運行Java代碼的假象計算機,主要有一套字節碼指令集,一組寄存器,一個棧,一個垃圾回收,堆 和 一個存儲方法域。
  • 2.JVM運行在操作系統之上,與硬件沒有直接的交互。

Java程序執行過程:

  • 1.編譯->源文件由編譯器編譯成字節碼[ByteCode]

Java 源文件—->編譯器—->字節碼文件

  • 2.運行->字節碼由java虛擬機解釋運行

字節碼文件—->JVM—->機器碼

Java類的加載步驟:

  • 1.加載->主要是完成3個階段的提交:

通過類的全限定名來獲取定義類的二進制字節流
將字節流所代表的靜態存儲結構轉化為方法區的運行時數據結構
在內存中生成一個代表這個類的java.lang.Class對象,作為方法區這個類的各種數據的訪問入口。

  • 2.驗證->四個階段的檢驗動作:

文件格式驗證
元數據驗證
字節碼驗證
符號引用驗證

  • 3.準備->為類變量(static)分配內存並設置類變量的初始值。
  • 4.解析->將常量池內的符號引用轉為直接的引用
  • 5.初始化->按照static塊和static變量在文件中的出現順序,合併到 ()方法中。實例變量由 ()函數賦值。

JVM線程實體:

JVM線程->程序執行過程中的一個線程實體,JVM 允許一個應用併發執行多個線程。

從此,我們應該意識到,在Java中,當提到線程就應該是指JVM線程和Java線程。其中JVM線程指的是Hotspot JVM 後台運行的系統線程,而且Hotspot JVM 中的 Java 線程與原生操作系統線程有直接的映射關係。

️[注意事項]:
[1].當線程本地存儲、緩衝區分配、同步對象、棧、程序計數器等準備好以後,就會創建一個操作系統原生線程。
[2].Java 線程結束,原生線程隨之被回收。操作系統負責調度所有線程,並把它們分配到任何可用的 CPU 上。
[3].當原生線程初始化完畢,就會調用 Java 線程的 run() 方法。當線程結束時,會釋放原生線程和 Java 線程的所有資源

特別需要知道的是,Hotspot JVM 後台運行的系統線程主要是:

  • 虛擬機線程->VM thread:等待 JVM 到達安全點操作出現。這些操作必須要在獨立的線程里執行,因為當堆修改無法進行時,線程都需要 JVM 位於安全點。這些操作的類型有:stop-theworld垃圾回收、線程棧 dump、線程暫停、線程偏向鎖(biased locking)解除。
  • 周期性任務線程->負責定時器事件(也就是中斷),用來調度周期性操作的執行
  • GC線程->支持 JVM 中不同的垃圾回收活動
  • 編譯器線程->在運行時將字節碼動態編譯成本地平台相關的機器碼
  • 信號分發線程->程接收發送到 JVM 的信號並調用適當的 JVM 方法處理

JVM內存

版權聲明:本文為博主原創文章,遵循相關版權協議,如若轉載或者分享請附上原文出處鏈接和鏈接來源。

本站聲明:網站內容來源於博客園,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

※自行創業缺乏曝光? 網頁設計幫您第一時間規劃公司的形象門面

※如何讓商品強力曝光呢? 網頁設計公司幫您建置最吸引人的網站,提高曝光率!

※綠能、環保無空污,成為電動車最新代名詞,目前市場使用率逐漸普及化

※廣告預算用在刀口上,台北網頁設計公司幫您達到更多曝光效益

※教你寫出一流的銷售文案?

※別再煩惱如何寫文案,掌握八大原則!

golang連接達夢數據庫的一個坑

golang連接達夢數據庫的一個坑

有一次項目中用到了達夢數據庫,後端語言使用的golang,達夢官方並未適配專門的golang連接方式,正一籌莫展的時候發現達夢提供了odbc的連接,這樣可以使用類似mssqlodbc連接方式連接達夢數據庫。

使用的達夢數據庫版本為DM8

達夢數據庫開啟odbc連接

參考博客1、參考博客2

參照上面兩個博客內容配置odbc連接

golang代碼

一些參考文檔:

package main
import (
	"fmt"
	_ "github.com/alexbrainman/odbc"  // google's odbc driver
	"github.com/go-xorm/xorm"
	"xorm.io/core"
	"github.com/axgle/mahonia"
)

type Address struct {
    Addressid int64 `xorm:"addressid"`
    Address1 string `xorm:"address1"`
    Address2 string `xorm:"address2"`
    City string `xorm:"city"`
    Postalcode string `xorm:"postalcode"`
}

// 字符串解碼函數,處理中文亂碼
func ConvertToString(src string, srcCode string, tagCode string) string {
    srcCoder := mahonia.NewDecoder(srcCode)
    srcResult := srcCoder.ConvertString(src)
    tagCoder := mahonia.NewDecoder(tagCode)
    _, cdata, _ := tagCoder.Translate([]byte(srcResult), true)
    result := string(cdata)
    return result
}

func main() {
	engine, err := xorm.NewEngine("odbc", "driver={DM8 ODBC DRIVER};server=127.0.0.1:5236;database=DM;uid=SYSDBA;pwd=password;charset=utf8")
	if err != nil {
		fmt.Println("new engine got error:", err)
		return
	}
	engine.ShowSQL(true)//控制台打印出生成的SQL語句;
	engine.Logger().SetLevel(core.LOG_DEBUG)
	if err := engine.Ping(); err != nil {
		fmt.Println("ping got error:", err)
		return
	}

	// 1) sql查詢
	results, err := engine.Query("select addressid, address1, address2, city, postalcode from person.address limit 5 offset 2")
	if err != nil {
		fmt.Println("查詢出錯:", err)
		return
	}
	for i, e := range results {
		fmt.Printf("%v\t", i)
		for k, v := range e {
			// 達夢數據庫中文默認為gbk
			fmt.Printf("%v=%v\t", k, ConvertToString(string(v), "gbk", "utf-8"))
		}
		fmt.Printf("\n")
	}
	fmt.Println("*******************************")
	// 2) 使用struct 映射結果
	engine.SetMapper(core.SameMapper{})
	var sliceOfAddress []Address
	err = engine.Table("person.address").Limit(5, 0).Find(&sliceOfAddress)
	if err != nil {
		fmt.Println("查詢出錯:", err)
		return
	}
	for i,e := range sliceOfAddress {
		e.Address1 = ConvertToString(e.Address1, "gbk", "utf-8")
		e.Address2 = ConvertToString(e.Address2, "gbk", "utf-8")
		e.City = ConvertToString(e.City, "gbk", "utf-8")
		fmt.Printf("%v=%v\n", i, e)
	}
}

1)解決 golang.org/x/ 下包下載不下來的問題

https://studygolang.com/articles/19051?fr=sidebar

https://studygolang.com/articles/24075?fr=sidebar

2)無效的表或視圖名[person.address](這個也是最坑的一點)

原因:我們使用的是odbc的方式連接達夢數據庫,實際上使用的是mssql的驅動,第一個1) SQL查詢結果是OK的,但是2) struct 查詢就會報錯:

[xorm] [info]  2020/06/08 16:52:40.183731 [SQL] SELECT TOP 5 "addressid", "address1", "address2", "city", "postalcode" FROM "person.address"
查詢出錯: SQLPrepare: {42S02} 第1 行附近出現錯誤:
無效的表或視圖名[person.address]

通過日誌發現,xorm吧沒一個字段和表名都添加上了雙引號:SELECT TOP 5 "addressid", "address1", "address2", "city", "postalcode" FROM "person.address"這個sql在mssql中執行是沒問題的,但是放到達夢數據庫中就會報錯,因為達夢不支持雙引號包裹字段、命名空間、表名:

這樣就很坑爹了,我嘗試着修改結構體的xorm:"addressid"去掉其中的雙引號,但是問題還存在,最後想到打印出來的sql中待了雙引號,說明xorm後台還是拼接的sql語句,如果找到拼接sql語句的代碼然後去掉其中的雙引號不是就好了么。於是跟蹤代碼查找拼接sql的代碼:

解決方式一)

1、engine.Table("person.address").Limit(5, 0).Find(&sliceOfAddress)

先找到engin模塊的Find()方法:

代碼路徑:src.github.com/go-xorm/xorm/engine.go

// Find retrieve records from table, condiBeans's non-empty fields
// are conditions. beans could be []Struct, []*Struct, map[int64]Struct
// map[int64]*Struct
func (engine *Engine) Find(beans interface{}, condiBeans ...interface{}) error {
	session := engine.NewSession()
	defer session.Close()
	return session.Find(beans, condiBeans...)
}

發現其實調用的是session.Find() 方法

2、src.github.com/go-xorm/session_find.go

// Find retrieve records from table, condiBeans's non-empty fields
// are conditions. beans could be []Struct, []*Struct, map[int64]Struct
// map[int64]*Struct
func (session *Session) Find(rowsSlicePtr interface{}, condiBean ...interface{}) error {
	if session.isAutoClose {
		defer session.Close()
	}
	return session.find(rowsSlicePtr, condiBean...)
}

發現實際上調用的是session.find(rowsSlicePtr, condiBean...)

func (session *Session) find(rowsSlicePtr interface{}, condiBean ...interface{}) error {
	
	defer session.resetStatement()

	// 代碼省略 。。。

	var sqlStr string
	var args []interface{}
	var err error
    // 此處就是拼接sql的代碼
	if session.statement.RawSQL == "" {
		// 代碼省略 。。。
	} else {
		sqlStr = session.statement.RawSQL
		args = session.statement.RawParams
	}
    // 獲得配置信息判斷當前數據庫類型
	uri := session.engine.Dialect().URI()
	// 判斷當前是否是達夢數據庫
	if uri.DbType == "mssql" && uri.DbName == "DM" {
		newSqlStr := strings.Replace(sqlStr, "\"", "", -1) // 去掉雙引號
		sqlStr = newSqlStr
	}

	// 代碼省略 。。。
	return session.noCacheFind(table, sliceValue, sqlStr, args...)
}

通過session.engine.Dialect().URI()獲得配置信息,這段代碼怎麼來的,實際上是在xorm.NewEngine()的時候會解析配置信息,並賦值給enginedialect屬性,代碼位置:src.github.com/go-xorm/xorm/xorm.go

engine := &Engine{
		db:             db,
		dialect:        dialect,
		Tables:         make(map[reflect.Type]*core.Table),
		mutex:          &sync.RWMutex{},
		TagIdentifier:  "xorm",
		TZLocation:     time.Local,
		tagHandlers:    defaultTagHandlers,
		cachers:        make(map[string]core.Cacher),
		defaultContext: context.Background(),
	}

找到sql之後去掉雙引號即可,因為做了判斷只有是達夢的類型數據庫的時候才修改,所以不會影響其他類型的數據庫。至此問題得到了解決。

解決方式二)

上面的方式一是一種解決方案,其實有更簡便的,因為我們在創建engine的時候已經確定了dialect類型為dialect_mssql,找到src.github.com/go-xorm/xorm/dialect_mssql.go找到方法Quote稍作修改即可:

func (db *mssql) Quote(name string) string {
	fmt.Println("Quote -> ", db.URI().DbName) // DM
	if  db.URI().DbName == "DM" { // 如果是達夢數據庫不添加雙引號
		return name
	}
	return "\"" + name + "\""
}

這樣,是在拼接SQL之前修改了邏輯,二方式一是在拼接之後再去掉引號,方式二更方便一點。
注意:
查閱最新的文檔,發現最新的xorm.io/core v1.0.1版本已經支持使用engine.Dialect().SetQuotePolicy(core.QuotePolicyNone)來設置引號策略。如果你使用的是該版本,直接設置即可。
Dialect結構體、QuotePolicy常量值
輸出結果:

[xorm] [info]  2020/06/08 17:14:18.061667 PING DATABASE odbc
[xorm] [info]  2020/06/08 17:14:19.315349 [SQL] select addressid, address1, address2, city, postalcode from person.address limit 5 offset 2
0       ADDRESSID=3     ADDRESS1=青山區青翠苑1號        ADDRESS2=       CITY=武漢市青山區       POSTALCODE=430080
1       ADDRESSID=4     ADDRESS1=武昌區武船新村115號    ADDRESS2=       CITY=武漢市武昌區       POSTALCODE=430063
2       ADDRESSID=5     ADDRESS1=漢陽大道熊家灣15號     ADDRESS2=       CITY=武漢市漢陽區       POSTALCODE=430050
3       ADDRESSID=6     ADDRESS1=洪山區保利花園50-1-304 ADDRESS2=       CITY=武漢市洪山區       POSTALCODE=430073
4       ADDRESSID=7     ADDRESS1=洪山區保利花園51-1-702 ADDRESS2=       CITY=武漢市洪山區       POSTALCODE=430073
*******************************
[xorm] [info]  2020/06/08 17:14:19.324291 [SQL] SELECT TOP 5 addressid, address1, address2, city, postalcode FROM person.address
0={1 洪山區369號金地太陽城56-1-202  武漢市洪山區 430073}
1={2 洪山區369號金地太陽城57-2-302  武漢市洪山區 430073}
2={3 青山區青翠苑1號  武漢市青山區 430080}
3={4 武昌區武船新村115號  武漢市武昌區 430063}
4={5 漢陽大道熊家灣15號  武漢市漢陽區 430050}

參考文檔

xorm的操作指南

xorm的pkg文檔

go語言中文文檔

本站聲明:網站內容來源於博客園,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

台北網頁設計公司這麼多該如何選擇?

※智慧手機時代的來臨,RWD網頁設計為架站首選

※評比南投搬家公司費用收費行情懶人包大公開

※回頭車貨運收費標準

網頁設計最專業,超強功能平台可客製化

※別再煩惱如何寫文案,掌握八大原則!

面試官:換人!他連 TCP 這幾個參數都不懂

每日一句英語學習,每天進步一點點:

前言

TCP 性能的提升不僅考察 TCP 的理論知識,還考察了對於操作系統提供的內核參數的理解與應用。

TCP 協議是由操作系統實現,所以操作系統提供了不少調節 TCP 的參數。

Linux TCP 參數

如何正確有效的使用這些參數,來提高 TCP 性能是一個不那麼簡單事情。我們需要針對 TCP 每個階段的問題來對症下藥,而不是病急亂投醫。

接下來,將以三個角度來闡述提升 TCP 的策略,分別是:

  • TCP 三次握手的性能提升;
  • TCP 四次揮手的性能提升;
  • TCP 數據傳輸的性能提升;

本節提綱

正文

01 TCP 三次握手的性能提升

TCP 是面向連接的、可靠的、雙向傳輸的傳輸層通信協議,所以在傳輸數據之前需要經過三次握手才能建立連接。

三次握手與數據傳輸

那麼,三次握手的過程在一個 HTTP 請求的平均時間佔比 10% 以上,在網絡狀態不佳、高併發或者遭遇 SYN 攻擊等場景中,如果不能有效正確的調節三次握手中的參數,就會對性能產生很多的影響。

如何正確有效的使用這些參數,來提高 TCP 三次握手的性能,這就需要理解「三次握手的狀態變遷」,這樣當出現問題時,先用 netstat 命令查看是哪個握手階段出現了問題,再來對症下藥,而不是病急亂投醫。

TCP 三次握手的狀態變遷

客戶端和服務端都可以針對三次握手優化性能。主動發起連接的客戶端優化相對簡單些,而服務端需要監聽端口,屬於被動連接方,其間保持許多的中間狀態,優化方法相對複雜一些。

所以,客戶端(主動發起連接方)和服務端(被動連接方)優化的方式是不同的,接下來分別針對客戶端和服務端優化。

客戶端優化

三次握手建立連接的首要目的是「同步序列號」。

只有同步了序列號才有可靠傳輸,TCP 許多特性都依賴於序列號實現,比如流量控制、丟包重傳等,這也是三次握手中的報文稱為 SYN 的原因,SYN 的全稱就叫 Synchronize Sequence Numbers(同步序列號)。

TCP 頭部

SYN_SENT 狀態的優化

客戶端作為主動發起連接方,首先它將發送 SYN 包,於是客戶端的連接就會處於 SYN_SENT 狀態。

客戶端在等待服務端回復的 ACK 報文,正常情況下,服務器會在幾毫秒內返回 SYN+ACK ,但如果客戶端長時間沒有收到 SYN+ACK 報文,則會重發 SYN 包,重發的次數由 tcp_syn_retries 參數控制,默認是 5 次:

通常,第一次超時重傳是在 1 秒后,第二次超時重傳是在 2 秒,第三次超時重傳是在 4 秒后,第四次超時重傳是在 8 秒后,第五次是在超時重傳 16 秒后。沒錯,每次超時的時間是上一次的 2 倍

當第五次超時重傳后,會繼續等待 32 秒,如果仍然服務端沒有回應 ACK,客戶端就會終止三次握手。

所以,總耗時是 1+2+4+8+16+32=63 秒,大約 1 分鐘左右。

SYN 超時重傳

你可以根據網絡的穩定性和目標服務器的繁忙程度修改 SYN 的重傳次數,調整客戶端的三次握手時間上限。比如內網中通訊時,就可以適當調低重試次數,儘快把錯誤暴露給應用程序。

服務端優化

當服務端收到 SYN 包后,服務端會立馬回復 SYN+ACK 包,表明確認收到了客戶端的序列號,同時也把自己的序列號發給對方。

此時,服務端出現了新連接,狀態是 SYN_RCV。在這個狀態下,Linux 內核就會建立一個「半連接隊列」來維護「未完成」的握手信息,當半連接隊列溢出后,服務端就無法再建立新的連接。

半連接隊列與全連接隊列

SYN 攻擊,攻擊的是就是這個半連接隊列。

如何查看由於 SYN 半連接隊列已滿,而被丟棄連接的情況?

我們可以通過該 netstat -s 命令給出的統計結果中, 可以得到由於半連接隊列已滿,引發的失敗次數:

上面輸出的數值是累計值,表示共有多少個 TCP 連接因為半連接隊列溢出而被丟棄。隔幾秒執行幾次,如果有上升的趨勢,說明當前存在半連接隊列溢出的現象

如何調整 SYN 半連接隊列大小?

要想增大半連接隊列,不能只單純增大 tcp_max_syn_backlog 的值,還需一同增大 somaxconn 和 backlog,也就是增大 accept 隊列。否則,只單純增大 tcp_max_syn_backlog 是無效的。

增大 tcp_max_syn_backlog 和 somaxconn 的方法是修改 Linux 內核參數:

增大 backlog 的方式,每個 Web 服務都不同,比如 Nginx 增大 backlog 的方法如下:

最後,改變了如上這些參數后,要重啟 Nginx 服務,因為 SYN 半連接隊列和 accept 隊列都是在 listen() 初始化的。

如果 SYN 半連接隊列已滿,只能丟棄連接嗎?

並不是這樣,開啟 syncookies 功能就可以在不使用 SYN 半連接隊列的情況下成功建立連接

syncookies 的工作原理:服務器根據當前狀態計算出一個值,放在己方發出的 SYN+ACK 報文中發出,當客戶端返回 ACK 報文時,取出該值驗證,如果合法,就認為連接建立成功,如下圖所示。

開啟 syncookies 功能

syncookies 參數主要有以下三個值:

  • 0 值,表示關閉該功能;
  • 1 值,表示僅當 SYN 半連接隊列放不下時,再啟用它;
  • 2 值,表示無條件開啟功能;

那麼在應對 SYN 攻擊時,只需要設置為 1 即可:

SYN_RCV 狀態的優化

當客戶端接收到服務器發來的 SYN+ACK 報文後,就會回復 ACK 給服務器,同時客戶端連接狀態從 SYN_SENT 轉換為 ESTABLISHED,表示連接建立成功。

服務器端連接成功建立的時間還要再往後,等到服務端收到客戶端的 ACK 后,服務端的連接狀態才變為 ESTABLISHED。

如果服務器沒有收到 ACK,就會重發 SYN+ACK 報文,同時一直處於 SYN_RCV 狀態。

當網絡繁忙、不穩定時,報文丟失就會變嚴重,此時應該調大重發次數。反之則可以調小重發次數。修改重發次數的方法是,調整 tcp_synack_retries 參數

tcp_synack_retries 的默認重試次數是 5 次,與客戶端重傳 SYN 類似,它的重傳會經歷 1、2、4、8、16 秒,最後一次重傳後會繼續等待 32 秒,如果服務端仍然沒有收到 ACK,才會關閉連接,故共需要等待 63 秒。

服務器收到 ACK 后連接建立成功,此時,內核會把連接從半連接隊列移除,然後創建新的完全的連接,並將其添加到 accept 隊列,等待進程調用 accept 函數時把連接取出來。

如果進程不能及時地調用 accept 函數,就會造成 accept 隊列(也稱全連接隊列)溢出,最終導致建立好的 TCP 連接被丟棄。

accept 隊列溢出

accept 隊列已滿,只能丟棄連接嗎?

丟棄連接只是 Linux 的默認行為,我們還可以選擇向客戶端發送 RST 複位報文,告訴客戶端連接已經建立失敗。打開這一功能需要將 tcp_abort_on_overflow 參數設置為 1。

tcp_abort_on_overflow 共有兩個值分別是 0 和 1,其分別表示:

  • 0 :如果 accept 隊列滿了,那麼 server 扔掉 client 發過來的 ack ;
  • 1 :如果 accept 隊列滿了,server 發送一個 RST 包給 client,表示廢掉這個握手過程和這個連接;

如果要想知道客戶端連接不上服務端,是不是服務端 TCP 全連接隊列滿的原因,那麼可以把 tcp_abort_on_overflow 設置為 1,這時如果在客戶端異常中可以看到很多 connection reset by peer 的錯誤,那麼就可以證明是由於服務端 TCP 全連接隊列溢出的問題。

通常情況下,應當把 tcp_abort_on_overflow 設置為 0,因為這樣更有利於應對突發流量。

舉個例子,當 accept 隊列滿導致服務器丟掉了 ACK,與此同時,客戶端的連接狀態卻是 ESTABLISHED,客戶端進程就在建立好的連接上發送請求。只要服務器沒有為請求回復 ACK,客戶端的請求就會被多次「重發」。如果服務器上的進程只是短暫的繁忙造成 accept 隊列滿,那麼當 accept 隊列有空位時,再次接收到的請求報文由於含有 ACK,仍然會觸發服務器端成功建立連接。

tcp_abort_on_overflow 為 0 可以應對突發流量

所以,tcp_abort_on_overflow 設為 0 可以提高連接建立的成功率,只有你非常肯定 TCP 全連接隊列會長期溢出時,才能設置為 1 以儘快通知客戶端。

如何調整 accept 隊列的長度呢?

accept 隊列的長度取決於 somaxconn 和 backlog 之間的最小值,也就是 min(somaxconn, backlog),其中:

  • somaxconn 是 Linux 內核的參數,默認值是 128,可以通過 net.core.somaxconn 來設置其值;
  • backlog 是 listen(int sockfd, int backlog) 函數中的 backlog 大小;

Tomcat、Nginx、Apache 常見的 Web 服務的 backlog 默認值都是 511。

如何查看服務端進程 accept 隊列的長度?

可以通過 ss -ltn 命令查看:

  • Recv-Q:當前 accept 隊列的大小,也就是當前已完成三次握手並等待服務端 accept() 的 TCP 連接;
  • Send-Q:accept 隊列最大長度,上面的輸出結果說明監聽 8088 端口的 TCP 服務,accept 隊列的最大長度為 128;

如何查看由於 accept 連接隊列已滿,而被丟棄的連接?

當超過了 accept 連接隊列,服務端則會丟掉後續進來的 TCP 連接,丟掉的 TCP 連接的個數會被統計起來,我們可以使用 netstat -s 命令來查看:

上面看到的 41150 times ,表示 accept 隊列溢出的次數,注意這個是累計值。可以隔幾秒鐘執行下,如果這個数字一直在增加的話,說明 accept 連接隊列偶爾滿了。

如果持續不斷地有連接因為 accept 隊列溢出被丟棄,就應該調大 backlog 以及 somaxconn 參數。

如何繞過三次握手?

以上我們只是在對三次握手的過程進行優化,接下來我們看看如何繞過三次握手發送數據。

三次握手建立連接造成的後果就是,HTTP 請求必須在一個 RTT(從客戶端到服務器一個往返的時間)后才能發送。

常規 HTTP 請求

在 Linux 3.7 內核版本之後,提供了 TCP Fast Open 功能,這個功能可以減少 TCP 連接建立的時延。

接下來說說,TCP Fast Open 功能的工作方式。

開啟 TCP Fast Open 功能

在客戶端首次建立連接時的過程:

  1. 客戶端發送 SYN 報文,該報文包含 Fast Open 選項,且該選項的 Cookie 為空,這表明客戶端請求 Fast Open Cookie;
  2. 支持 TCP Fast Open 的服務器生成 Cookie,並將其置於 SYN-ACK 數據包中的 Fast Open 選項以發回客戶端;
  3. 客戶端收到 SYN-ACK 后,本地緩存 Fast Open 選項中的 Cookie。

所以,第一次發起 HTTP GET 請求的時候,還是需要正常的三次握手流程。

之後,如果客戶端再次向服務器建立連接時的過程:

  1. 客戶端發送 SYN 報文,該報文包含「數據」(對於非 TFO 的普通 TCP 握手過程,SYN 報文中不包含「數據」)以及此前記錄的 Cookie;
  2. 支持 TCP Fast Open 的服務器會對收到 Cookie 進行校驗:如果 Cookie 有效,服務器將在 SYN-ACK 報文中對 SYN 和「數據」進行確認,服務器隨後將「數據」遞送至相應的應用程序;如果 Cookie 無效,服務器將丟棄 SYN 報文中包含的「數據」,且其隨後發出的 SYN-ACK 報文將只確認 SYN 的對應序列號;
  3. 如果服務器接受了 SYN 報文中的「數據」,服務器可在握手完成之前發送「數據」,這就減少了握手帶來的 1 個 RTT 的時間消耗
  4. 客戶端將發送 ACK 確認服務器發回的 SYN 以及「數據」,但如果客戶端在初始的 SYN 報文中發送的「數據」沒有被確認,則客戶端將重新發送「數據」;
  5. 此後的 TCP 連接的數據傳輸過程和非 TFO 的正常情況一致。

所以,之後發起 HTTP GET 請求的時候,可以繞過三次握手,這就減少了握手帶來的 1 個 RTT 的時間消耗。

開啟了 TFO 功能,cookie 的值是存放到 TCP option 字段里的:

TCP option 字段 – TFO

注:客戶端在請求並存儲了 Fast Open Cookie 之後,可以不斷重複 TCP Fast Open 直至服務器認為 Cookie 無效(通常為過期)。

Linux 下怎麼打開 TCP Fast Open 功能呢?

在 Linux 系統中,可以通過設置 tcp_fastopn 內核參數,來打開 Fast Open 功能

tcp_fastopn 各個值的意義:

  • 0 關閉
  • 1 作為客戶端使用 Fast Open 功能
  • 2 作為服務端使用 Fast Open 功能
  • 3 無論作為客戶端還是服務器,都可以使用 Fast Open 功能

TCP Fast Open 功能需要客戶端和服務端同時支持,才有效果。

小結

本小結主要介紹了關於優化 TCP 三次握手的幾個 TCP 參數。

三次握手優化策略

客戶端的優化

當客戶端發起 SYN 包時,可以通過 tcp_syn_retries 控制其重傳的次數。

服務端的優化

當服務端 SYN 半連接隊列溢出后,會導致後續連接被丟棄,可以通過 netstat -s 觀察半連接隊列溢出的情況,如果 SYN 半連接隊列溢出情況比較嚴重,可以通過 tcp_max_syn_backlog、somaxconn、backlog 參數來調整 SYN 半連接隊列的大小。

服務端回復 SYN+ACK 的重傳次數由 tcp_synack_retries 參數控制。如果遭受 SYN 攻擊,應把 tcp_syncookies 參數設置為 1,表示僅在 SYN 隊列滿后開啟 syncookie 功能,可以保證正常的連接成功建立。

服務端收到客戶端返回的 ACK,會把連接移入 accpet 隊列,等待進行調用 accpet() 函數取出連接。

可以通過 ss -lnt 查看服務端進程的 accept 隊列長度,如果 accept 隊列溢出,系統默認丟棄 ACK,如果可以把 tcp_abort_on_overflow 設置為 1 ,表示用 RST 通知客戶端連接建立失敗。

如果 accpet 隊列溢出嚴重,可以通過 listen 函數的 backlog 參數和 somaxconn 系統參數提高隊列大小,accept 隊列長度取決於 min(backlog, somaxconn)。

繞過三次握手

TCP Fast Open 功能可以繞過三次握手,使得 HTTP 請求減少了 1 個 RTT 的時間,Linux 下可以通過 tcp_fastopen 開啟該功能,同時必須保證服務端和客戶端同時支持。

02 TCP 四次揮手的性能提升

接下來,我們一起看看針對 TCP 四次揮手關不連接時,如何優化性能。

在開始之前,我們得先了解四次揮手狀態變遷的過程。

客戶端和服務端雙方都可以主動斷開連接,通常先關閉連接的一方稱為主動方,后關閉連接的一方稱為被動方。

客戶端主動關閉

可以看到,四次揮手過程只涉及了兩種報文,分別是 FIN 和 ACK

  • FIN 就是結束連接的意思,誰發出 FIN 報文,就表示它將不會再發送任何數據,關閉這一方向上的傳輸通道;
  • ACK 就是確認的意思,用來通知對方:你方的發送通道已經關閉;

四次揮手的過程:

  • 當主動方關閉連接時,會發送 FIN 報文,此時發送方的 TCP 連接將從 ESTABLISHED 變成 FIN_WAIT1。
  • 當被動方收到 FIN 報文後,內核會自動回復 ACK 報文,連接狀態將從 ESTABLISHED 變成 CLOSE_WAIT,表示被動方在等待進程調用 close 函數關閉連接。
  • 當主動方收到這個 ACK 后,連接狀態由 FIN_WAIT1 變為 FIN_WAIT2,也就是表示主動方的發送通道就關閉了
  • 當被動方進入 CLOSE_WAIT 時,被動方還會繼續處理數據,等到進程的 read 函數返回 0 后,應用程序就會調用 close 函數,進而觸發內核發送 FIN 報文,此時被動方的連接狀態變為 LAST_ACK。
  • 當主動方收到這個 FIN 報文後,內核會回復 ACK 報文給被動方,同時主動方的連接狀態由 FIN_WAIT2 變為 TIME_WAIT,在 Linux 系統下大約等待 1 分鐘后,TIME_WAIT 狀態的連接才會徹底關閉
  • 當被動方收到最後的 ACK 報文後,被動方的連接就會關閉

你可以看到,每個方向都需要一個 FIN 和一個 ACK,因此通常被稱為四次揮手

這裏一點需要注意是:主動關閉連接的,才有 TIME_WAIT 狀態。

主動關閉方和被動關閉方優化的思路也不同,接下來分別說說如何優化他們。

主動方的優化

關閉的連接的方式通常有兩種,分別是 RST 報文關閉和 FIN 報文關閉。

如果進程異常退出了,內核就會發送 RST 報文來關閉,它可以不走四次揮手流程,是一個暴力關閉連接的方式。

安全關閉連接的方式必須通過四次揮手,它由進程調用 closeshutdown 函數發起 FIN 報文(shutdown 參數須傳入 SHUT_WR 或者 SHUT_RDWR 才會發送 FIN)。

調用 close 函數 和 shutdown 函數有什麼區別?

調用了 close 函數意味着完全斷開連接,完全斷開不僅指無法傳輸數據,而且也不能發送數據。 此時,調用了 close 函數的一方的連接叫做「孤兒連接」,如果你用 netstat -p 命令,會發現連接對應的進程名為空。

使用 close 函數關閉連接是不優雅的。於是,就出現了一種優雅關閉連接的 shutdown 函數,它可以控制只關閉一個方向的連接

第二個參數決定斷開連接的方式,主要有以下三種方式:

  • SHUT_RD(0):關閉連接的「讀」這個方向,如果接收緩衝區有已接收的數據,則將會被丟棄,並且後續再收到新的數據,會對數據進行 ACK,然後悄悄地丟棄。也就是說,對端還是會接收到 ACK,在這種情況下根本不知道數據已經被丟棄了。
  • SHUT_WR(1):關閉連接的「寫」這個方向,這就是常被稱為「半關閉」的連接。如果發送緩衝區還有未發送的數據,將被立即發送出去,併發送一個 FIN 報文給對端。
  • SHUT_RDWR(2):相當於 SHUT_RD 和 SHUT_WR 操作各一次,關閉套接字的讀和寫兩個方向

close 和 shutdown 函數都可以關閉連接,但這兩種方式關閉的連接,不只功能上有差異,控制它們的 Linux 參數也不相同。

FIN_WAIT1 狀態的優化

主動方發送 FIN 報文後,連接就處於 FIN_WAIT1 狀態,正常情況下,如果能及時收到被動方的 ACK,則會很快變為 FIN_WAIT2 狀態。

但是當遲遲收不到對方返回的 ACK 時,連接就會一直處於 FIN_WAIT1 狀態。此時,內核會定時重發 FIN 報文,其中重發次數由 tcp_orphan_retries 參數控制(注意,orphan 雖然是孤兒的意思,該參數卻不只對孤兒連接有效,事實上,它對所有 FIN_WAIT1 狀態下的連接都有效),默認值是 0。

你可能會好奇,這 0 表示幾次?實際上當為 0 時,特指 8 次,從下面的內核源碼可知:

如果 FIN_WAIT1 狀態連接很多,我們就需要考慮降低 tcp_orphan_retries 的值,當重傳次數超過 tcp_orphan_retries 時,連接就會直接關閉掉。

對於普遍正常情況時,調低 tcp_orphan_retries 就已經可以了。如果遇到惡意攻擊,FIN 報文根本無法發送出去,這由 TCP 兩個特性導致的:

  • 首先,TCP 必須保證報文是有序發送的,FIN 報文也不例外,當發送緩衝區還有數據沒有發送時,FIN 報文也不能提前發送。
  • 其次,TCP 有流量控制功能,當接收方接收窗口為 0 時,發送方就不能再發送數據。所以,當攻擊者下載大文件時,就可以通過接收窗口設為 0 ,這就會使得 FIN 報文都無法發送出去,那麼連接會一直處於 FIN_WAIT1 狀態。

解決這種問題的方法,是調整 tcp_max_orphans 參數,它定義了「孤兒連接」的最大數量

當進程調用了 close 函數關閉連接,此時連接就會是「孤兒連接」,因為它無法在發送和接收數據。Linux 系統為了防止孤兒連接過多,導致系統資源長時間被佔用,就提供了 tcp_max_orphans 參數。如果孤兒連接數量大於它,新增的孤兒連接將不再走四次揮手,而是直接發送 RST 複位報文強制關閉。

FIN_WAIT2 狀態的優化

當主動方收到 ACK 報文後,會處於 FIN_WAIT2 狀態,就表示主動方的發送通道已經關閉,接下來將等待對方發送 FIN 報文,關閉對方的發送通道。

這時,如果連接是用 shutdown 函數關閉的,連接可以一直處於 FIN_WAIT2 狀態,因為它可能還可以發送或接收數據。但對於 close 函數關閉的孤兒連接,由於無法在發送和接收數據,所以這個狀態不可以持續太久,而 tcp_fin_timeout 控制了這個狀態下連接的持續時長,默認值是 60 秒:

它意味着對於孤兒連接(調用 close 關閉的連接),如果在 60 秒后還沒有收到 FIN 報文,連接就會直接關閉。

這個 60 秒不是隨便決定的,它與 TIME_WAIT 狀態持續的時間是相同的,後面我們在來說說為什麼是 60 秒。

TIME_WAIT 狀態的優化

TIME_WAIT 是主動方四次揮手的最後一個狀態,也是最常遇見的狀態。

當收到被動方發來的 FIN 報文後,主動方會立刻回復 ACK,表示確認對方的發送通道已經關閉,接着就處於 TIME_WAIT 狀態。在 Linux 系統,TIME_WAIT 狀態會持續 60 秒后才會進入關閉狀態。

TIME_WAIT 狀態的連接,在主動方看來確實快已經關閉了。然後,被動方沒有收到 ACK 報文前,還是處於 LAST_ACK 狀態。如果這個 ACK 報文沒有到達被動方,被動方就會重發 FIN 報文。重發次數仍然由前面介紹過的 tcp_orphan_retries 參數控制。

TIME-WAIT 的狀態尤其重要,主要是兩個原因:

  • 防止具有相同「四元組」的「舊」數據包被收到;
  • 保證「被動關閉連接」的一方能被正確的關閉,即保證最後的 ACK 能讓被動關閉方接收,從而幫助其正常關閉;

原因一:防止舊連接的數據包

TIME-WAIT 的一個作用是防止收到歷史數據,從而導致數據錯亂的問題。

假設 TIME-WAIT 沒有等待時間或時間過短,被延遲的數據包抵達後會發生什麼呢?

接收到歷史數據的異常

  • 如上圖黃色框框服務端在關閉連接之前發送的 SEQ = 301 報文,被網絡延遲了。
  • 這時有相同端口的 TCP 連接被複用后,被延遲的 SEQ = 301 抵達了客戶端,那麼客戶端是有可能正常接收這個過期的報文,這就會產生數據錯亂等嚴重的問題。

所以,TCP 就設計出了這麼一個機制,經過 2MSL 這個時間,足以讓兩個方向上的數據包都被丟棄,使得原來連接的數據包在網絡中都自然消失,再出現的數據包一定都是新建立連接所產生的。

原因二:保證連接正確關閉

TIME-WAIT 的另外一個作用是等待足夠的時間以確保最後的 ACK 能讓被動關閉方接收,從而幫助其正常關閉。

假設 TIME-WAIT 沒有等待時間或時間過短,斷開連接會造成什麼問題呢?

沒有確保正常斷開的異常

  • 如上圖紅色框框客戶端四次揮手的最後一個 ACK 報文如果在網絡中被丟失了,此時如果客戶端 TIME-WAIT 過短或沒有,則就直接進入了 CLOSE 狀態了,那麼服務端則會一直處在 LASE-ACK 狀態。
  • 當客戶端發起建立連接的 SYN 請求報文後,服務端會發送 RST 報文給客戶端,連接建立的過程就會被終止。

我們再回過頭來看看,為什麼 TIME_WAIT 狀態要保持 60 秒呢?這與孤兒連接 FIN_WAIT2 狀態默認保留 60 秒的原理是一樣的,因為這兩個狀態都需要保持 2MSL 時長。MSL 全稱是 Maximum Segment Lifetime,它定義了一個報文在網絡中的最長生存時間(報文每經過一次路由器的轉發,IP 頭部的 TTL 字段就會減 1,減到 0 時報文就被丟棄,這就限制了報文的最長存活時間)。

為什麼是 2 MSL 的時長呢?這其實是相當於至少允許報文丟失一次。比如,若 ACK 在一個 MSL 內丟失,這樣被動方重發的 FIN 會在第 2 個 MSL 內到達,TIME_WAIT 狀態的連接可以應對。

為什麼不是 4 或者 8 MSL 的時長呢?你可以想象一個丟包率達到百分之一的糟糕網絡,連續兩次丟包的概率只有萬分之一,這個概率實在是太小了,忽略它比解決它更具性價比。

因此,TIME_WAIT 和 FIN_WAIT2 狀態的最大時長都是 2 MSL,由於在 Linux 系統中,MSL 的值固定為 30 秒,所以它們都是 60 秒。

雖然 TIME_WAIT 狀態有存在的必要,但它畢竟會消耗系統資源。如果發起連接一方的 TIME_WAIT 狀態過多,佔滿了所有端口資源,則會導致無法創建新連接。

  • 客戶端受端口資源限制:如果客戶端 TIME_WAIT 過多,就會導致端口資源被佔用,因為端口就65536個,被佔滿就會導致無法創建新的連接;
  • 服務端受系統資源限制:由於一個 四元組表示TCP連接,理論上服務端可以建立很多連接,服務端確實只監聽一個端口 但是會把連接扔給處理線程,所以理論上監聽的端口可以繼續監聽。但是線程池處理不了那麼多一直不斷的連接了。所以當服務端出現大量 TIME_WAIT 時,系統資源被佔滿時,會導致處理不過來新的連接;

另外,Linux 提供了 tcp_max_tw_buckets 參數,當 TIME_WAIT 的連接數量超過該參數時,新關閉的連接就不再經歷 TIME_WAIT 而直接關閉:

當服務器的併發連接增多時,相應地,同時處於 TIME_WAIT 狀態的連接數量也會變多,此時就應當調大 tcp_max_tw_buckets 參數,減少不同連接間數據錯亂的概率。

tcp_max_tw_buckets 也不是越大越好,畢竟內存和端口都是有限的。

有一種方式可以在建立新連接時,復用處於 TIME_WAIT 狀態的連接,那就是打開 tcp_tw_reuse 參數。但是需要注意,該參數是只用於客戶端(建立連接的發起方),因為是在調用 connect() 時起作用的,而對於服務端(被動連接方)是沒有用的。

tcp_tw_reuse 從協議角度理解是安全可控的,可以復用處於 TIME_WAIT 的端口為新的連接所用。

什麼是協議角度理解的安全可控呢?主要有兩點:

  • 只適用於連接發起方,也就是 C/S 模型中的客戶端;
  • 對應的 TIME_WAIT 狀態的連接創建時間超過 1 秒才可以被複用。

使用這個選項,還有一個前提,需要打開對 TCP 時間戳的支持(對方也要打開 ):

由於引入了時間戳,它能帶來了些好處:

  • 我們在前面提到的 2MSL 問題就不復存在了,因為重複的數據包會因為時間戳過期被自然丟棄;
  • 同時,它還可以防止序列號繞回,也是因為重複的數據包會由於時間戳過期被自然丟棄;

時間戳是在 TCP 的選擇字段里定義的,開啟了時間戳功能,在 TCP 報文傳輸的時候會帶上發送報文的時間戳。

TCP option 字段 – 時間戳

我們來看看開啟了 tcp_tw_reuse 功能,如果四次揮手中的最後一次 ACK 在網絡中丟失了,會發生什麼?

四次揮手中的最後一次 ACK 在網絡中丟失

上圖的流程:

  • 四次揮手中的最後一次 ACK 在網絡中丟失了,服務端一直處於 LAST_ACK 狀態;
  • 客戶端由於開啟了 tcp_tw_reuse 功能,客戶再次發起新連接的時候,會復用超過 1 秒后的 time_wait 狀態的連接。但客戶端新發的 SYN 包會被忽略(由於時間戳),因為服務端比較了客戶端的上一個報文與 SYN 報文的時間戳,過期的報文就會被服務端丟棄
  • 服務端 FIN 報文遲遲沒有收到四次揮手的最後一次 ACK,於是超時重發了 FIN 報文給客戶端;
  • 處於 SYN_SENT 狀態的客戶,由於收到了 FIN 報文,則會回 RST 給服務端,於是服務端就離開了 LAST_ACK 狀態;
  • 最初的客戶端 SYN 報文超時重發了( 1 秒鐘后),此時就與服務端能正確的三次握手了。

所以大家都會說開啟了 tcp_tw_reuse,可以在復用了 time_wait 狀態的 1 秒過後成功建立連接,這 1 秒主要是花費在 SYN 包重傳。

另外,老版本的 Linux 還提供了 tcp_tw_recycle 參數,但是當開啟了它,就有兩個坑:

  • Linux 會加快客戶端和服務端 TIME_WAIT 狀態的時間,也就是它會使得 TIME_WAIT 狀態會小於 60 秒,很容易導致數據錯亂;
  • 另外,Linux 會丟棄所有來自遠端時間戳小於上次記錄的時間戳(由同一個遠端發送的)的任何數據包。就是說要使用該選項,則必須保證數據包的時間戳是單調遞增的。那麼,問題在於,此處的時間戳並不是我們通常意義上面的絕對時間,而是一個相對時間。很多情況下,我們是沒法保證時間戳單調遞增的,比如使用了 NAT、LVS 等情況;

所以,不建議設置為 1 ,在 Linux 4.12 版本后,Linux 內核直接取消了這一參數,建議關閉它:

另外,我們可以在程序中設置 socket 選項,來設置調用 close 關閉連接行為。

如果 l_onoff 為非 0, 且 l_linger 值為 0,那麼調用 close 后,會立該發送一個 RST 標誌給對端,該 TCP 連接將跳過四次揮手,也就跳過了 TIME_WAIT 狀態,直接關閉。

但這為跨越 TIME_WAIT 狀態提供了一個可能,不過是一個非常危險的行為,不值得提倡。

被動方的優化

當被動方收到 FIN 報文時,內核會自動回復 ACK,同時連接處於 CLOSE_WAIT 狀態,顧名思義,它表示等待應用進程調用 close 函數關閉連接。

內核沒有權利替代進程去關閉連接,因為如果主動方是通過 shutdown 關閉連接,那麼它就是想在半關閉連接上接收數據或發送數據。因此,Linux 並沒有限制 CLOSE_WAIT 狀態的持續時間。

當然,大多數應用程序並不使用 shutdown 函數關閉連接。所以,當你用 netstat 命令發現大量 CLOSE_WAIT 狀態。就需要排查你的應用程序,因為可能因為應用程序出現了 Bug,read 函數返回 0 時,沒有調用 close 函數。

處於 CLOSE_WAIT 狀態時,調用了 close 函數,內核就會發出 FIN 報文關閉發送通道,同時連接進入 LAST_ACK 狀態,等待主動方返回 ACK 來確認連接關閉。

如果遲遲收不到這個 ACK,內核就會重發 FIN 報文,重發次數仍然由 tcp_orphan_retries 參數控制,這與主動方重發 FIN 報文的優化策略一致。

還有一點我們需要注意的,如果被動方迅速調用 close 函數,那麼被動方的 ACK 和 FIN 有可能在一個報文中發送,這樣看起來,四次揮手會變成三次揮手,這隻是一種特殊情況,不用在意。

如果連接雙方同時關閉連接,會怎麼樣?

由於 TCP 是雙全工的協議,所以是會出現兩方同時關閉連接的現象,也就是同時發送了 FIN 報文。

此時,上面介紹的優化策略仍然適用。兩方發送 FIN 報文時,都認為自己是主動方,所以都進入了 FIN_WAIT1 狀態,FIN 報文的重發次數仍由 tcp_orphan_retries 參數控制。

同時關閉

接下來,雙方在等待 ACK 報文的過程中,都等來了 FIN 報文。這是一種新情況,所以連接會進入一種叫做 CLOSING 的新狀態,它替代了 FIN_WAIT2 狀態。接着,雙方內核回復 ACK 確認對方發送通道的關閉后,進入 TIME_WAIT 狀態,等待 2MSL 的時間后,連接自動關閉。

小結

針對 TCP 四次揮手的優化,我們需要根據主動方和被動方四次揮手狀態變化來調整系統 TCP 內核參數。

四次揮手的優化策略

主動方的優化

主動發起 FIN 報文斷開連接的一方,如果遲遲沒收到對方的 ACK 回復,則會重傳 FIN 報文,重傳的次數由 tcp_orphan_retries 參數決定。

當主動方收到 ACK 報文後,連接就進入 FIN_WAIT2 狀態,根據關閉的方式不同,優化的方式也不同:

  • 如果這是 close 函數關閉的連接,那麼它就是孤兒連接。如果 tcp_fin_timeout 秒內沒有收到對方的 FIN 報文,連接就直接關閉。同時,為了應對孤兒連接佔用太多的資源,tcp_max_orphans 定義了最大孤兒連接的數量,超過時連接就會直接釋放。
  • 反之是 shutdown 函數關閉的連接,則不受此參數限制;

當主動方接收到 FIN 報文,並返回 ACK 后,主動方的連接進入 TIME_WAIT 狀態。這一狀態會持續 1 分鐘,為了防止 TIME_WAIT 狀態佔用太多的資源,tcp_max_tw_buckets 定義了最大數量,超過時連接也會直接釋放。

當 TIME_WAIT 狀態過多時,還可以通過設置 tcp_tw_reusetcp_timestamps 為 1 ,將 TIME_WAIT 狀態的端口復用於作為客戶端的新連接,注意該參數只適用於客戶端。

被動方的優化

被動關閉的連接方應對非常簡單,它在回復 ACK 后就進入了 CLOSE_WAIT 狀態,等待進程調用 close 函數關閉連接。因此,出現大量 CLOSE_WAIT 狀態的連接時,應當從應用程序中找問題。

當被動方發送 FIN 報文後,連接就進入 LAST_ACK 狀態,在未等到 ACK 時,會在 tcp_orphan_retries 參數的控制下重發 FIN 報文。

03 TCP 傳輸數據的性能提升

在前面介紹的是三次握手和四次揮手的優化策略,接下來主要介紹的是 TCP 傳輸數據時的優化策略。

TCP 連接是由內核維護的,內核會為每個連接建立內存緩衝區:

  • 如果連接的內存配置過小,就無法充分使用網絡帶寬,TCP 傳輸效率就會降低;
  • 如果連接的內存配置過大,很容易把服務器資源耗盡,這樣就會導致新連接無法建立;

因此,我們必須理解 Linux 下 TCP 內存的用途,才能正確地配置內存大小。

滑動窗口是如何影響傳輸速度的?

TCP 會保證每一個報文都能夠抵達對方,它的機制是這樣:報文發出去后,必須接收到對方返回的確認報文 ACK,如果遲遲未收到,就會超時重發該報文,直到收到對方的 ACK 為止。

所以,TCP 報文發出去后,並不會立馬從內存中刪除,因為重傳時還需要用到它。

由於 TCP 是內核維護的,所以報文存放在內核緩衝區。如果連接非常多,我們可以通過 free 命令觀察到 buff/cache 內存是會增大。

如果 TCP 是每發送一個數據,都要進行一次確認應答。當上一個數據包收到了應答了, 再發送下一個。這個模式就有點像我和你面對面聊天,你一句我一句,但這種方式的缺點是效率比較低的。

按數據包進行確認應答

所以,這樣的傳輸方式有一個缺點:數據包的往返時間越長,通信的效率就越低

要解決這一問題不難,并行批量發送報文,再批量確認報文即刻。

并行處理

然而,這引出了另一個問題,發送方可以隨心所欲的發送報文嗎?當然這不現實,我們還得考慮接收方的處理能力。

當接收方硬件不如發送方,或者系統繁忙、資源緊張時,是無法瞬間處理這麼多報文的。於是,這些報文只能被丟掉,使得網絡效率非常低。

為了解決這種現象發生,TCP 提供一種機制可以讓「發送方」根據「接收方」的實際接收能力控制發送的數據量,這就是滑動窗口的由來。

接收方根據它的緩衝區,可以計算出後續能夠接收多少字節的報文,這個数字叫做接收窗口。當內核接收到報文時,必須用緩衝區存放它們,這樣剩餘緩衝區空間變小,接收窗口也就變小了;當進程調用 read 函數后,數據被讀入了用戶空間,內核緩衝區就被清空,這意味着主機可以接收更多的報文,接收窗口就會變大。

因此,接收窗口並不是恆定不變的,接收方會把當前可接收的大小放在 TCP 報文頭部中的窗口字段,這樣就可以起到窗口大小通知的作用。

發送方的窗口等價於接收方的窗口嗎?如果不考慮擁塞控制,發送方的窗口大小「約等於」接收方的窗口大小,因為窗口通知報文在網絡傳輸是存在時延的,所以是約等於的關係。

TCP 頭部

從上圖中可以看到,窗口字段只有 2 個字節,因此它最多能表達 65535 字節大小的窗口,也就是 64KB 大小。

這個窗口大小最大值,在當今高速網絡下,很明顯是不夠用的。所以後續有了擴充窗口的方法:在 TCP 選項(option)字段定義了窗口擴大因子,用於擴大 TCP 通告窗口,其值大小是 2^14,這樣就使 TCP 的窗口大小從 16 位擴大為 30 位(2^16 * 2^ 14 = 2^30),所以此時窗口的最大值可以達到 1GB。

TCP option 選項 – 窗口擴展

Linux 中打開這一功能,需要把 tcp_window_scaling 配置設為 1(默認打開):

要使用窗口擴大選項,通訊雙方必須在各自的 SYN 報文中發送這個選項:

  • 主動建立連接的一方在 SYN 報文中發送這個選項;
  • 而被動建立連接的一方只有在收到帶窗口擴大選項的 SYN 報文之後才能發送這個選項。

這樣看來,只要進程能及時地調用 read 函數讀取數據,並且接收緩衝區配置得足夠大,那麼接收窗口就可以無限地放大,發送方也就無限地提升發送速度。

這是不可能的,因為網絡的傳輸能力是有限的,當發送方依據發送窗口,發送超過網絡處理能力的報文時,路由器會直接丟棄這些報文。因此,緩衝區的內存並不是越大越好。

如果確定最大傳輸速度?

在前面我們知道了 TCP 的傳輸速度,受制於發送窗口與接收窗口,以及網絡設備傳輸能力。其中,窗口大小由內核緩衝區大小決定。如果緩衝區與網絡傳輸能力匹配,那麼緩衝區的利用率就達到了最大化。

問題來了,如何計算網絡的傳輸能力呢?

相信大家都知道網絡是有「帶寬」限制的,帶寬描述的是網絡傳輸能力,它與內核緩衝區的計量單位不同:

  • 帶寬是單位時間內的流量,表達是「速度」,比如常見的帶寬 100 MB/s;
  • 緩衝區單位是字節,當網絡速度乘以時間才能得到字節數;

這裏需要說一個概念,就是帶寬時延積,它決定網絡中飛行報文的大小,它的計算方式:

比如最大帶寬是 100 MB/s,網絡時延(RTT)是 10ms 時,意味着客戶端到服務端的網絡一共可以存放 100MB/s * 0.01s = 1MB 的字節。

這個 1MB 是帶寬和時延的乘積,所以它就叫「帶寬時延積」(縮寫為 BDP,Bandwidth Delay Product)。同時,這 1MB 也表示「飛行中」的 TCP 報文大小,它們就在網絡線路、路由器等網絡設備上。如果飛行報文超過了 1 MB,就會導致網絡過載,容易丟包。

由於發送緩衝區大小決定了發送窗口的上限,而發送窗口又決定了「已發送未確認」的飛行報文的上限。因此,發送緩衝區不能超過「帶寬時延積」。

發送緩衝區與帶寬時延積的關係:

  • 如果發送緩衝區「超過」帶寬時延積,超出的部分就沒辦法有效的網絡傳輸,同時導致網絡過載,容易丟包;
  • 如果發送緩衝區「小於」帶寬時延積,就不能很好的發揮出網絡的傳輸效率。

所以,發送緩衝區的大小最好是往帶寬時延積靠近。

怎樣調整緩衝區大小?

在 Linux 中發送緩衝區和接收緩衝都是可以用參數調節的。設置完后,Linux 會根據你設置的緩衝區進行動態調節

調節發送緩衝區範圍

先來看看發送緩衝區,它的範圍通過 tcp_wmem 參數配置;

上面三個数字單位都是字節,它們分別表示:

  • 第一個數值是動態範圍的最小值,4096 byte = 4K;
  • 第二個數值是初始默認值,87380 byte ≈ 86K;
  • 第三個數值是動態範圍的最大值,4194304 byte = 4096K(4M);

發送緩衝區是自行調節的,當發送方發送的數據被確認后,並且沒有新的數據要發送,就會把發送緩衝區的內存釋放掉。

調節接收緩衝區範圍

而接收緩衝區的調整就比較複雜一些,先來看看設置接收緩衝區範圍的 tcp_rmem 參數:

上面三個数字單位都是字節,它們分別表示:

  • 第一個數值是動態範圍的最小值,表示即使在內存壓力下也可以保證的最小接收緩衝區大小,4096 byte = 4K;
  • 第二個數值是初始默認值,87380 byte ≈ 86K;
  • 第三個數值是動態範圍的最大值,6291456 byte = 6144K(6M);

接收緩衝區可以根據系統空閑內存的大小來調節接收窗口:

  • 如果系統的空閑內存很多,就可以自動把緩衝區增大一些,這樣傳給對方的接收窗口也會變大,因而提升發送方發送的傳輸數據數量;
  • 反正,如果系統的內存很緊張,就會減少緩衝區,這雖然會降低傳輸效率,可以保證更多的併發連接正常工作;

發送緩衝區的調節功能是自動開啟的,而接收緩衝區則需要配置 tcp_moderate_rcvbuf 為 1 來開啟調節功能

調節 TCP 內存範圍

接收緩衝區調節時,怎麼知道當前內存是否緊張或充分呢?這是通過 tcp_mem 配置完成的:

上面三個数字單位不是字節,而是「頁面大小」,1 頁表示 4KB,它們分別表示:

  • 當 TCP 內存小於第 1 個值時,不需要進行自動調節;
  • 在第 1 和第 2 個值之間時,內核開始調節接收緩衝區的大小;
  • 大於第 3 個值時,內核不再為 TCP 分配新內存,此時新連接是無法建立的;

一般情況下這些值是在系統啟動時根據系統內存數量計算得到的。根據當前 tcp_mem 最大內存頁面數是 177120,當內存為 (177120 * 4) / 1024K ≈ 692M 時,系統將無法為新的 TCP 連接分配內存,即 TCP 連接將被拒絕。

根據實際場景調節的策略

在高併發服務器中,為了兼顧網速與大量的併發連接,我們應當保證緩衝區的動態調整的最大值達到帶寬時延積,而最小值保持默認的 4K 不變即可。而對於內存緊張的服務而言,調低默認值是提高併發的有效手段。

同時,如果這是網絡 IO 型服務器,那麼,調大 tcp_mem 的上限可以讓 TCP 連接使用更多的系統內存,這有利於提升併發能力。需要注意的是,tcp_wmem 和 tcp_rmem 的單位是字節,而 tcp_mem 的單位是頁面大小。而且,千萬不要在 socket 上直接設置 SO_SNDBUF 或者 SO_RCVBUF,這樣會關閉緩衝區的動態調整功能。

小結

本節針對 TCP 優化數據傳輸的方式,做了一些介紹。

數據傳輸的優化策略

TCP 可靠性是通過 ACK 確認報文實現的,又依賴滑動窗口提升了發送速度也兼顧了接收方的處理能力。

可是,默認的滑動窗口最大值只有 64 KB,不滿足當今的高速網絡的要求,要想要想提升發送速度必須提升滑動窗口的上限,在 Linux 下是通過設置 tcp_window_scaling 為 1 做到的,此時最大值可高達 1GB。

滑動窗口定義了網絡中飛行報文的最大字節數,當它超過帶寬時延積時,網絡過載,就會發生丟包。而當它小於帶寬時延積時,就無法充分利用網絡帶寬。因此,滑動窗口的設置,必須參考帶寬時延積。

內核緩衝區決定了滑動窗口的上限,緩衝區可分為:發送緩衝區 tcp_wmem 和接收緩衝區 tcp_rmem。

Linux 會對緩衝區動態調節,我們應該把緩衝區的上限設置為帶寬時延積。發送緩衝區的調節功能是自動打開的,而接收緩衝區需要把 tcp_moderate_rcvbuf 設置為 1 來開啟。其中,調節的依據是 TCP 內存範圍 tcp_mem。

但需要注意的是,如果程序中的 socket 設置 SO_SNDBUF 和 SO_RCVBUF,則會關閉緩衝區的動態整功能,所以不建議在程序設置它倆,而是交給內核自動調整比較好。

有效配置這些參數后,既能夠最大程度地保持併發性,也能讓資源充裕時連接傳輸速度達到最大值。

巨人的肩膀

[1] 系統性能調優必知必會.陶輝.極客時間.

[2] 網絡編程實戰專欄.盛延敏.極客時間.

[3] http://www.blogjava.net/yongboy/archive/2013/04/11/397677.html

[4] http://blog.itpub.net/31559359/viewspace-2284113/

[5] https://blog.51cto.com/professor/1909022

[6] https://vincent.bernat.ch/en/blog/2014-tcp-time-wait-state-linux

嘮嗑嘮嗑

跟大家說個沉痛的事情。

我想大部分小夥伴都發現了,最近公眾號改版,訂閱號里的信息流不再是以時間順序了,而是以推薦算法方式显示順序。

這對小林這種「周更」的作者,真的一次重重打擊,非常的不友好。

因為長時間沒發文,公眾號可能會把推薦的權重降低,這就會導致很多讀者,會收不到我的「最新」的推文,如此下去,那小林文章不就無人問津了?(抱頭痛哭 …)

另外,小林更文時間長的原因,不是因為偷懶。

而是為了把知識點「寫的更清楚,畫的更清晰」,所以這必然會花費更多更長的時間。

如果你認可和喜歡小林的文章,不想錯過文章的第一時間推送,可以動動你的小手手,給小林公眾號一個「星標」。

平時沒事,就讓「小林coding」靜靜地躺在你的訂閱號底部,但是你要知道它在這其間並非無所事事,而是在努力地準備着更好的內容,等準備好了,它自然會「蹦出」在你面前。

小林是專為大家圖解的工具人,Goodbye,我們下次見!

讀者問答

讀者問:“小林,請教個問題,somaxconn和backlog是不是都是指的是accept隊列?然後somaxconn是內核參數,backlog是通過系統調用間隔地修改somaxconn,比如Linux中listen()函數?”

兩者取最小值才是 accpet 隊列。

讀者問:“小林,還有個問題要請教下,“如果 accept 隊列滿了,那麼 server 扔掉 client 發過來的 ack”,也就是說該TCP連接還是位於半連接隊列中,沒有丟棄嗎?”

  1. 當 accept 隊列滿了,後續新進來的syn包都會被丟失
  2. 我文章的突發流量例子是,那個連接進來的時候 accept 隊列還沒滿,但是在第三次握手的時候,accept 隊列突然滿了,就會導致 ack 被丟棄,就一直處於半連接隊列。

本站聲明:網站內容來源於博客園,如有侵權,請聯繫我們,我們將及時處理

【其他文章推薦】

網頁設計一頭霧水該從何著手呢? 台北網頁設計公司幫您輕鬆架站!

※台北網頁設計公司全省服務真心推薦

※想知道最厲害的網頁設計公司"嚨底家"!

※推薦評價好的iphone維修中心

網頁設計最專業,超強功能平台可客製化

※別再煩惱如何寫文案,掌握八大原則!