server.c

把accept也看成是一个read类型的函数, 于是我们可以把sockfd也放入到select中

maxi标记当前客户端连接数组的最大下标

select返回值为当前已经准备就绪的fd总数

#include <unistd.h>
#include <netdb.h>
#include <errno.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <signal.h>
#include <sys/wait.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h> #define MAX(a,b) a>b?a:b; void err_quit(const char *s){
perror(s);
exit(1);
} ssize_t readn(int fd,void *buff,size_t count){
char *buffp;
ssize_t nread;
size_t nleft; buffp=(char *)buff;
nleft=count;
while(nleft > 0){
if((nread = read(fd,buffp,nleft)) < 0){
if(errno == EINTR)
continue;
else
return -1;
}else if(nread == 0)
break;
nleft -= nread;
buffp += nread;
}
return count-nleft;
} ssize_t writen(int fd,const void *buff,size_t n){
size_t nleft;
ssize_t nwritten;
const char *ptr; ptr=buff;
nleft=n;
while(nleft > 0){
if((nwritten=write(fd,ptr,nleft)) < 0){
if(nwritten < 0 && errno == EINTR)
continue;
else
return -1;
}else if(nwritten == 0)
break;
nleft -= nwritten;
ptr += nwritten;
}
return n-nleft;
} ssize_t recv_peek(int fd,void *buf,size_t len){
ssize_t ret;
while(1){
ret=recv(fd,buf,len,MSG_PEEK);
if(ret == -1 && errno == EINTR)
continue;
return ret;
}
} ssize_t readline(int fd,void *buf,size_t maxline){
ssize_t ret;
size_t nread;
size_t nleft;
char *bufp; bufp=buf;
nleft=maxline;
while(1){
ret=recv_peek(fd,buf,nleft);
if(ret < 0)
return ret;
else if(ret == 0)
return ret; nread=ret;
int i;
for(i=0;i<nread;i++){
if(bufp[i] == '\n'){
ret=readn(fd,bufp,i+1);
if(ret != i+1)
err_quit("readn"); return ret;
}
} if(nread > nleft)
err_quit("readn"); nleft -= nread;
ret=readn(fd,bufp,nread);
if(ret != nread)
err_quit("readn"); bufp += nread;
} return -1;
} int main(int argc,char *argv[]){
int i,maxi,maxfd,tmpfd,sockfd,connfd;
socklen_t len;
struct sockaddr_in addr,client;
fd_set rset,allset;
int nready,clientfd[FD_SETSIZE];
ssize_t n;
char buf[1024]; if((sockfd=socket(PF_INET,SOCK_STREAM,0)) < 0)
err_quit("sockfd"); bzero(&addr,sizeof(addr));
addr.sin_family=AF_INET;
addr.sin_addr.s_addr=htonl(INADDR_ANY);
addr.sin_port=htons(5566); int on=1;
if(setsockopt(sockfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on)) <0)
err_quit("setsockopt"); if(bind(sockfd,(struct sockaddr *)&addr,sizeof(addr))<0)
err_quit("bind"); if(listen(sockfd,10)<0)
err_quit("listen"); maxfd=sockfd;
maxi=-1;
for(i=0;i<FD_SETSIZE;i++)
clientfd[i]=-1;
FD_ZERO(&allset);
FD_SET(sockfd,&allset);
while(1){
rset=allset;
nready=select(maxfd+1,&rset,NULL,NULL,NULL); if(nready == -1){
if(errno == EINTR)
continue;
else
err_quit("select");
} if(FD_ISSET(sockfd,&rset)){
len=sizeof(client);
connfd=accept(sockfd,(struct sockaddr *)&client,&len);
if(connfd < 0)
err_quit("accept"); for(i=0;i<FD_SETSIZE;i++){
if(clientfd[i] < 0){
clientfd[i]=connfd;
break;
}
}
if(i == FD_SETSIZE)
err_quit("too many clients"); FD_SET(connfd,&allset);
if(connfd > maxfd)
maxfd=connfd;
if(i>maxi)
maxi=i;
if(--nready <= 0)
continue;
} for(i=0;i<=maxi;i++){
if((tmpfd=clientfd[i]) < 0)
continue;
if(FD_ISSET(tmpfd,&rset)){
bzero(buf,sizeof(buf));
if((n=readline(tmpfd,buf,sizeof(buf))) == 0){
close(tmpfd);
FD_CLR(tmpfd,&allset);
clientfd[i]=-1;
}
write(STDOUT_FILENO,buf,n);
writen(tmpfd,buf,n); if(--nready <= 0)
break;
}
}
}
}

