c - 套接字接受在没有释放的情况下消耗了我在 Windows 上的内存

标签 c windows sockets gcc memory-management

我用 C 写了一个非常小的函数,打开一个套接字,接受连接并立即关闭它们。

问题是,每个连接都会消耗一些内存,而不会随时将其释放给操作系统。我用大约 30 万个请求运行了 ab(apache 基准测试),并且进程内存在不断增长(最后是几百兆字节)。

我知道进程并不总是将其空闲内存返回给操作系统。但是一旦它超过几兆字节,我认为它应该返回内存或者我错了吗?

这只发生在 Windows 上。在 Linux 上,我的内存使用量一直接近进程启动使用量。

使用 GCC 4.8.2 编译。在 Windows Server 2008 R2 和 Windows 8.1 上测试。

void http_server_start(void) {
  int rc;
  struct sockaddr_in cfg;

  #ifdef _WIN32
  WORD ver;
  WSADATA data;
  ver=MAKEWORD(2,2);
  rc=WSAStartup(ver, &data);
  if(rc != 0){
    printf("Error: Unable to initialize WSA (%d)", rc);
  }
  #endif

  memset(&cfg, 0, sizeof(cfg));
  cfg.sin_family = AF_INET;
  cfg.sin_addr.s_addr = htonl(INADDR_ANY);
  cfg.sin_port = htons(PORT);
  server = socket(AF_INET, SOCK_STREAM, 6);
  int reuseaddr=1;
  if (setsockopt(server, SOL_SOCKET, SO_REUSEADDR, (char*)&reuseaddr, sizeof(reuseaddr)) == -1){
    rc=GetLastErrorEx();
    printf("Error: Unable to set SO_REUSEADDR (%d)\n", rc);
  } else if (bind(server, (struct sockaddr *)&cfg, sizeof(cfg)) < 0 ) {
    rc=GetLastErrorEx();
    printf("Error: Unable to bind socket (%d)\n", rc);
    close(server);
  } else if (listen(server, QUEUE_SIZE) < 0) {
    rc=GetLastErrorEx();
    printf("Error: Unable to listen (%d)\n", rc);
    close(server);
  } else {
    printf("Listening on %s:%d\n", inet_ntoa(cfg.sin_addr), ntohs(cfg.sin_port));
    int client;
    struct sockaddr_in addr;
    int addrlen=sizeof(addr);
    do {
      client=accept(server, (struct sockaddr*)&addr, &addrlen);
      if(client != -1){
        shutdown(client, SHUT_RDWR);
        close(client);
      }
    } while(1);
  }
}

Windws 8 Memory usage Windows 2008R2 Memory usage

最佳答案

要关闭套接字,您可以使用 shutdown() 和 close()。 实际上 shutdown() 会强制终止连接,即使有 几个描述符引用同一个套接字, shutdown() 只允许在一个方向关闭而 close() 不允许 影响连接的状态,直到 last 描述符是 关闭。

尝试使用

 int closesocket(Socket S);

每次成功调用 socket 时,应用程序都应始终匹配调用 closesocket() 以将任何套接字资源返回给系统。

关于c - 套接字接受在没有释放的情况下消耗了我在 Windows 上的内存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24033007/

相关文章:

linux - 我的程序应该如何处理错误?

c - 指针在 printf() 中不起作用

c - 加密 API 调用失败

C - 无法绑定(bind)服务器套接字

c++ - 异步,非阻塞套接字行为-WSAEWOULDBLOCK

java - 在同一台机器上调试多个Socket IP地址

add/sub/div/mul 的 C 解析器程序。读取用户的输入想要在分号处终止我的程序而不是按回车

windows - 记录系统调用

c++ - 如何将 DLL 注入(inject)任何进程?

.net - 单击托盘图标的正确行为?