在多进程、多任务程序中,函数执行过程中会遇到阻塞无法继续执行等问题,需要对函数进行唤醒。可以创建子进程继续执行子函数,也可以在单进程中使用select函数进行阻塞唤醒。
通过man命令查看select函数原型是
int select(int n,fd_set *read_fds,fd_set *write_fds,fd_set *except_fds,struct timeval *timeout)
下面是具体使用方法
#include<stdio.h>
#include<sys/socket.h>
#include<sys/types.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<string.h>
#include<stdlib.h>
int main(int argc, const char *argv[])
{
int sockfd;
char buf[64];
if((sockfd = socket(PF_INET,SOCK_DGRAM,0)) == -1)
{
perror("socket");
exit(1);
}
struct sockaddr_in myaddr,cli_addr;
memset(&myaddr,0,sizeof(myaddr));
myaddr.sin_family = PF_INET;
myaddr.sin_port = htons(50001);
myaddr.sin_addr.s_addr = inet_addr("192.168.4.69");
if(bind(sockfd,(struct sockaddr *)&myaddr,sizeof(myaddr)) == -1)
{
perror("bind");
exit(1);
}
int len = sizeof(cli_addr);
fd_set readfds;
FD_ZERO(&readfds);
while(1)
{
FD_SET(0,&readfds);
FD_SET(sockfd,&readfds);
select(4,&readfds,NULL,NULL,NULL);
if(FD_ISSET(0,&readfds))//如果FD_ISSET返回值为非0,说明文件描述符就绪
{
memset(buf,0,sizeof(buf));
fgets(buf,sizeof(buf),stdin);
printf("stdin:%s\n",buf);
}
if(FD_ISSET(sockfd,&readfds))
{
memset(buf,0,sizeof(buf));
recvfrom(sockfd,buf,sizeof(buf),0,(struct sockaddr *)&cli_addr,&len);
printf("client:%s\n",buf);
}
}
return 0;
}
select函数暂时介绍到这里。