**Linux 大規模請求服務器連接數相關設置**
一般一個大規模Linux服務器請求數可能是幾十萬上百萬的情況,需要足夠的連接數來使用,所以務必進行相應的設置。
默認的Linux服務器文件描述符等打開最大是1024,用ulimit -a 查看:
~~~
[viewuser@~]$ ulimit -acore file size (blocks, -c) 0 #coredump 文件大小data seg size (kbytes, -d) unlimited
scheduling priority (-e) 0
file size (blocks, -f) unlimited
pending signals (-i) 255622
max locked memory (kbytes, -l) 64
max memory size (kbytes, -m) unlimitedopen files (-n) 1024 #打開文件數量,root賬戶無限制pipe size (512 bytes, -p) 8
POSIX message queues (bytes, -q) 819200
real-time priority (-r) 0
stack size (kbytes, -s) 8192
cpu time (seconds, -t) unlimitedmax user processes (-u) 4096 #root用戶本項是無限virtual memory (kbytes, -v) unlimited
file locks (-x) unlimited
~~~
設置要求:假設我們要設置為200W最大打開文件描述符
1. 修改 nr_open 限制 (用途:能夠配置nofile最大數)
~~~
cat /proc/sys/fs/nr_open
~~~
Linux 內核 2.6.25 以前,在內核里面宏定義是1024*1024,最大只能是100w(1048576),所以不要設置更大的值,如果Linux內核大于 2.6.25 則可以設置更大值。
設置方法:
~~~
sudo bash -c 'echo 2000000 > /proc/sys/fs/nr_open'
~~~
注意:只有修改了 nr_open 限制,才能修改下面的限制。
2. 打開文件描述符限制:修改 limits.conf 的nofile軟硬打開文件限制(用途:tcp連接數)
文件位置:/etc/security/limits.conf
查找 nofile ,如果沒有,則在自己最后加上:
2.6.25 及以前內核設置為100W:
~~~
* soft nofile 1000000
* hard nofile 1000000
~~~
2.6.25 以后版本內核可以設置為200W:
~~~
* soft nofile 2000000
* hard nofile 2000000
~~~
設置后保存本文件。(本操作必須重啟才生效,如果無法重啟,會無法生效,不確定是否使用 /sbin/sysctl -p 是否可以直接生效)
3. 打開進程限制:修改 limits.conf 中的nproc限制 (用途:進程數)
說明:如果你對進程總數量沒有特殊要求,可以不修改本選項,如果你是一個高性能多進程的server,需要很多進程來處理,那么可以修改本選項。
ulimit -a 里可以看到 max user processes 如果值是比較大的,可以不用設置 nproc 項。
配置文件:/etc/security/limits.d/20-nproc.conf (RHEL 7/CentOS 7)
~~~
* soft nproc 4096
root soft nproc unlimited
~~~
就是root無限(實際root用戶限制是:255622),其他非root用戶是4096個進程。
說明:
硬限制表明soft限制中所能設定的最大值。 soft限制指的是當前系統生效的設置值。 hard限制值可以被普通用戶降低。但是不能增加。 soft限制不能設置的比hard限制更高。 只有root用戶才能夠增加hard限制值。
當增加文件限制描述,可以簡單的把當前值雙倍。 例子如下, 如果你要提高默認值1024, 最好提高到2048, 如果還要繼續增加, 就需要設置成4096。
4. 修改 file-max 選項 (用途:可分配文件句柄數目)
file-max 價值:指定了可以分配的文件句柄的最大數目(可以使用 /proc/sys/fs/file-nr 文件查看到當前已經使用的文件句柄和總句柄數。)
(1) 臨時生效:
~~~
文件路徑:/proc/sys/fs/file-max
cat /proc/sys/fs/file-max
3252210
~~~
如果要修改,直接覆蓋文件:(比如改成200w)
~~~
sudo echo 2000000 > /proc/sys/fs/file-max
~~~
注意:如果你想每次啟動都自動執行上面的命令,可以在系統啟動配置文件/etc/rc.local里面添加一句命令:(跟永久生效差不多)echo 2000000 > /proc/sys/fs/file-max或者直接Shell全搞定:echo "echo 2000000 > /proc/sys/fs/file-max" >> /etc/rc.local
(2) 永久生效:
修改配置文件,文件位置:/etc/sysctl.conf
打開配置文件到最末尾,如果配置文件里沒有則可以直接添加:
~~~
sudo echo "fs.file-max = 2000000" >>/etc/sysctl.conf
~~~
配置文件生效:sudo /sbin/sysctl -p
5. 修改TCP等相關選項
配置文件:/etc/sysctl.conf
修改選項:
~~~
net.core.somaxconn = 2048
net.core.rmem_default = 262144
net.core.wmem_default = 262144
net.core.rmem_max = 16777216
net.core.wmem_max = 16777216
net.ipv4.tcp_rmem = 4096 4096 16777216
net.ipv4.tcp_wmem = 4096 4096 16777216
net.ipv4.tcp_mem = 786432 2097152 3145728
net.ipv4.tcp_max_syn_backlog = 16384
net.core.netdev_max_backlog = 20000
net.ipv4.tcp_fin_timeout = 15
net.ipv4.tcp_max_syn_backlog = 16384
net.ipv4.tcp_tw_reuse = 1
net.ipv4.tcp_tw_recycle = 1
net.ipv4.tcp_max_orphans = 131072
~~~
配置文件生效:sudo /sbin/sysctl -p
以上選項也可以直接給 /proc/sys/net/ 目錄下面按照各個選項可以直接使用 echo VALUE > /proc/sys/net/core/wmem_max 來直接修改內存臨時值生效。
主要看這幾項:
net.ipv4.tcp_rmem 用來配置讀緩沖的大小,三個值,第一個是這個讀緩沖的最小值,第三個是最大值,中間的是默認值。我們可以在程序中修改讀緩沖的大小,但是不能超過最小與最大。為了使每個socket所使用的內存數最小,我這里設置默認值為4096。
net.ipv4.tcp_wmem 用來配置寫緩沖的大小。讀緩沖與寫緩沖在大小,直接影響到socket在內核中內存的占用。
net.ipv4.tcp_mem 則是配置tcp的內存大小,其單位是頁,而不是字節。當超過第二個值時,TCP進入 pressure模式,此時TCP嘗試穩定其內存的使用,當小于第一個值時,就退出pressure模式。當內存占用超過第三個值時,TCP就拒絕分配 socket了,查看dmesg,會打出很多的日志“TCP: too many of orphaned sockets”。
net.ipv4.tcp_max_orphans 這個值也要設置一下,這個值表示系統所能處理不屬于任何進程的 socket數量,當我們需要快速建立大量連接時,就需要關注下這個值了。當不屬于任何進程的socket的數量大于這個值時,dmesg就會看 到”too many of orphaned sockets”。
注意:如果是客戶端程序,為了更好的訪問server程序不是卡在端口分配上,建議把客戶端的端口(port_range)范圍開大一些:
修改文件:/etc/sysctl.conf
~~~
net.ipv4.ip_local_port_range = 1024 65535
~~~
配置生效:sudo /sbin/sysctl -p
如果是客戶端,其他文件打開限制等可以參考上面的來設置。
6. 其他一些配置
(1) 打開core文件
如果為了觀察程序是否正常,出現問題后生成相應映像文件,可以開啟coredump相關的操作,可以打開:(非必須,如果線上環境,擔心影響穩定性,可以考慮不開啟)
配置文件:/etc/security/limits.conf
修改配置文件:
增加:
~~~
* soft core 102400
* hard core 2048003
~~~
建議設置為無限大小:
~~~
* soft core unlimited
* hard core unlimited
~~~
然后重啟機器生效(不確定是否可以使用 /sbin/sysctl -p 生效),使用: ulimit -a 或 ulimit -c 查看結果,后續如果程序出現棧溢出等都會生成coredump文件,方便用gdb等追查問題原因。
**客戶端也需要修改參數來加大websocket的連接數**
**客戶端壓測代碼**
~~~
package main
import (
"fmt"
"log"
"math/rand"
"runtime"
"sync"
"time"
"golang.org/x/net/websocket"
)
var (
origin = "http://localhost:6968/"
url = "ws://localhost:6968/sub?key=Go_Push&heartbeat=60&token=0&room=1"
exit chan bool = make(chan bool)
cycle int = 500
wsm sync.Map
)
const maxPostHandlerReadBytes = 256 << 10
func main() {
runtime.GOMAXPROCS(runtime.NumCPU() * 2)
rand.Seed(time.Now().Unix())
t := time.NewTicker(time.Second * 5)
var k int = 0
for v := range t.C {
for i := 0; i < cycle; i++ {
url = fmt.Sprintf("ws://localhost:6968/sub?key=%v&heartbeat=60&token=0&room=%d", rand.Intn(cycle), 1)
// fmt.Println(url)
go wsClient(url, origin, wsm)
fmt.Println("i => ", (i + 1))
}
fmt.Println(v)
k++
if k == 100 {
fmt.Printf("退出\n")
time.Sleep(time.Hour)
return
}
}
<-exit
}
func wsClient(url, origin string, wsm sync.Map) {
ws, err := websocket.Dial(url, "", origin)
wsm.Store(ws, "inter")
if err != nil {
fmt.Println(err)
return
}
go func() {
message := []byte("h")
for {
val, ok := wsm.Load(ws)
if !ok {
break
}
if val.(string) == "exit" {
break
}
_, err = ws.Write(message)
if err != nil {
log.Fatal(err)
break
}
fmt.Printf("Send: %s\n", message)
time.Sleep(time.Second * 30)
}
}()
var msg = make([]byte, 512)
var m int
for {
m, err = ws.Read(msg)
if err != nil {
fmt.Println(err)
wsm.Store(ws, "exit")
break
}
fmt.Printf("Receive: %s\n", msg[:m])
}
fmt.Println("關閉鏈接")
ws.Close() //關閉連接
}
~~~
- 序言
- 目錄
- 環境搭建
- Linux搭建golang環境
- Windows搭建golang環境
- Mac搭建golang環境
- Go 環境變量
- 編輯器
- vs code
- Mac 安裝vs code
- Windows 安裝vs code
- vim編輯器
- 介紹
- 1.Go語言的主要特征
- 2.golang內置類型和函數
- 3.init函數和main函數
- 4.包
- 1.工作空間
- 2.源文件
- 3.包結構
- 4.文檔
- 5.編寫 Hello World
- 6.Go語言 “ _ ”(下劃線)
- 7.運算符
- 8.命令
- 類型
- 1.變量
- 2.常量
- 3.基本類型
- 1.基本類型介紹
- 2.字符串String
- 3.數組Array
- 4.類型轉換
- 4.引用類型
- 1.引用類型介紹
- 2.切片Slice
- 3.容器Map
- 4.管道Channel
- 5.指針
- 6.自定義類型Struct
- 流程控制
- 1.條件語句(if)
- 2.條件語句 (switch)
- 3.條件語句 (select)
- 4.循環語句 (for)
- 5.循環語句 (range)
- 6.循環控制Goto、Break、Continue
- 函數
- 1.函數定義
- 2.參數
- 3.返回值
- 4.匿名函數
- 5.閉包、遞歸
- 6.延遲調用 (defer)
- 7.異常處理
- 8.單元測試
- 壓力測試
- 方法
- 1.方法定義
- 2.匿名字段
- 3.方法集
- 4.表達式
- 5.自定義error
- 接口
- 1.接口定義
- 2.執行機制
- 3.接口轉換
- 4.接口技巧
- 面向對象特性
- 并發
- 1.并發介紹
- 2.Goroutine
- 3.Chan
- 4.WaitGroup
- 5.Context
- 應用
- 反射reflection
- 1.獲取基本類型
- 2.獲取結構體
- 3.Elem反射操作基本類型
- 4.反射調用結構體方法
- 5.Elem反射操作結構體
- 6.Elem反射獲取tag
- 7.應用
- json協議
- 1.結構體轉json
- 2.map轉json
- 3.int轉json
- 4.slice轉json
- 5.json反序列化為結構體
- 6.json反序列化為map
- 終端讀取
- 1.鍵盤(控制臺)輸入fmt
- 2.命令行參數os.Args
- 3.命令行參數flag
- 文件操作
- 1.文件創建
- 2.文件寫入
- 3.文件讀取
- 4.文件刪除
- 5.壓縮文件讀寫
- 6.判斷文件或文件夾是否存在
- 7.從一個文件拷貝到另一個文件
- 8.寫入內容到Excel
- 9.日志(log)文件
- server服務
- 1.服務端
- 2.客戶端
- 3.tcp獲取網頁數據
- 4.http初識-瀏覽器訪問服務器
- 5.客戶端訪問服務器
- 6.訪問延遲處理
- 7.form表單提交
- web模板
- 1.渲染終端
- 2.渲染瀏覽器
- 3.渲染存儲文件
- 4.自定義io.Writer渲染
- 5.模板語法
- 時間處理
- 1.格式化
- 2.運行時間
- 3.定時器
- 鎖機制
- 互斥鎖
- 讀寫鎖
- 性能比較
- sync.Map
- 原子操作
- 1.原子增(減)值
- 2.比較并交換
- 3.導入、導出、交換
- 加密解密
- 1.md5
- 2.base64
- 3.sha
- 4.hmac
- 常用算法
- 1.冒泡排序
- 2.選擇排序
- 3.快速排序
- 4.插入排序
- 5.睡眠排序
- 限流器
- 日志包
- 日志框架logrus
- 隨機數驗證碼
- 生成指定位數的隨機數
- 生成圖形驗證碼
- 編碼格式轉換
- UTF-8與GBK
- 解決中文亂碼
- 設計模式
- 創建型模式
- 單例模式
- singleton.go
- singleton_test.go
- 抽象工廠模式
- abstractfactory.go
- abstractfactory_test.go
- 工廠方法模式
- factorymethod.go
- factorymethod_test.go
- 原型模式
- prototype.go
- prototype_test.go
- 生成器模式
- builder.go
- builder_test.go
- 結構型模式
- 適配器模式
- adapter.go
- adapter_test.go
- 橋接模式
- bridge.go
- bridge_test.go
- 合成/組合模式
- composite.go
- composite_test.go
- 裝飾模式
- decoretor.go
- decorator_test.go
- 外觀模式
- facade.go
- facade_test.go
- 享元模式
- flyweight.go
- flyweight_test.go
- 代理模式
- proxy.go
- proxy_test.go
- 行為型模式
- 職責鏈模式
- chainofresponsibility.go
- chainofresponsibility_test.go
- 命令模式
- command.go
- command_test.go
- 解釋器模式
- interpreter.go
- interperter_test.go
- 迭代器模式
- iterator.go
- iterator_test.go
- 中介者模式
- mediator.go
- mediator_test.go
- 備忘錄模式
- memento.go
- memento_test.go
- 觀察者模式
- observer.go
- observer_test.go
- 狀態模式
- state.go
- state_test.go
- 策略模式
- strategy.go
- strategy_test.go
- 模板模式
- templatemethod.go
- templatemethod_test.go
- 訪問者模式
- visitor.go
- visitor_test.go
- 數據庫操作
- golang操作MySQL
- 1.mysql使用
- 2.insert操作
- 3.select 操作
- 4.update 操作
- 5.delete 操作
- 6.MySQL事務
- golang操作Redis
- 1.redis介紹
- 2.golang鏈接redis
- 3.String類型 Set、Get操作
- 4.String 批量操作
- 5.設置過期時間
- 6.list隊列操作
- 7.Hash表
- 8.Redis連接池
- 其它Redis包
- go-redis/redis包
- 安裝介紹
- String 操作
- List操作
- Set操作
- Hash操作
- golang操作ETCD
- 1.etcd介紹
- 2.鏈接etcd
- 3.etcd存取
- 4.etcd監聽Watch
- golang操作kafka
- 1.kafka介紹
- 2.寫入kafka
- 3.kafka消費
- golang操作ElasticSearch
- 1.ElasticSearch介紹
- 2.kibana介紹
- 3.寫入ElasticSearch
- NSQ
- 安裝
- 生產者
- 消費者
- zookeeper
- 基本操作測試
- 簡單的分布式server
- Zookeeper命令行使用
- GORM
- gorm介紹
- gorm查詢
- gorm更新
- gorm刪除
- gorm錯誤處理
- gorm事務
- sql構建
- gorm 用法介紹
- Go操作memcached
- beego框架
- 1.beego框架環境搭建
- 2.參數配置
- 1.默認參數
- 2.自定義配置
- 3.config包使用
- 3.路由設置
- 1.自動匹配
- 2.固定路由
- 3.正則路由
- 4.注解路由
- 5.namespace
- 4.多種數據格式輸出
- 1.直接輸出字符串
- 2.模板數據輸出
- 3.json格式數據輸出
- 4.xml格式數據輸出
- 5.jsonp調用
- 5.模板處理
- 1.模板語法
- 2.基本函數
- 3.模板函數
- 6.請求處理
- 1.GET請求
- 2.POST請求
- 3.文件上傳
- 7.表單驗證
- 1.表單驗證
- 2.定制錯誤信息
- 3.struct tag 驗證
- 4.XSRF過濾
- 8.靜態文件處理
- 1.layout設計
- 9.日志處理
- 1.日志處理
- 2.logs 模塊
- 10.會話控制
- 1.會話控制
- 2.session 包使用
- 11.ORM 使用
- 1.鏈接數據庫
- 2. CRUD 操作
- 3.原生 SQL 操作
- 4.構造查詢
- 5.事務處理
- 6.自動建表
- 12.beego 驗證碼
- 1.驗證碼插件
- 2.驗證碼使用
- beego admin
- 1.admin安裝
- 2.admin開發
- beego 熱升級
- beego實現https
- gin框架
- 安裝使用
- 路由設置
- 模板處理
- 文件上傳
- gin框架中文文檔
- gin錯誤總結
- 項目
- 秒殺項目
- 日志收集
- 面試題
- 面試題一
- 面試題二
- 錯題集
- Go語言陷阱和常見錯誤
- 常見語法錯誤
- 初級
- 中級
- 高級
- Go高級應用
- goim
- goim 啟動流程
- goim 工作流程
- goim 結構體
- gopush
- gopush工作流程
- gopush啟動流程
- gopush業務流程
- gopush應用
- gopush新添功能
- gopush壓力測試
- 壓測注意事項
- rpc
- HTTP RPC
- TCP RPC
- JSON RPC
- 常見RPC開源框架
- pprof
- pprof介紹
- pprof應用
- 使用pprof及Go 程序的性能優化
- 封裝 websocket
- cgo
- Golang GC
- 查看程序運行過程中的GC信息
- 定位gc問題所在
- Go語言 demo
- 用Go語言計算一個人的年齡,生肖,星座
- 超簡易Go語言實現的留言板代碼
- 信號處理模塊,可用于在線加載配置,配置動態加載的信號為SIGHUP
- 陽歷和陰歷相互轉化的工具類 golang版本
- 錯誤總結
- 網絡編程
- 網絡編程http
- 網絡編程tcp
- Http請求
- Go語言必知的90個知識點
- 第三方庫應用
- cli應用
- Cobra
- 圖表庫
- go-echarts
- 開源IM
- im_service
- 機器學習庫
- Tensorflow
- 生成二維碼
- skip2/go-qrcode生成二維碼
- boombuler/barcode生成二維碼
- tuotoo/qrcode識別二維碼
- 日志庫
- 定時任務
- robfig/cron
- jasonlvhit/gocron
- 拼多多開放平臺 SDK
- Go編譯
- 跨平臺交叉編譯
- 一問一答
- 一問一答(一)
- 為什么 Go 標準庫中有些函數只有簽名,沒有函數體?
- Go開發的應用
- etcd
- k8s
- Caddy
- nsq
- Docker
- web框架