上一篇博客用多线程实现服务端和多个客户端的通信,但是在实际应用中如果服务端有高并发的需求,多线程并不是一个好选择。

实现高并发的一种方法是IO多路复用,也就是select,poll,epoll等等。

于是我采用epoll再修改了服务端,实现单线程服务多个客户端。

服务端:

 #include <stdio.h>
#include <stdlib.h>
#include <fcntl.h> // open function
#include <unistd.h> // close function
#include <sys/socket.h>
#include <netinet/in.h>
#include <string.h>
#include <pthread.h>
#include <errno.h>
#include <sys/epoll.h> const int PORT = ;
/*
listen_loop(): epoll监听套接字,作不同处理
accept_conn(): 新的客户端连接进来,执行accept,将fd加入epoll set
recv_message(): recv并且重复输出一份给客户端
*/
void listen_loop();
void accept_conn(unsigned int sock_fd, unsigned int epollfd);
void recv_message(unsigned int sock_fd); int main(void) {
int sock_fd;
struct sockaddr_in server_addr; //初始化socket
sock_fd = socket(AF_INET, SOCK_STREAM, );
if (sock_fd < ) {
perror("socket:");
return ;
} //编辑地址
memset(&server_addr, , sizeof(server_addr));
server_addr.sin_family = AF_INET;//ipv_4
server_addr.sin_port = htons(PORT);//监听端口8888
server_addr.sin_addr.s_addr = INADDR_ANY;//本地的任意地址 //绑定然后监听
if (bind(sock_fd, (struct sockaddr*)&server_addr, sizeof(server_addr)) < ) {
perror("bind:");
return ;
}
if (listen(sock_fd, ) < ) {
perror("listen");
return ;
} listen_loop(sock_fd); return ;
}
void accept_conn(unsigned int sock_fd, unsigned int epollfd) {
struct sockaddr_in clientaddr;
struct epoll_event event;
socklen_t len = sizeof(struct sockaddr);
int accept_fd = ; accept_fd = accept(sock_fd, (struct sockaddr*)&clientaddr, &len); if (accept_fd <= ) {
perror("accept error");
return;
} //将新建连接加入epoll set
event.data.fd = accept_fd;
event.events = EPOLLIN | EPOLLET | EPOLLRDHUP;
epoll_ctl(epollfd, EPOLL_CTL_ADD, accept_fd, &event);
return;
} void recv_message(unsigned int sock_fd) {
char recv_buf[], send_buf[]; memset(recv_buf, , sizeof(recv_buf));
memset(send_buf, , sizeof(send_buf)); recv(sock_fd, recv_buf, sizeof(recv_buf), );
fputs(recv_buf, stdout);
strcpy(send_buf, recv_buf);
send(sock_fd, send_buf, sizeof(send_buf), ); return;
}
void listen_loop(unsigned int sock_fd)
{
int epollfd, i, ret;
int timeout = ;
struct epoll_event event;
struct epoll_event eventList[]; /*创建epoll监听事件*/
epollfd = epoll_create();
event.events = EPOLLIN | EPOLLET;
event.data.fd = sock_fd; /*注册epoll监听事件.*/
if (epoll_ctl(epollfd, EPOLL_CTL_ADD, sock_fd, &event) < ) {
printf("register epoll event err !");
return;
} while () {
ret = epoll_wait(epollfd, eventList, , timeout); /*epoll事件错误.*/
if (ret < ) {
printf("epoll event err!");
break;
}
/*无事件返回.*/
else if (ret == ) {
continue;
} /*epoll返回事件.*/
for (i = ; i < ret; i++) {
/*epoll 错误*/
if ((eventList[i].events & EPOLLERR) || (eventList[i].events & EPOLLHUP) || !(eventList[i].events & EPOLLIN)) {
printf("epoll error\n");
close(eventList[i].data.fd);
exit(-);
} //half connection
if (eventList[i].events & EPOLLRDHUP) {
printf("//one client close the conne.//\n");
close(eventList[i].data.fd);
} /*accept事件*/
if (eventList[i].data.fd == sock_fd) {
accept_conn(sock_fd, epollfd);
}
/*非sock_fd则为其他事件.*/
else {
recv_message(eventList[i].data.fd);
}
}
}
close(epollfd);
close(sock_fd);
return;
}

