本章tcp通信,用 select() 函数监视
内容:
该程序通过创建套接字、绑定地址、监听网络和接收客户端连接等步骤实现了一个基本的 TCP 服务器。
它的主要逻辑如下:
1:创建套接字:
使用 socket() 函数创建一个套接字,返回一个套接字文件描述符 sfd。
如果创建失败,会打印错误信息并返回。
2:绑定地址信息:
使用 bind() 函数将套接字和指定的 IP 地址和端口绑定在一起。
如果绑定失败,会打印错误信息并返回。
3:监听网络:
使用 listen() 函数开始监听传入的连接请求。
如果监听失败,会打印错误信息并返回。
4:主循环:
使用 select() 函数监视套接字和已连接的客户端套接字。
如果 select() 返回值小于 0,表示出错,会打印错误信息并继续下一次循环。
遍历所有可能的文件描述符:
如果是服务器套接字 sfd,表示有新的客户端连接请求:
使用 accept() 函数接受客户端连接,并返回一个新的套接字文件描述符 cfd。
将该套接字添加到 lock 表中。
更新最大文件描述符的值 MAX。
打印客户端的 IP 地址和端口信息。
如果是客户端套接字,表示有数据传入:
使用 read() 函数读取客户端传入的数据,并存储在缓冲区 buf 中。
如果返回值为 0,表示客户端已关闭连接,打印客户端的下线信息,并从 lock 表中移除该套接字。
如果返回值大于 0,表示读取到了数据,打印读取到的数据。
5::注释::这段代码实现了一个简单的 TCP 服务器,可以接受多个客户端的连接,并能够读取客户端发送的数据。但是需要注意的是,该程序只处理一个客户端发送的一条数据,然后关闭客户端连接。如果你想要实现更复杂的功能,比如保持长连接、处理多个客户端的并发请求等,需要对代码进行进一步修改和完善。
代码如下:
#include<stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include<unistd.h>
int sfd,cfd;
int ret,i;
int flag =1;
struct sockaddr_in seraddr;
int main()
{
//1:创建套接字
sfd=socket(AF_INET,SOCK_STREAM,0);
if(sfd < 0)
{
perror("socket");
return -1;
}
//2:绑定地址信息
setsockopt(sfd,IPPROTO_IP,SO_REUSEADDR,&flag,sizeof(flag));//端口的复用
seraddr.sin_family=AF_INET;
seraddr.sin_port=htons(8888);
seraddr.sin_addr.s_addr=inet_addr("192.168.31.160");
ret=bind(sfd,(struct sockaddr *)&seraddr,sizeof(seraddr));
if(ret < 0)
{
perror("bind");
return -1;
}
//3:监听网络
ret=listen(sfd,100);
if(ret < 0)
{
perror("listen");
return -1;
}
int MAX=0;//最大值
fd_set list,lock; //创建两个表
FD_ZERO(&lock); //清空(lock)表
FD_SET(sfd,&lock); //添加套接字到表里
MAX=sfd; //赋值
while(1)
{
list=lock; //复制表
ret=select(MAX+1,&list,NULL,NULL,NULL);
if(ret < 0)
{
perror("select");
continue;
}
for(i=sfd;i<MAX+1;i++)
{
if(FD_ISSET(i,&list))//判断表里有没有套接字
{
if(i == sfd)
{
struct sockaddr_in cliaddr;
socklen_t len=sizeof(cliaddr);
cfd=accept(i,(struct sockaddr *)&cliaddr,&len);//请求链接
if(cfd < 0)
{
perror("accept");
continue;
}
FD_SET(cfd,&lock);//添加套接字到表里
if(MAX < cfd)
{
MAX=cfd;
}
printf("客户端%d已上线ip为:%s\n",cfd,inet_ntoa(cliaddr.sin_addr));
}
else
{
char buf[128]={0};
if(read(i,buf,sizeof(buf))==0)
{
printf("客户端%d已下线\n",i);
FD_CLR(i,&lock);
if(i==MAX)
{
MAX=MAX-1;
}
}
else
{
printf("read:%s\n",buf);
}
}
}
}
}
return 0;
}