UDP通信流程步骤:

服务端: 等待(被动)接收发送

1: 创建 socket:  socket()

2: 绑定端口:      bind()

3: 读取消息:      read()

4: 发送消息:      write()

5: 关闭套接字:  close()

客户端:主动发送接收

1: 创建 socket:   socket()

2: 发送数据:        write()

3: 接受结果:         read()

4: 关闭套接字:     close()

UDP通信流程图:

UDP通信

1.没有固定连接

2.客户端发完包,就不管了,也不知道服务端是不是收到了

UDP创建套接字、绑定套接字的方式与TCP一样,可参考TCP。

发送消息

sendto(int sockfd, void* buf,  size_t len,  int flags,

struct sockaddr *to,  socklen_t tolen);

sockaddr由sockaddr_in转换。

UDP 没有accept创建新的通信fd,需要指定目标地址

函数可以用于TCP通信,后面两个参数会忽略

接收消息

recvfrom(int sockfd, void *buf,  size_t len ,  int flags,

sturct sockaddr *from,  socklen_t *fromlen)

UDP 没有 accept 函数来获取对端地址,这里增加了2个参数

函数可以用于TCP通信

例子:

#include<stdio.h>

#include<stdlib.h>

#include<string.h>

#include<sys/socket.h>

#include<netinet/in.h>

#include<arpa/inet.h>

#include<unistd.h>

#define SRV_PORT  8888

#define CLT_PORT  6666

void Udp_server()

{

int fd;

int iRet;

struct sockaddr_in addr;

socklen_t addrlen = sizeof(addr);

//创建套接字

fd = socket(PF_INET, SOCK_DGRAM, 0);

if (fd < 0)

{

perror("Fail to socket!");

return;

}

addr.sin_family = AF_INET;

addr.sin_port = htons(SRV_PORT);

addr.sin_addr.s_addr = htonl(INADDR_ANY);

//绑定,可以让客户端知道通过什么IP地址和端口号来连接

iRet = bind(fd, (struct sockaddr*)&addr, addrlen);

if (iRet)

{

perror("Fail to bind!");

close(fd);

return;

}

struct sockaddr_in srcaddr;

char szBuf[1000];

char szMsg[] = "[UDP]I Received!";

while(1)

{

//接收,并获取客户端的IP和端口号(struct sockaddr*)&srcaddr

memset(szBuf, 0, 1000);

iRet = recvfrom(fd, szBuf, 1000, 0, (struct sockaddr*)&srcaddr, &addrlen);

if (iRet < 0)

{

perror("Fail to recvfrom!");

break;

}

printf("Recv:%s\n", szBuf);

//发送

fprintf(stderr,"Echo:");

scanf("%s",szMsg);

sendto(fd, szMsg, strlen(szMsg), 0, (struct sockaddr*)&srcaddr, addrlen);

}

close(fd);

return;

}

void Udp_client()

{

int fd;

int iRet;

struct sockaddr_in addr;

socklen_t  addrlen = sizeof(addr);

fd = socket(PF_INET, SOCK_DGRAM, 0);

if (fd < 0)

{

perror("Fail to socket!");

return;

}

/*

addr.sin_family = AF_INET;

addr.sin_port = htons(CLT_PORT);

addr.sin_addr.s_addr = htonl(INADDR_ANY);

iRet = bind(fd, (struct sockaddr*)&addr, addrlen);

if (iRet)

{

perror("Fail to bind!");

close(fd);

return;

}

*/

struct sockaddr_in srvaddr;

char szIp[16] = ;

int  port;

fprintf(stderr, "Input server IP and port:");

scanf("%s%d", szIp, &port);

srvaddr.sin_family = AF_INET;

srvaddr.sin_port = htons((short)port);

srvaddr.sin_addr.s_addr = inet_addr(szIp);

char szBuf[100];

char szRcv[1000];

while(1)

{

memset(szBuf, 0, 100);

fprintf(stderr, "->");

read(STDIN_FILENO, szBuf, 100);

sendto(fd, szBuf, strlen(szBuf), 0, (struct sockaddr*)&srvaddr, addrlen);

memset(szRcv, 0, 1000);

iRet = recvfrom(fd, szRcv, 1000, 0, (struct sockaddr*)&srvaddr, &addrlen);

if (iRet < 0)

{

perror("Fail to recvfrom!");

break;

}

printf("Recv:%s\n", szRcv);

}

close(fd);

return;

}

