我用 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);
}
}
最佳答案
要关闭套接字,您可以使用 shutdown() 和 close()。 实际上 shutdown() 会强制终止连接,即使有 几个描述符引用同一个套接字, shutdown() 只允许在一个方向关闭而 close() 不允许 影响连接的状态,直到 last 描述符是 关闭。
尝试使用
int closesocket(Socket S);
每次成功调用 socket 时,应用程序都应始终匹配调用 closesocket() 以将任何套接字资源返回给系统。
关于c - 套接字接受在没有释放的情况下消耗了我在 Windows 上的内存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24033007/