网络编程学习日记-----UDP协议、IO模型、阻塞IO

一、网络编程通信协议。 -- UDP协议。

1、UDP协议特性是什么? UDP协议全称"User Data Protocol/用户数据包协议",这种协议特点基于无连接的通信,所以UDP协议的代码中是没有connect()和accept()。

2、UDP协议设计步骤? 详细参考: UDP协议的设计流程.jpg

客户端:

#include "head.h"

int main(int argc,char *argv[]) // ./Jack 192.168.19.3 50001

{ //1. 创建UDP协议套接字。

int sockfd = socket(AF_INET,SOCK_DGRAM,0);

//2. 直接写信。
struct sockaddr_in srvaddr;
socklen_t len = sizeof(srvaddr);
bzero(&srvaddr,len);
​
srvaddr.sin_family = AF_INET;
srvaddr.sin_port = htons(atoi(argv[2]));
inet_pton(AF_INET,argv[1],&srvaddr.sin_addr);
​
char buf[100];
while(1)
{
    bzero(buf,sizeof(buf));
    fgets(buf,sizeof(buf),stdin);
    sendto(sockfd,buf,strlen(buf),0,(struct sockaddr *)&srvaddr,len);
​
    if(strncmp(buf,"quit",4) == 0)
    {
        break;
    }
}
​
//3. 销毁信箱
close(sockfd);
​
return 0;

}

服务器:

 

#include "head.h"

int main(int argc,char *argv[]) // ./Rose 50001

{ //1. 创建UDP协议套接字。

int sockfd = socket(AF_INET,SOCK_DGRAM,0);

//2. 绑定IP地址到套接字上。
struct sockaddr_in srvaddr;
socklen_t len = sizeof(srvaddr);
bzero(&srvaddr,len);
​
srvaddr.sin_family = AF_INET;
srvaddr.sin_port = htons(atoi(argv[1]));
srvaddr.sin_addr.s_addr = htonl(INADDR_ANY);
​
bind(sockfd,(struct sockaddr *)&srvaddr,len);
​
//3. 不断等待对方写信给我。
char buf[100];
struct sockaddr_in cliaddr;
bzero(&cliaddr,sizeof(cliaddr));
​
while(1)
{
    bzero(buf,sizeof(buf));
    recvfrom(sockfd,buf,sizeof(buf),0,(struct sockaddr *)&cliaddr,&len);
​
    printf("%s:%s",inet_ntoa(cliaddr.sin_addr),buf);
​
    if(strncmp(buf,"quit",4) == 0)
    {
        break;
    }
}
​
//4. 销毁信箱
close(sockfd);
​
return 0;

}

二、IO模型。

1、什么是IO模型?

在网络编程中,对于套接字,我们可以进行读写的操作,例如connfd,那么我们可以recv(connfd),也可以send(connfd)。

当进程在读取套接字时,revc(connfd),那么当这个套接字上没有数据时,这个函数就会阻塞等待。

如果需要读取多个套接字的数据时,就不能阻塞等待,就要使用IO模型

 

2.IO模型有哪些:?

一共有四种IO模型分别是,阻塞IO、非阻塞IO、多路复用、信号驱动、

三、非阻塞IO模型

 1.    阻塞IO:当多个套接字给我发送数据时,我会先询问第一个套接字 有没有数据到达,如果有则读取,则阻塞等待以一个套接字的数据

     非阻塞IO:当多个套接字给我发送数据时,我会先询问第一个套接字 有没有数据到达,如果有则读取,没有则读取下一个。   

2.如何设置非阻塞

      正常情况下我们使用用一个函数去读取数据时,默认都是阻塞的。例如:read(fd)accept、recv、recvfrom。实际上这些函数并不是造成阻塞的原因,而是这些函数中的套接字/文件描述符阻塞。

 1)创建一个新的文件描述符/套接字   ----------》默认为时阻塞的

2)添加非阻塞属性给套接字 /文件描述符 ------------》这时就不是阻塞了

3)那么再调用read(fd)accept()等就不会阻塞了

