高并发服务器学习笔记之十:PreForking+epoll

这个就不多说了,各种模型的一种综合应用,最终目的就是尽一切手段去压榨服务器资源,完整代码戳这里​​​​​​​

#include 
#include 
#include 
#include 

#include 
#include 
#include           /* See NOTES */
#include 
#include 
#include 
#include 
#include  /* superset of previous */
#include 

#include "public_head.h"

#define LISTEN_BACKLOG 50
#define MAX_PROGRESS	1
#define MAX_EVENTS	5

static void grim_reaper(int sig)
{
	int saved_error = errno;
	while (waitpid(-1, NULL, WNOHANG) >= 0)
		continue;

	errno = saved_error;
}

static ssize_t handle_request(int acceptfd)
{
	ssize_t readret = 0;
	char read_buff[256] = { 0 };
	char write_buff[256] = { 0 };


	memset(read_buff, 0, sizeof(read_buff));
	readret = read(acceptfd, read_buff, sizeof(read_buff));
	if (readret == 0)
		return readret;

	printf("acceptfd:%d, recv message:%s\n", acceptfd, read_buff);

	memset(write_buff, 0, sizeof(write_buff));
	sprintf(write_buff, "This is server send message");
	write(acceptfd, write_buff, sizeof(write_buff));

	printf("\n");
	return readret;
}

int main(int argc, char ** argv)
{
	int i;
	pid_t pid;
    int sockfd = 0;
    int acceptfd = 0;
    socklen_t client_addr_len = 0;
	struct sigaction sa;
    struct sockaddr_in server_addr, client_addr;

    char client_ip[16] = { 0 };

	int epfd = -1, ready = -1;
	struct epoll_event ev;
	struct epoll_event evlist[MAX_EVENTS];

    memset(&server_addr, 0, sizeof(server_addr));
    memset(&client_addr, 0, sizeof(client_addr));

	sigemptyset(&sa.sa_mask); 
	sa.sa_flags = SA_RESTART;
	sa.sa_handler = grim_reaper;
	if(sigaction(SIGCHLD, &sa, NULL) < 0)
		handle_error("sigaction");

    if((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
        handle_error("socket");

    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(9527);
    server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
    if(bind(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0)
    {
		char buff[256] = { 0 };
        close(sockfd);
		strerror_r(errno, buff, sizeof(buff));
        handle_error("bind");
    }

    if(listen(sockfd, LISTEN_BACKLOG) < 0)
    {
        close(sockfd);
        handle_error("listen");
    }

	for (i = 0; i < MAX_PROGRESS; ++i)
	{
		pid = fork();
		if (pid > 0)	//parent
		{
			continue;
		}
		else if (pid == 0)	//child
		{
			break;
		}
		else
		{
			close(sockfd);
			while (wait(NULL))
				continue;
			handle_error("fork");
		}
	}

	epfd = epoll_create1(0);
	if (epfd < 0)
	{
		close(sockfd);
		handle_error("epoll_create1");
	}

	ev.data.fd = sockfd;
	ev.events = EPOLLIN;
	if (epoll_ctl(epfd, EPOLL_CTL_ADD, sockfd, &ev) < 0)
	{
		close(sockfd);
		close(epfd);
		handle_error("epoll_ctl");
	}
	
	while (1)
	{
		ready = epoll_wait(epfd, evlist, MAX_EVENTS, -1);
		if (ready < 0)
		{
			if (errno == EINTR)
				continue;

			close(sockfd);
			close(epfd);
			handle_error("epoll_wait");
		}
		else if (ready == 0)
		{
			continue;
		}

		for (i = 0; i < ready; ++i)
		{
			if (evlist[i].events != EPOLLIN)
				continue;

			if (evlist[i].data.fd == sockfd)
			{
				client_addr_len = sizeof(client_addr);
				if ((acceptfd = accept(sockfd, (struct sockaddr *)&client_addr, &client_addr_len)) < 0)
				{
					handle_warning("accept");
					continue;
				}

				memset(client_ip, 0, sizeof(client_ip));
				inet_ntop(AF_INET, &client_addr.sin_addr, client_ip, sizeof(client_ip));
				printf("client:%s:%d\n", client_ip, ntohs(client_addr.sin_port));

				ev.data.fd = acceptfd;
				ev.events = EPOLLIN;
				if (epoll_ctl(epfd, EPOLL_CTL_ADD, acceptfd, &ev) < 0)
				{
					close(acceptfd);
					handle_warning("epoll_ctl");
					continue;
				}
			}
			else
			{
				if (handle_request(evlist[i].data.fd) <= 0)
				{
					ev.data.fd = evlist[i].data.fd;
					ev.events = EPOLLIN;
					if (epoll_ctl(epfd, EPOLL_CTL_DEL, evlist[i].data.fd, &ev) < 0)
						handle_warning("epoll_ctl");
					close(evlist[i].data.fd);
				}
			}
		}
	}
    
    close(sockfd);

	exit(EXIT_SUCCESS);
}

 

 

 

你可能感兴趣的:(一步步打造高并发服务器)