client.c

fileno函数的作用把FILE *型转换电脑int fd型

此实例用了shutdown方法

#include <unistd.h>
#include <signal.h>
#include <errno.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <netinet/in.h> #define MAX(a,b) a>b?a:b; void err_quit(const char *s){
perror(s);
exit(1);
} ssize_t readn(int fd,void *buff,size_t count){
size_t nleft;
ssize_t nread;
char *ptr; nleft=count;
ptr=(char *)buff;
while(nleft > 0){
if((nread=read(fd,ptr,nleft)) < 0){
if(errno == EINTR)
continue;
else
return -1;
}else if(nread == 0)
break; ptr += nread;
nleft -= nread;
} return count - nleft;
} ssize_t writen(int fd,const void *buff,size_t n){
size_t nleft;
ssize_t nwritten;
const char *ptr; nleft=n;
while(n > 0){
if((nwritten=write(fd,buff,nleft)) < 0){
if(errno == EINTR)
continue;
else
return -1;
}else if(nwritten == 0)
break; nleft -= nwritten;
ptr += nwritten;
} return n-nleft;
}
/* if have any data then return */
ssize_t recv_peek(int fd,void *buf,size_t len){
ssize_t ret;
while(1){
ret=recv(fd,buf,len,MSG_PEEK);
if(ret == -1 && errno == EINTR)
continue;
return ret;
}
} ssize_t readline(int fd,void *buf,size_t maxline){
ssize_t ret;
size_t nread;
size_t nleft;
char *bufp; bufp=buf;
nleft=maxline;
while(1){
ret=recv_peek(fd,buf,nleft);
if(ret < 0)
return ret;
else if(ret == 0)
return ret; nread=ret;
int i;
for(i=0;i<nread;i++){
if(bufp[i] == '\n'){
ret=readn(fd,bufp,i+1);
if(ret != i+1)
err_quit("readn"); return ret;
}
} if(nread > nleft)
err_quit("nread"); nleft -= nread;
ret=readn(fd,bufp,nread);
if(ret != nread)
err_quit("readn"); bufp += nread;
} return -1;
} int main(int argc,char *argv[]){
int sockfd,maxfd;
int nready;
struct sockaddr_in servaddr;
fd_set fileset;
char buf[1024];
int stdineof; if((sockfd=socket(PF_INET,SOCK_STREAM,0)) < 0)
err_quit("socket"); bzero(&servaddr,sizeof(servaddr));
servaddr.sin_family=AF_INET;
servaddr.sin_addr.s_addr=inet_addr("127.0.0.1");
servaddr.sin_port=htons(5566); if(connect(sockfd,(struct sockaddr *)&servaddr,sizeof(servaddr)) <0)
err_quit("connect"); bzero(buf,sizeof(buf)); FD_ZERO(&fileset);
stdineof=0;
while(1){
if(stdineof == 0)
FD_SET(fileno(stdin),&fileset);
FD_SET(sockfd,&fileset); maxfd=MAX(fileno(stdin),sockfd);
nready=select(maxfd+1,&fileset,NULL,NULL,NULL); if(nready == -1){
if(errno == EINTR)
continue;
else
err_quit("select");
} if(FD_ISSET(fileno(stdin),&fileset)){
if(fgets(buf,sizeof(buf),stdin) == NULL){
stdineof=1;
shutdown(sockfd,SHUT_WR);
FD_CLR(fileno(stdin),&fileset);
continue;
}else{
writen(sockfd,buf,strlen(buf));
bzero(buf,sizeof(buf));
}
} if(FD_ISSET(sockfd,&fileset)){
int ret=readline(sockfd,buf,sizeof(buf));
if(ret == -1)
err_quit("read");
if(ret == 0){
if(stdineof == 1)
exit(0);
else
err_quit("readline");
}
fputs(buf,stdout);
bzero(buf,sizeof(buf));
}
}
}

