标签:
我们先从著名的C10K问题开始探讨,由于早期在网络还不普及的时候,互联网的用户并不是很多,一台服务器同时在线100个用户估计在当时已经算是大型应用了。但是随着互联网的发展,用户群体迅速的扩大,每一个用户都必须与服务器保持TCP连接才能进行实时的数据交互。Facebook这样的网站同一时间的并发TCP连接可能会过亿。这时候问题就来了。
解决这种问题的思路主要有两个:一个是对于每一个连接处理分配一个独立的进程或线程;另一种就是用同一进程或线程来同时处理若干个连接。
第一种解决方案,即来一个TCP链接,就需要分配一个进程/线程。而进程又是操作系统最昂贵的资源,一台机器无法创建很多进程,比如C10K问题就要创建1万个进程,操作系统是无法承受的。
第二中解决方案,即每个进程/线程同时处理多个连接(IO多路复用),我们讨论的select,poll,epoll都是基于这种思路的。
#include <sys/time.h> #include <netinet/in.h> #include <sys/select.h> #include <sys/types.h> #include <sys/socket.h> #include <unistd.h> #include <stdio.h> #include <stdlib.h> int main(){ int socket1, socket2; struct sockaddr_in socket1addr, socket2addr; fd_set fds; struct timeval timeout = {3, 0}; char buffer[256] = {0}; socket1 = socket(PF_INET, SOCK_DGRAM, 0); //bzero(&socket1addr, sizeof(socket1addr)); socket1addr.sin_family = AF_INET; socket1addr.sin_addr.s_addr = htonl(INADDR_ANY); socket1addr.sin_port = htons(36500); bind(socket1, (struct sockaddr*)&socket1addr, sizeof(socket1addr)); socket2 = socket(PF_INET, SOCK_DGRAM, 0); //bzero(&socket2addr, sizeof(socket2addr)); socket2addr.sin_family = AF_INET; socket2addr.sin_addr.s_addr = htonl(INADDR_ANY); socket2addr.sin_port = htons(36501); bind(socket2, (struct sockaddr*)&socket2addr, sizeof(socket2addr)); while(1){ FD_ZERO(&fds); FD_SET(socket1, &fds); FD_SET(socket2, &fds); int maxfdp = (socket1 > socket2) ? (socket1+1) : (socket2 + 1); int retval = select(maxfdp, &fds, NULL, NULL, &timeout); if(retval == -1){ printf("error\n"); return -1; }else if(retval == 0){ continue; }else{ struct sockaddr_in client; int len = sizeof(client); if(FD_ISSET(socket1, &fds)){ recvfrom(socket1, buffer, 256, 0, (struct sockaddr*)&client, &len); printf("%u says:%s\n", ntohs(client.sin_port), buffer); } if(FD_ISSET(socket2, &fds)){ recvfrom(socket2, buffer, 256, 0, (struct sockaddr*)&client, &len); printf("%u says:%s\n", ntohs(client.sin_port), buffer); } } } }
#include <stdio.h> #include <string.h> #include <sys/socket.h> #include <netinet/in.h> int main(int argc, char **argv) { if(argc < 2){ printf("usage: %s port\n", argv[0]); return -1; } int sockfd; struct sockaddr_in servaddr; sockfd = socket(PF_INET, SOCK_DGRAM, 0); bzero(&servaddr, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_port = htons(atoi(argv[1])); servaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); char sendline[100]; sprintf(sendline, "Hello, world!"); sendto(sockfd, sendline, strlen(sendline), 0, (struct sockaddr *)&servaddr, sizeof(servaddr)); close(sockfd); return 1; }
标签:
原文地址:http://www.cnblogs.com/sensal/p/5493057.html