poll和kqueue的使用(网络编程的多路复用)

网络编程的多路复用

mac下的kqueue的使用!!!

  • 客户端和服务端通信
	//服务端
	#include<stdio.h>
	#include<sys/types.h>          /* See NOTES */
	#include<sys/socket.h>
	#include<sys/unistd.h>
	#include<arpa/inet.h>
	#include <netinet/in.h>
	#include<string.h>
	#include<stdlib.h>
	#include <unistd.h>
	#include <arpa/inet.h>
	#include <sys/socket.h>
	#include <sys/event.h>
	#include <sys/time.h>
	
	
	//程序退出的标志
	int terminate = 0 ;
	//信号处理函数
	static void sig_handler(int sig)
	{
	    switch (sig)
	    {
	        case SIGINT:
	            printf("kill \n");
	            terminate = 1;
	            break;
	        default:
	          break;
	    }
	}
	
	int main(int argc , char * argv[])
	{
	    if(argc != 3)    //在运可执行文件时,要在后面先加IP和端口号
	    {
	        printf("输入有误\n");
	        return 0;
	    }
	    signal(SIGINT,sig_handler);
	    signal(SIGTERM,sig_handler);
	    //创建套接字
	    int sockfd = socket(AF_INET,SOCK_STREAM,0);  //IPV4协议族,流式套接字
	    if(sockfd == -1)
	    {
	        perror("socket error\n");
	        return 0;
	    }
	    //允许本地端口重用
	    int on =1 ;
	    setsockopt(sockfd,SOL_SOCKET,SO_RANDOMPORT,&on,sizeof(on));
	
	    //如果你想要让其他人来主动连接或联系你,你就需要bind一个地址,并且吧这个地址告诉其他人
	    //绑定服务端自己的网络地址
	    //定义并设置服务端自己的网络地址
	    struct sockaddr_in saddr;
	    memset(&saddr,0,sizeof(saddr)); //把addr变量内存所有的字节设置为0
	    saddr.sin_family = AF_INET;     //指定协议族
	    saddr.sin_port = htons(atoi(argv[2])); 
	    //指定端口号 htons是把16bits的主机字节序 转换成 网络字节序  atoi是将字符串转换成整型
	    inet_aton(argv[1],&saddr.sin_addr);  
	    //ip地址  inrt_aton是把点分式转换成网络地址
	    //绑定自己ipv4的网络地址绑定到一个sockfd上去
	    int r = bind(sockfd,(struct sockaddr *)&saddr , sizeof(saddr));
	    if(r==-1)
	    {
	        perror("bind error");
	        close(sockfd);
	        return 0;
	    }
	
	    //监听
	    r = listen(sockfd,3);
	    if(r==-1)
	    {
	        perror("listen error");
	        close(sockfd);
	        return 0;
	    }
	    int connfd;
	    struct sockaddr_in caddr;   //用来保存向我(服务端)发送连接请求的客户端的网络地址
	    int len = sizeof(caddr);
	    connfd = accept(sockfd,(struct sockaddr *)&caddr , &len);
	
	    if(connfd == -1)
	    {
	        perror("accept error");
	        close(sockfd);
	        return 0;
	    }
	    printf("ip地址为%s,端口号为%hu的客户端与我建立连接\n",inet_ntoa(caddr.sin_addr),caddr.sin_port);
	
	    char buf[50]="";
	    
	    while(!terminate)
	    {
	       
	        const static int FD_NUM = 2; // 要监视多少个文件描述符
	    
	        int kq = kqueue(); // kqueue对象
	        
	        // kqueue的事件结构体,不需要直接操作
	        struct kevent changes[FD_NUM]; // 要监视的事件列表
	        struct kevent events[FD_NUM]; 
	        // kevent返回的事件列表(参考后面的kevent函数)
	        
	        int f1 = STDIN_FILENO;
	        int f2=connfd;
	        
	        // 在changes列表中注册标准输入流的读事件 以及 标准输出流的写事件
	        // 最后一个参数可以是任意的附加数据(void * 类型),在这里给事件附上了当前的文件描述符,后面会用到
	        EV_SET(&changes[0], f1, EVFILT_READ, EV_ADD | EV_ENABLE, 0, 0, &f1); 
	        EV_SET(&changes[1], f2, EVFILT_READ, EV_ADD | EV_ENABLE, 0, 0, &f2);
	        
	        // 进行kevent函数调用,如果changes列表里有任何就绪的fd,则把该事件对应的结构体放进events列表里面
	        // 返回值是这次调用得到了几个就绪的事件 (nev = number of events)
	        int nev = kevent(kq, changes, FD_NUM, events, FD_NUM, NULL); // 已经就绪的文件描述符数量
	        for(int i=0; i<nev; i++){
	            struct kevent event = events[i]; // 一个个取出已经就绪的事件
	        
	            int ready_fd = *((int *)event.udata); 
	            // 从附加数据里面取回文件描述符的值
	            if( ready_fd == f1 )
	            {
	                // 标准输入事件
	                scanf("%s",buf);
	                printf("%s\n",buf);
	
	            }
	            else if( ready_fd == f2)
	            {
	                // connfd事件
	                read(connfd,buf,50);
	                printf("%s\n",buf);
	                
	            }
	        }
	
	    }
	    close(connfd);
	    close(sockfd);
	    return 0;
	}

	//客户端
	#include<stdio.h>
	#include<sys/types.h>          /* See NOTES */
	#include<sys/socket.h>
	#include<sys/unistd.h>
	#include<arpa/inet.h>
	#include <netinet/in.h>
	#include<string.h>
	#include<stdlib.h>
	 #include <unistd.h>
	 #include <arpa/inet.h>
	#include <sys/socket.h>
	
	
	// ./xxxx 服务端ip 服务端port
	int main(int argc , char * argv[])
	{
	    if(argc!= 3)
	    {
	        printf("输入有误\n");
	        return 0;
	    }
	    int sockfd = socket(AF_INET,SOCK_STREAM,0);
	    if(sockfd == -1)
	    {
	        perror("socket error\n");
	        return 0;
	    }
	
	    //允许本地端口重用
	    //int on = 1;
	    #if 0
	    struct sockaddr_in myaddr;  //自己的网络地址
	    memset(&myaddr,0,sizeof(myaddr));
	    myaddr.sin_family = AF_INET;
	    myaddr.sin_port = htons(5555);
	    inet_aton("172.20.10.5",&myaddr.sin_addr);
	    bind(sockfd,(struct sockaddr *)&myaddr , sizeof(myaddr));
	    #endif
	
	    struct sockaddr_in addr;
	    memset(&addr , 0 , sizeof(addr));
	    addr.sin_family = AF_INET;
	    addr.sin_port = htons(atoi(argv[2]));
	    inet_aton(argv[1], &addr.sin_addr);
	
	    int r = connect(sockfd,(struct sockaddr *)&addr,sizeof(addr));
	    if(r==-1)
	    {
	        perror("connect error");
	        close(sockfd);
	        return 0;
	    }
	
	    sleep(5);
	    char buf[]="hello";
	    write(sockfd, buf , sizeof(buf));
	    char buf1[100];
	    read(sockfd,buf1,100);
	    printf("%s\n",buf1);
	    close(sockfd);
	    return 0;
	}