Linux下socket通信和epoll的更多相关文章

  1. Linux 下socket通信终极指南(附TCP、UDP完整代码)

    linux下用socket通信,有TCP.UDP两种协议,网上的很多教程把两个混在了一起,或者只讲其中一种.现在我把自己这两天研究的成果汇总下来,写了一个完整的,适合初学者参考,也方便自己以后查阅. ...

  2. Linux下socket通信和多线程

    服务端socket流程:socket() –> bind() –> listen() –> accept() –> 读取.发送信息(recv,send等) 客户端socket流 ...

  3. (8)Linux(客户端)和Windows(服务端)下socket通信实例

    Linux(客户端)和Windows(服务端)下socket通信实例: (1)首先是Windows做客户端,Linux做服务端的程序 Windows   Client端 #include <st ...

  4. Linux下进程通信的八种方法

    Linux下进程通信的八种方法:管道(pipe),命名管道(FIFO),内存映射(mapped memeory),消息队列(message queue),共享内存(shared memory),信号量 ...

  5. linux下socket编程实例

    linux下socket编程实例一.基本socket函数Linux系统是通过提供套接字(socket)来进行网络编程的.网络的socket数据传输是一种特殊的I/O,socket也是一种文件描述符.s ...

  6. (转)Linux下select, poll和epoll IO模型的详解

    Linux下select, poll和epoll IO模型的详解 原文:http://blog.csdn.net/tianmohust/article/details/6677985 一).Epoll ...

  7. Linux下Socket编程的端口问题( Bind error: Address already in use )

    Linux下Socket编程的端口问题( Bind error: Address already in use ) 在进行linux网络编程时,每次修改了源代码并再次编译运行时,常遇到下面的地使用错误 ...

  8. linux下串口通信与管理

    linux下的串口与windows有一些区别,下面将介绍一下linux下串口通信管理 查看是否支持USB串口: #lsmod | grep usbserial 如果没有信息:sudo apt-get ...

  9. UNIX下socket通信 - UDP通信

    一.UNIX下socket通信: socket套接字是一种可以进行网络通信的内核对象,它是一个唯一的标示符,一般称它为socket描述符. 注意:UDP通信需要客户端先发送消息,服务端先进行等待客户端 ...

随机推荐

  1. jquery判断滚动到某个div显示底部按钮

    判读滚动某个div显示底部按钮 <!DOCTYPE html> <html lang="zh-CN"> <head> <meta char ...

  2. LayUI把表格中的时间戳改成格式化的时间

  3. Storm集群参数调整

    Supervisor 参数调整 修改${STORM_HOME}conf/storm.yaml文件内容 supervisor变更参数 slots 配置: 若storm host仅仅执行superviso ...

  4. [bug]微信小程序使用 <scroll-view> 和 box-shadow 引起页面抖动

    背景 为了实现点点点动态loading效果,并且方便使用(只需要给一个空元素加一个.loading),有如下代码: .loader { background-color: #fff; font-siz ...

  5. iOS完全自学手册——[一]Ready?No!

    1.前言 今天开始我会不定期写一些iOS自学的相关文章.毕竟,自己是自学开始,知道自学有哪些坑,知道自学对于开发欠缺什么,此外,加上现在的实际开发经验,希望能给自学的iOS开发者一些建议. 2.Rea ...

  6. POJ 1129

    #include<iostream> #include<stdio.h> #include<string> #define MAXN 60 using namesp ...

  7. Scala中使用implict 扩展现有类的方法

    Scala中implict的一种用法就是扩展现有类的方法,有点类似于.Net中的扩展方法(MS对扩展方法的介绍:扩展方法使你能够向现有类型“添加”方法,而无需创建新的派生类型.重新编译或以其他方式修改 ...

  8. 11、使用xamarin实现全屏播放rtmp之类的直播视频

    直播类的app大部分都是以rtmp hls播放为主.目前主流的app解决方案大部分是引入ijkplayer 这个是基于ffmpeg中的ffplayer实现的. 众所周知ffmpeg的解码能力是一流的. ...

  9. Mac在终端用命令装载dmg文件

    今天碰到个问题,下载了一个dmg文件,然后双击/右键安装,一点反应都没有.一开始以为是电脑的缘故,重启,依旧没有反应,然后想到用终端装载试试. 打开终端,输入命令: hdiutil attach we ...

  10. docker容器网络通信原理分析(转)

    概述 自从docker容器出现以来,容器的网络通信就一直是大家关注的焦点,也是生产环境的迫切需求.而容器的网络通信又可以分为两大方面:单主机容器上的相互通信和跨主机的容器相互通信.而本文将分别针对这两 ...