** 原创文章,请勿转载 **

并发服务器是一个老生常谈的话题,今天这里也写一个。

1. 目标

同时在线连接20万(c20w)。

开发语言:重要的事情说三遍,GOLANG, GOLANG, GOLANG!

那为什么是20W,不是30W或其它? 这个数字随意。   :)

2. 环境:

虚拟机(xenserver),    虚出6台机器(OS: CentOS 6.6 64bit) :
       . 一台服务器8核CPU,2G内存
       . 五台客户端2核CPU,2G内存

3.  改centos几个参数, 6台机器一样:
  
# ulimit -a

看 open files
      改成300000(大于20W就行) :
   #  ulimit -n 300000

   改端口范围
  

# echo 1025 65000 > /proc/sys/net/ipv4/ip_local_port_range

4. golang代码

    服务端: server.go

  1. package main
  2.  
  3. import (
  4. "log"
  5. "net"
  6. "sync"
  7. "time"
  8. )
  9.  
  10. func main() {
  11. tcpAddr, err := net.ResolveTCPAddr("tcp", ":5000")
  12.  
  13. if err != nil {
  14. log.Fatalln("net.ResolveTCPAddr fail", err)
  15. }
  16.  
  17. listener, err := net.ListenTCP("tcp4", tcpAddr)
  18. if err != nil {
  19. log.Println("server failed to start...")
  20. return
  21. }
  22.  
  23. defer listener.Close()
  24. go print()
  25. log.Println("begin listening, addr : ", tcpAddr)
  26.  
  27. for {
  28. conn, err := listener.Accept()
  29. if err != nil {
  30. log.Println("socket accept failed. reason: ", err)
  31. continue
  32. }
  33.  
  34. go handleConnection(conn)
  35. }
  36.  
  37. }
  38.  
  39. func print() {
  40. ticker := time.NewTicker(5 * time.Second)
  41.  
  42. for {
  43. <-ticker.C
  44. log.Println(len(clients.Map))
  45. }
  46. }
  47.  
  48. var num = 0
  49. var num2 = 0
  50. var clients = Clients{Map: make(map[string]int, 0)}
  51.  
  52. type Clients struct {
  53. sync.RWMutex
  54. Map map[string]int
  55. }
  56.  
  57. func (c *Clients) Add(ip string) {
  58. c.Lock()
  59. c.Map[ip] = 1
  60. c.Unlock()
  61. }
  62.  
  63. func (c *Clients) Del(ip string) {
  64. c.Lock()
  65. delete(c.Map, ip)
  66. c.Unlock()
  67. }
  68. func handleConnection(conn net.Conn) {
  69. ip := conn.RemoteAddr().String()
  70. clients.Add(ip)
  71.  
  72. buf := make([]byte, 64)
  73. for {
  74. _, err := conn.Read(buf)
  75. //log.Println("n=", n)
  76. if err != nil {
  77. //log.Println(err)
  78. conn.Close()
  79. clients.Del(ip)
  80. }
  81. }
  82. }

客户端: client.go

  1. package main
  2.  
  3. import (
  4. "log"
  5. "net"
  6. )
  7.  
  8. func main() {
  9. for i := 0; i < 40000; i++ {
  10. conn, err := net.Dial("tcp", "x.x.x.x:5000")
  11. if err != nil {
  12. log.Println(err)
  13. return
  14. }
  15.  
  16. go Read(conn)
  17. }
  18.  
  19. log.Println("ok")
  20. select {}
  21. }
  22.  
  23. func Read(conn net.Conn) {
  24. buf := make([]byte, 64)
  25. for {
  26. _, err := conn.Read(buf)
  27. if err != nil {
  28. log.Println(err)
  29. return
  30. }
  31. }
  32. }

结论:

开始时,服务器是2核,2G内存, 达到8W连接左右,接受新连接的速度就慢了。后来改成8核,20W连接无压力。

最终5个客户端,每个客户端发起40000个连接, 共20W连接。
      服务器单进程, 接受20W个连接。CPU,内存压力都很小。