win下的poll使用

	//服务端
	#include<stdio.h>
	#include <sys/types.h>			/* See NOTES */
	#include <sys/socket.h>
	#include <unistd.h>
	#include <arpa/inet.h>
	#include <string.h>
	#include <stdlib.h>
	#include <poll.h>
	
	//./xxx 服务端ip 服务端端口
	// select 实现多路复用
	int main(int argc ,char *argv[])
	{
		if(argc != 3)
		{
			printf("输入有误\n");
			return 0;
		}
	
		//1,创建套接字
		int sockfd = socket(AF_INET,SOCK_STREAM,0);
		if(sockfd == -1)
		{
			perror("socket error");
			return 0;
		}
		
		//允许本地端口重用
		int on = 1;
		setsockopt(sockfd,SOL_SOCKET,SO_REUSEPORT,&on,sizeof(on));
	
		//2,绑定服务端自己的网络地址
		//定义并设置服务端自己的网络地址
		struct sockaddr_in saddr;
		memset(&saddr,0,sizeof(saddr));//把addr变量内存所有字节设置为 0
		
		saddr.sin_family = AF_INET;
		saddr.sin_port = htons(atoi(argv[2]));//指定端口号,并进行字节序的转换
		inet_aton(argv[1],&saddr.sin_addr);//指定IP地址
	
		int r = bind(sockfd,(struct sockaddr *)&saddr,sizeof(saddr));
		if(r==-1)
		{
			perror("bind error");
			close(sockfd);
			return 0;
		}
	
		//3,监听
		r = listen(sockfd,3);
		if(r==-1)
		{
			perror("listen error");
			close(sockfd);
			return 0;
		}
		int connfd;
		struct sockaddr_in caddr;//用来保存向我(服务端)发送连接请求的客户端的网络地址
		int len = sizeof(caddr);
		connfd = accept(sockfd,(struct sockaddr*)&caddr,&len);//阻塞的
		if(connfd == -1)
		{
			perror("accept error");
			close(sockfd);
			return 0;
		}
	
		printf("ip地址为%s,端口号为%hu的客户端与我建立了连接\n",inet_ntoa(caddr.sin_addr),
				caddr.sin_port);
	
		/*
		char buf1[10] = "";
		scanf("%s",buf1);//从标准输入设备获取数据
		printf("%s\n",buf1);
	
		
		read(connfd,buf1,10);//从网络客户端获取数据
		printf("%s\n",buf1);
	
		//上面这种做法是有先后顺序的。不是平行的
		*/
	
		//IO复用:同时监听多个文件描述符,谁先就绪(有数据),我就先处理谁(读取数据)
	
		char buf[50] = "";
		while(1)
		{
			//我要监听 connfd 和 STDIN_FILENO 这两个文件描述符
			//1,定义一个 struct pollfd 数组,两个元素。并赋值
			struct pollfd fds[2];
			fds[0].fd = connfd;
			fds[0].events = POLLIN;
	
			fds[1].fd = STDIN_FILENO;
			fds[1].events = POLLIN;
	
			int n = poll(fds,2,10000);
	
			if(n<0)
			{
				perror("poll error");
				break;
			}
			else if(n == 0)
			{
				printf("超时了\n");
			}
			else//有文件描述符就绪了
			{
				//需要判断哪个文件描述符就绪
				if(POLLIN & fds[0].revents)//connfd可读了
				{
					read(connfd,buf,50);
					printf("%s\n",buf);
				}
	
				if(POLLIN & fds[1].revents)//STDIN_FILENO可读了
				{
					scanf("%s",buf);
					printf("%s\n",buf);
				}
				break;
			}
		}
		char buf2[] = "bey";
		write(connfd,buf2,strlen(buf2));
		close(connfd);//通信完成关闭 connfd
	
		close(sockfd);
		return 0;
	}



  //客户端
	#include<stdio.h>
	#include <sys/types.h>			/* See NOTES */
	#include <sys/socket.h>
	#include <unistd.h>
	#include <arpa/inet.h>
	#include <string.h>
	#include <stdlib.h>
	
	//./xxxx 服务端ip 服务端port
	int main(int argc,char *argv[])
	{
		if(argc != 3)
		{
			printf("输入有误\n");
			return 0;
		}
	
		int sockfd = socket(AF_INET,SOCK_STREAM,0);
		if(sockfd == -1)
		{
			perror("socket error");
			return 0;
		}
		//允许本地端口重用
		int on = 1;
		//setsockopt(sockfd,SOL_SOCKET,SO_REUSEPORT,&on,sizeof(on));
		//客户端绑定网络地址。可选的
		#if 0
		struct sockaddr_in myaddr;//自己的网络地址		
		//void *memset(void *s, int c, size_t n); //用于设置一块内存为指定值c
		memset(&myaddr,0,sizeof(myaddr));//把addr变量内存所有字节设置为 0
		myaddr.sin_family = AF_INET;
		myaddr.sin_port = htons(5555);//指定端口号,并进行字节序的转换
		inet_aton("192.168.100.5",&myaddr.sin_addr);//指定自己的IP地址
		bind(sockfd,(struct sockaddr*)&myaddr,sizeof(myaddr));
		#endif
	
		struct sockaddr_in addr;//服务端的网络地址		
		//void *memset(void *s, int c, size_t n); //用于设置一块内存为指定值c
		memset(&addr,0,sizeof(addr));//把addr变量内存所有字节设置为 0
		addr.sin_family = AF_INET;
		addr.sin_port = htons(atoi(argv[2]));//指定端口号,并进行字节序的转换
		inet_aton(argv[1],&addr.sin_addr);//指定IP地址
	
		int r = connect(sockfd,(struct sockaddr*)&addr,sizeof(addr));
		if(r == -1)
		{
			perror("connect error");
			close(sockfd);
			return 0;
		}
	
		//进行通信
	
		sleep(5);//连接成功5秒之后再发送数据
	
		char buf[] = "hello";
		write(sockfd,buf,strlen(buf));
	
		char buf1[100];
		read(sockfd,buf1,100);
		printf("%s\n",buf1);
		close(sockfd);
		return 0;
	}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值