3.如何添加非阻塞给套接字/文件描述符?  --------fcntl()------man 2 fcntl

使用方法:

       #include <unistd.h>
       #include <fcntl.h>

       int fcntl(int fd, int cmd, ... /* arg */ );
 

参数:

     fd:  需要设置属性的套接字/文件描述符

     cmd: F_GETFL (void)   -------代表后面那个参数不用填

                F_SETFL (int)      --------代表后面那个参数需要填

     arg:   O_RDONLY, O_WRONLY, O_RDWR

                 O_NONBLOCK    ----非阻塞属性

 

返回值    成功: 

                          F_GETFL   //返回文件属性

                          F_SETFL  //返回0

                失败:

                         -1

4.假设有一个文件描述符叫fd,那么如何给这个fd添加非阻塞属性?

     ①获取fd原来的属性

              int state=fcntl (fd,F_GETFL);

     ②在原来的属性上添加非阻塞的属性

              state=state | O_NONBLOCK   

      ③将state设置到fd上

              fcntl(fd,F_SETFL,state);

 

非阻塞eg:

client 

#include "head.h"

int main(int argc,char *argv[])   //  ./Jack 192.168.19.5 50001
								  //  ./Jack 服务器的IP地址 端口号
{
	//1. 创建TCP协议套接字。
	int sockfd;
	sockfd = socket(AF_INET,SOCK_STREAM,0);

	//2. 打电话。
	struct sockaddr_in srvaddr;
	socklen_t len = sizeof(srvaddr);
	bzero(&srvaddr,len);

	srvaddr.sin_family = AF_INET;
	srvaddr.sin_port = htons(atoi(argv[2]));
	inet_pton(AF_INET,argv[1],&srvaddr.sin_addr);

	int ret = connect(sockfd,(struct sockaddr *)&srvaddr,len);
	if(ret == -1)
	{
		printf("connect error!\n");
	}

	//3. 不断发送数据给服务器。
	char buf[100] = {0};
	while(1)
	{
		bzero(buf,sizeof(buf));
		fgets(buf,sizeof(buf),stdin);
		send(sockfd,buf,strlen(buf),0);

		if(strncmp(buf,"quit",4) == 0)
		{
			break;
		}
	}

	//4. 挂断。
	close(sockfd);

	return 0;
}

sever

#include "head.h"

int main(int argc, char *argv[])
{
	//1. 创建TCP套接字
	int sockfd = socket(AF_INET, SOCK_STREAM, 0);

	//2. 绑定IP地址
	struct sockaddr_in srvaddr;
	socklen_t len = sizeof(srvaddr);
	bzero(&srvaddr, len);

	srvaddr.sin_family = AF_INET;
	srvaddr.sin_port = htons(atoi(argv[1]));
	srvaddr.sin_addr.s_addr = htonl(INADDR_ANY);

	bind(sockfd, (struct sockaddr *)&srvaddr, len);

	//3. 设置监听套接字。
	listen(sockfd, 20); //sockfd -> 阻塞(监听套接字)

	//4. 添加非阻塞到监听套接字上。
	int state;
	state = fcntl(sockfd, F_GETFL);
	state |= O_NONBLOCK;
	fcntl(sockfd, F_SETFL, state); //sockfd -> 非阻塞(监听套接字)

	//5. 等待连接
	struct sockaddr_in cliaddr;
	bzero(&cliaddr, sizeof(cliaddr));

	int connfd;
	char buf[100] = {0};
	while (1)
	{
		
		connfd = accept(sockfd, (struct sockaddr *)&cliaddr, &len);
		if (connfd >= 0)
		{
			while (1)
			{
				bzero(buf, sizeof(buf));
				recv(connfd, buf, sizeof(buf), 0);
				printf("from:%s", buf);
				if (strncmp(buf, "quit", 4) == 0)
				{
					exit(0);
				}
			}
		}

	}

	//6. 关闭套接字。
	close(connfd);
	close(sockfd);

	return 0;
}

 

 

使用非阻塞IO、内核链表实现聊天室

sever