tcp并发服务器(c20w)的更多相关文章

  1. TCP并发服务器简单示例

    并发服务器的思想是每一个客户的请求并不由服务器直接处理,而是由服务器创建一个子进程来处理 1. 服务器端 #include <stdio.h> #include <sys/types ...

  2. Linux网络编程——tcp并发服务器(poll实现)

    想详细彻底地了解poll或看懂下面的代码请参考<Linux网络编程——I/O复用之poll函数> 代码: #include <string.h> #include <st ...

  3. Linux select TCP并发服务器与客户端编程

    介绍:运行在ubuntu linux系统,需要先打开一个终端运行服务端代码,这时,可以打开多个终端同时运行多个客户端代码(注意客户端数目要小于MAX_FD);在客户端输入数据后回车,可以看见服务器收到 ...

  4. UNIX网络编程——并发服务器(TCP)

    在迭代服务器中,服务器只能处理一个客户端的请求,如何同时服务多个客户端呢?在未讲到select/poll/epoll等高级IO之前,比较老土的办法是使用fork来实现. 网络服务器通常用fork来同时 ...

  5. tcp并发服务端

    TCP并发服务器:并发服务器的思想是每一个客户端的请求并不由服务器的主进程直接处理,而是服务器主进程创建一个子进程来处理. 创建TCP并发服务器的算法如下: socket(……): //创建一个TCP ...

  6. LINUX环境并发服务器的三种实现模型

    服务器设计技术有很多,按使用的协议来分有TCP服务器和UDP服务器.按处理方式来分有循环服务器和并发服务器. 1  循环服务器与并发服务器模型 在网络程序里面,一般来说都是许多客户对应一个服务器,为了 ...

  7. Linux网络编程服务器模型选择之并发服务器(上)

    与循环服务器的串行处理不同,并发服务器对服务请求并发处理.循环服务器只能够一个一个的处理客户端的请求,显然效率很低.并发服务器通过建立多个子进程来实现对请求的并发处理.并发服务器的一个难点是如何确定子 ...

  8. linux 网络编程 3---(io多路复用,tcp并发)

    1,io模型: 阻塞io.非阻塞io.io多路复用,信号驱动io. 阻塞Io与非阻塞io的转换,可用fcntl()函数 #include<unistd.h> #include<fcn ...

  9. 高并发服务器建议调小 TCP 协议的 time_wait 超时时间。

    1. [推荐]高并发服务器建议调小 TCP 协议的 time_wait 超时时间. 说明:操作系统默认 240 秒后,才会关闭处于 time_wait 状态的连接,在高并发访问下,服 务器端会因为处于 ...

随机推荐

  1. ThreadPoolExecutor系列<三、ThreadPoolExecutor 源码解析>

    本文系作者原创,转载请注明出处:http://www.cnblogs.com/further-further-further/p/7681826.html 在源码解析前,需要先理清线程池控制的运行状态 ...

  2. 突发小事件,USB接口问题

    昨天遇到的突发事件,突然USB接口全部瘫了,键盘鼠标全部不能用,换到别人电脑上可以,吓尿了,以为本子主板挂了,但是发现插U盘竟然可以识别而且可以打开,感觉可能是静电问题,果然,彻底关机,拔掉电池,然后 ...

  3. Python实现正交实验法自动设计测试用例

    1.简介 正交试验法是研究多因素.多水平的一种试验法,它是利用正交表来对试验进行设计,通过少数的试验替代全面试验,根据正交表的正交性从全面试验中挑选适量的.有代表性的点进行试验,这些有代表性的点具备了 ...

  4. JS框架设计读书笔记之-核心模块

    随笔记录一下读书心得 1. 框架模块-核心模块 该模块是框架最先执行的部分,jQuery与vue中都有初始化的代码. 模块的功能主要是:对象扩展.数组化.类型判定.事件绑定和解绑.无冲突处理.模块加载 ...

  5. git常用命令集合

    git命令 git init:创建一个仓库并在目录下新建一个.git的文件(初始化一个git仓库) 注:.git文件在工作区,是一个隐藏文件(用ls -ah命令查看),但是它不算工作区,而是git 的 ...

  6. java项目log4j日志打印配置

    #定义输出级别和输出平台  添加DEBUG表示打印sql 语句 log4j.rootLogger=DEBUG,INFO,ERROR,stdout,R log4j.category.org.spring ...

  7. 第三章 CUDA设备相关

    这章介绍了与CUDA设备相关的参数,并给出了了若干用于查询参数的函数. 章节代码(已合并): #include <stdio.h> #include "cuda_runtime. ...

  8. 无阻赛的脚本(js脚本延迟方法)

    js脚本的加载与执行 1.延迟脚本(defer属性) 带有defer属性的script标签,可以放置在文档的任何位置,在页面解析到该标签时,会开始下载该脚本,但是不会立即执行,直到dom加载完成(on ...

  9. 状态码为 200 from cache和304 Not modified的区别

    1.请求状态码为 200  from cache: 表示该资源已经被缓存过,并且在有效期内,所以不再向浏览器发出请求,直接使用本地缓存. 如下图: 2.状态码为 304 Not modified: 表 ...

  10. Python 字典和json的本质区别(个人理解)

    个人理解:字典和json显示的时候差不多,但是数据类型不同(如下图): 字典的类型是字典dict json的类型是字符串str 接口测试是传参数payload时有时候是传的字符串,应该将payload ...