一、引言
在实际情况中,人们往往遇到多个客户端连接服务器端的情况。由于之前介绍的函数如connect,recv,send等都是阻塞性函数,若资源没有充分准备好,则调用该函数的进程将进入睡眠状态,这样就无法处理I/O多路复用的情况了。
本文给出两种I/O多路复用的方法:fcntl(),select()。可以看到,由于Linux中把socket当作一种特殊的文件描述符,这给用户的处理带来很大方便。
二、fcntl
1)非阻塞I/O:可将cmd设为F_SETFL,将lock设为O_NONBLOCK
2)信号驱动I/O:可将cmd设为F_SETFL,将lock设为O_ASYNC.
#include<sys/types.h>
#include<sys/socket.h>
#include<sys/wait.h>
#include<sys/time.h>
#include<sys/ioctl.h>
#include<netinet/in.h>
#defineMAX_CONNECTED_NO10
structsockaddr_inserver_sockaddr,client_sockaddr;
intsin_size,recvbytes,flags;
if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1){
printf("socketsuccess!,sockfd=%d
server_sockaddr.sin_family=AF_INET;
server_sockaddr.sin_port=htons(SERVPORT);
server_sockaddr.sin_addr.s_addr=INADDR_ANY;
bzero(&(server_sockaddr.sin_zero),8);
if(bind(sockfd,(structsockaddr*)&server_sockaddr,sizeof(structsockaddr))==-1){
if(listen(sockfd,BACKLOG)==-1){
if((flags=fcntl(sockfd,F_SETFL,0))<0)
if(fcntl(sockfd,F_SETFL,flags)<0)
sin_size=sizeof(structsockaddr_in);
if((client_fd=accept(sockfd,(structsockaddr*)&client_sockaddr,&sin_size))==-1){//服务器接受客户端的请求,返回一个新的文件描述符
if((recvbytes=recv(client_fd,buf,MAXDATASIZE,0))==-1){
if(read(client_fd,buf,MAXDATASIZE)<0){
printf("receivedaconnection:%s",buf);
[root@localhostnet]#https://www.fruan.com/post/fcntl
accept:Resourcetemporarilyunavailable
if((flags=fcntl(sockfd,F_SETFL,0))<0)
if(fcntl(sockfd,F_SETFL,flags)<0)
perror("fcntl");
[root@localhostnet]#https://www.fruan.com/post/fcntl1
listening...
可以看到,进程一直处于等待中,直到另一相关信号驱动它为止。
三、select
#include<sys/types.h>
#include<sys/socket.h>
#include<sys/wait.h>
#include<sys/time.h>
#include<sys/ioctl.h>
#include<netinet/in.h>
#defineMAX_CONNECTED_NO10
structsockaddr_inserver_sockaddr,client_sockaddr;
if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1){
printf("socketsuccess!,sockfd=%d
server_sockaddr.sin_family=AF_INET;
server_sockaddr.sin_port=htons(SERVPORT);
server_sockaddr.sin_addr.s_addr=INADDR_ANY;
bzero(&(server_sockaddr.sin_zero),8);
if(bind(sockfd,(structsockaddr*)&server_sockaddr,sizeof(structsockaddr))==-1){
if(listen(sockfd,BACKLOG)==-1){
FD_ZERO(&readfd);//将readfd清空
FD_SET(sockfd,&readfd);//将sockfd加入到readfd集合中
sin_size=sizeof(structsockaddr_in);
if(select(MAX_CONNECTED_NO,&readfd,NULL,NULL,(structtimeval(FD_ISSET(sockfd,&readfd)>0){//FD_ISSET这个宏判断sockfd是否属于可读的文件描述符。从sockfd中读入,输出到标准输出上去.
if((client_fd=accept(sockfd,(structsockaddr*)&client_sockaddr,&sin_size))==-1){//client_sockaddr:客户端地址
if((recvbytes=recv(client_fd,buf,MAXDATASIZE,0))==-1){
if(read(client_fd,buf,MAXDATASIZE)<0){
printf("receivedaconnection:%s",buf);
[root@localhostnet]#gccselect1.c-oselect1
[root@localhostnet]#https://www.fruan.com/post/select1
listening...
以上就是linuxsocket怎么实现多个客户端连接服务器端的详细内容,更多请关注主机测评网其它相关文章!
本文来源:国外服务器--linux客户端与服务器的连接
本文地址:https://www.idcbaba.com/guowai/3396.html
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 1919100645@qq.com 举报,一经查实,本站将立刻删除。