用select实现多客户端连接的更多相关文章

  1. 配置ORACLE 客户端连接到数据库

    --================================= -- 配置ORACLE 客户端连接到数据库 --================================= Oracle ...

  2. Oracle RAC 客户端连接负载均衡(Load Balance)

    实现负载均衡(Load Balance)是Oracle RAC最重要的特性之一,主要是把负载平均分配到集群中的各个节点,以提高系统的整体吞吐能力.通常情况下有两种方式来实现负载均衡,一个是基于客户端连 ...

  3. [20171106]配置客户端连接注意.txt

    [20171106]配置客户端连接注意.txt --//在配置客户端连接时一般建议使用Net Manager工具,windows下调用执行Net Manager.--//linux下执行 netmgr ...

  4. 一个I/O线程可以并发处理N个客户端连接和读写操作 I/O复用模型 基于Buf操作NIO可以读取任意位置的数据 Channel中读取数据到Buffer中或将数据 Buffer 中写入到 Channel 事件驱动消息通知观察者模式

    Tomcat那些事儿 https://mp.weixin.qq.com/s?__biz=MzI3MTEwODc5Ng==&mid=2650860016&idx=2&sn=549 ...

  5. redis客户端连接到服务器的步骤

    和大多数客户端连接到服务器一样,redis-cli连接到服务器也主要分为两个阶段,请求连接阶段和数据传送阶段.具体来讲redis-cli做的事情有: 1.以socket方式建立连接: 2,选择相应的数 ...

  6. Netty源码分析 (六)----- 客户端连接接入accept过程

    通读本文,你会了解到1.netty如何接受新的请求2.netty如何给新请求分配reactor线程3.netty如何给每个新连接增加ChannelHandler netty中的reactor线程 ne ...

  7. MySql 8.0.11 客户端连接失败:2059 - Authentication plugin 'caching_sha2_password' cannot be loaded: ....

    近期,换了新笔记本,重新安装了MySql数据库和客户端工具Navicat Premium 12.我是从官网上下载的MySql数据库,版本为8.0.11,链接:https://dev.mysql.com ...

  8. netty(二)---客户端连接

    概述 先了解一下 netty 大概框架图 ,可以看到客户端的创建和服务端最大的区别 - 服务端传入两个 EventLoopGroup,客户端传入一个 EventLoopGroup - channel ...

  9. 设置MySQL客户端连接使用的字符集

    设置MySQL客户端连接使用的字符集 时间:2014-03-05    来源:服务器之家    投稿:root 考虑什么是一个"连接":它是连接服务器时所作的事情.客户端发送SQL ...

随机推荐

  1. stl应用(map)或字典树(有点东西)

    M - Violet Snow Gym - 101350M Every year, an elephant qualifies to the Arab Collegiate Programming C ...

  2. P2747 [USACO5.4]周游加拿大Canada Tour

    题目描述 你赢得了一场航空公司举办的比赛,奖品是一张加拿大环游机票.旅行在这家航空公司开放的最西边的城市开始,然后一直自西向东旅行,直到你到达最东边的城市,再由东向西返回,直到你回到开始的城市.除了旅 ...

  3. seaborn教程2——颜色调控

    原文转载 https://segmentfault.com/a/1190000014966210 Seaborn学习大纲 seaborn的学习内容主要包含以下几个部分: 风格管理 绘图风格设置 颜色风 ...

  4. CodeChef GCD2

    GCD2   Problem code: GCD2   Submit All Submissions   All submissions for this problem are available. ...

  5. php上传文件如何保证上传文件不被改变或者乱码

    很多网站上传文件都截取文件后缀,前面用时间错加后缀组成,然而一下下载的网站并不需要这样,需要保持原来的文件名,这里讲述一下 //上传操作 function uploadify(){ //var_dum ...

  6. bzoj2396 神奇的矩阵(随机化)

    Time Limit: 5 Sec  Memory Limit: 512 MB 给出三个行数和列数均为N的矩阵A.B.C,判断A*B=C是否成立. 题目可能包含若干组数据.    对于每组数据,第一行 ...

  7. Linux scp常用命令

    Linux scp命令用于Linux之间复制文件和目录. scp是 secure copy的缩写, scp是linux系统下基于ssh登陆进行安全的远程文件拷贝命令. 1.从本地复制到远程 命令格式: ...

  8. mysql处理重复数据仅保留一条记录

    目的:去除(或删除)一个表里面手机号重复的数据,但是需要保留其中一个记录,换句话说,表里面手机号不为空的数据,一个手机有且只有一条记录 表结构: CREATE TABLE `account` ( `i ...

  9. 【JAVA】 01-Java基础知识

    链接: 笔记目录:毕向东Java基础视频教程-笔记 GitHub库:JavaBXD33 测试 01-Java基础知识 一.数据类型 基本类型 包装类型 缓存池 二.String 概览 不可变的好处 S ...

  10. Linux查看关闭进程

    ps:进程的静态列表(Process status) - PID:进程号,每个进程独一无二的标识符(关闭进程需要使用) - TTY:终端所属,表明进程产生于哪一个终端,对于多用户使用的Linux服务器 ...