28UDP
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的更多相关文章
随机推荐
- Unity中对SQL数据库的操作
在Unity中,我们有时候需要连接数据库来达到数据的读取与储存.而在.NET平台下,ADO.NET为我们提供了公开数据访问服务的类.客户端应用程序可以使用ADO.NET来连接到数据源,并查询,添加,删 ...
- Windows网络接口API函数
Windows提供了一套非常轻量级的网络函数,方便进行网络应用开发,整理出来供参考使用. The following functions are used in Windows networking: ...
- hdu 2196(求树上每个节点到树上其他节点的最远距离)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2196 思路:首先任意一次dfs求出树上最长直径的一个端点End,然后以该端点为起点再次dfs求出另一个 ...
- Linux命令之乐--rename
用来修改文件名.重命名文件,批量重命名文件rename是最好的选择. 用法:rename from to files... [root@Director test]# ls a_01 a_02 [ro ...
- js如何计算当前日期的前一个月和后一个月?
<div class="query_title_div"><img src="../../images/task/before.png"/&g ...
- 通过python3学习编码
简介 今天在写python程序的时候,遇到了编码问题,今天,我准备好好了解一下编码问题 ASCII编码 计算机是美国人发明的,最初只有不超过256字符需要编码,1字节能编码2**8个,所以ASCII编 ...
- Hibernate的检索方式--查询数据的方式
Hibernate 提供了以下几种检索对象的方式1导航对象图检索方式: 根据已经加载的对象导航到其他对象(根据已经加载的对象,导航到其他对象-例如一对多的查询)2OID 检索方式: 按照对象的 OID ...
- Struts2的CRUD操作
Struts之CRUD 1何为CRUD:CRUD代表的是一个框架的Create(增),Read(读取),update(更新),Delete(删除) 2怎么做呢?? 其实Struts2的CRUD与现实的 ...
- Pycharm取消默认的右击运行unittest方法
Pycharm取消默认的右击运行unittest方法:File-> Settings -> Tools -> Python Integrated Tools -> Defaul ...
- 【Python之路】Python目录
Python基础1 -- Python由来.Python种类.编码方式, Python基础2 -- Python运算符.数据类型.enumerate.range.for循环 python基础3 -- ...