#include "head.h"
#include "kernel_list.h"
struct list_node
{
    int connfd;            //存放连接到服务器上的已连接套接字。
    struct list_head list; //统计当前连接到服务器中的人数。
};

struct list_node *init_list_head()
{

    struct list_node *head = malloc(sizeof(struct list_node));
    INIT_LIST_HEAD(&(head->list));
    return head;
}

void insert_data_to_list(struct list_node *head, int connfd)
{
    struct list_node *new = malloc(sizeof(struct list_node));
    new->connfd = connfd;
    list_add_tail(&(new->list), &(head->list));
    return;
}

int delete_list_node(struct list_node *head, int connfd)
{
    struct list_node *p = NULL;
    struct list_node *q = NULL;

    list_for_each_entry_safe(p, q, &(head->list), list)
    {
        if (p->connfd == connfd)
        {
            list_del(&(p->list));

            free(p);
            return 0;
        }
    }
    return -1;
}

void client_quest(struct list_node *head)
{   
    char buf[100] = {0};
    //遍历整个链表

    struct list_node *p = NULL;

    list_for_each_entry(p, &(head->list), list)
    {

        bzero(buf, sizeof(buf));

        if (recv(p->connfd, buf, sizeof(buf), 0) >= 0)
        {
            //说明者为顾客说话了
            printf("from %dclient:%s", p->connfd, buf);

            //如果该用户选择退出聊天室

            if (strncmp(buf, "quit", 4) == 0)
            {
                //让用户结束,将自己从链表中删除
                delete_list_node(head, p->connfd);
                return ;

            }
        }
    }

    return;
}

void delete_list(struct list_node *head)
{
    struct list_node *p = NULL;
    struct list_node *q = NULL;
    list_for_each_entry_safe(p, q, &(head->list), list)
    {
        //1. 从链表中将节点脱离了
        list_del(&(p->list));

        //2. 释放空间。
        free(p);
    }

    free(head);
    return;
}

int main(int argc, char *argv[]) // ./server 50001
{
    //0. 初始头节点。
    struct list_node *head = NULL;
    head = init_list_head();

    //1. 创建TCP套接字。
    int sockfd = socket(AF_INET, SOCK_STREAM, 0);

    //2. 绑定IP地址。
    struct sockaddr_in srvaddr;
    socklen_t len = sizeof(srvaddr);
    bzero(&srvaddr, len);

    srvaddr.sin_family = AF_INET;
    srvaddr.sin_port = htons(atoi(argv[1]));
    srvaddr.sin_addr.s_addr = htonl(INADDR_ANY);

    bind(sockfd, (struct sockaddr *)&srvaddr, len);

    //3. 设置监听套接字。
    listen(sockfd, 5);

    //4. 设置非阻塞给监听套接字
    int state;
    state = fcntl(sockfd, F_GETFL);
    state |= O_NONBLOCK;
    fcntl(sockfd, F_SETFL, state);

    //5. 非阻塞地等待客户端的连接
    struct sockaddr_in cliaddr;
    bzero(&cliaddr, sizeof(cliaddr));
    int connfd, ret, i;
    char buf[100];

    while (1)
    {

        sleep(1);
        //6. 老板在门口非阻塞地迎宾。

        connfd = accept(sockfd, (struct sockaddr *)&cliaddr, &len);

        //7. 如果真的有人进来吃饭,那么就在数组中安排桌位。
        if (connfd > 0)
        {
            //8. 将这个新人设置为非阻塞。
            state = fcntl(connfd, F_GETFL);
            state |= O_NONBLOCK;
            fcntl(connfd, F_SETFL, state);

            //9. 将这个人储存到链表中。
            insert_data_to_list(head, connfd);

            //10.将新连接到服务器的那个人的IP地址打印出来。
            printf("新用户连接进来: %s(%d)\n", inet_ntoa(cliaddr.sin_addr), connfd);
        }

        //11. 如果没有人进来吃饭,那么老板就会开始询问那个链接到服务器的客户端有没有话说。
        client_quest(head);
    }

    delete_list(head);

    return 0;
}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值