int main(int argc, char** argv)

{

if (argc!=2

|| (strcmp(argv[1], "s") && strcmp(argv[1], "c"))

)

{

printf("Usage: %s [ c | s ]\n", argv[0]);

printf("\t s: For start udp server\n");

printf("\t c: For start udp client\n");

return 0;

}

if (argv[1][0] == 's')

{

Udp_server();

}

else if (argv[1][0] == 'c')

{

Udp_client();

}

return 0;

}

UDP打洞:

打洞就是让对方不需要通过服务器的转换,直接使用公网IP地址进行通信。

解析:

一般的主机都是使用私有IP地址(在不同的局域网中私有IP地址可以重复的),对方主机通过路由路转换成公有IP地址,再进行通信。

打洞,当主机A向某一服务器发送数据时,此时在服务器上,显示的是主机A的公网IP地址和端口号,要做的就是将该IP地址和端口号再一次返回给主机A。那么主机A就知道了自己的公网IP和端口号。此时主机B就可以直接对主机A发送数据,进而可以相互通信。

28UDP的更多相关文章

随机推荐

  1. WPF - 绑定及惯用法(一)

    写在前面:这仍然是一些没有经过严格审阅的文字.虽然我的确执行了初稿.复稿以及审阅等一系列用以保证文章质量的方法,但是仍然担心其中是否有错误.希望您能帮助指出,以在下一次我在版本更新时进行修正.所有的错 ...

  2. 详谈JavaScript 匿名函数及闭包

    1.匿名函数函数是JavaScript中最灵活的一种对象,这里只是讲解其匿名函数的用途.匿名函数:就是没有函数名的函数. 1.1 函数的定义,首先简单介绍一下函数的定义,大致可分为三种方式 第一种:这 ...

  3. Unity中对SQL数据库的操作

    在Unity中,我们有时候需要连接数据库来达到数据的读取与储存.而在.NET平台下,ADO.NET为我们提供了公开数据访问服务的类.客户端应用程序可以使用ADO.NET来连接到数据源,并查询,添加,删 ...

  4. axios如何进行跨域以及对返回格式为回调函数字符串的处理

    自从vue2.0开始不对vue-resouce进行维护了,转而用axios进行代替,axios的官方文档写的很详细,附上链接一枚:http://www.jianshu.com/p/df464b26ae ...

  5. Layui前后台交互数据获取java

    Layui简介 Layui是一款适用于后台程序员的UI框架,学习成本低.Json数据格式交互前后台,并且也相当适用单页面开发.有兴趣的朋友可以看看layui官网. Layui前后台数据交互 layui ...

  6. JZOJ.5288【NOIP2017模拟8.17】球场大佬

    Description       每天下午,古猴都会去打羽毛球.但是古猴实在是太强了,他必须要到一些比较强的场去打.但是每个羽毛球场都有许多的人排着队,每次都只能上四个人,每个人都有自己的能力值,然 ...

  7. shell命令发送网站请求

    GET请求:curl "http://192.168.87.195:8888/refresh" POST请求:curl -d "name=value" &quo ...

  8. Java+selenium+Fitnesse

    刚开始接触selenium是进公司后,老大给我们培训了一下UI自动化(其实也不叫培训啦,就是让我们知道有这么个东西吧,我这么说,老大看到得打人了,哈哈).要进行自动化测试,当然就得搭建一个自动化测试框 ...

  9. 二项分布。计算binomial(100,50,0.25)将会产生的递归调用次数(算法第四版1.1.27)

    算法第四版35页问题1.1.27,估计用一下代码计算binomial(100,50,0.25)将会产生的递归调用次数: public static double binomial(int n,int ...

  10. 经常会碰到css的bug

    1.a标签做为空的时候,只做链接的时候,ie是无法点击链接. a{background:url(about:blank);} ;filter:alpha(opacity=0);} 2.给导航做下拉菜单 ...