广播(BROADCAST)
相关概念
前面介绍的数据包发送方式只有一个接受方,称为单播
如果同时发给局域网中的所有主机,称为广播
如果同时发给局域网中的部分主机,称为组播
只有用户数据报(使用UDP协议)套接字才能广播
注意:同一个套接字只能发单播、广播、组播中的一种
如果项目中三种通信方式都需要用到,可以创建多个套接字来处理。
广播的地址
以192.168.1.0 (255.255.255.0) 网段为例,最大的主机地址192.168.1.255代表该网段的广播地址。发到该地址的数据包被所有的主机接收,255.255.255.255在所有网段中都代表广播地址
广播是如何发给所有主机的?
广播的MAC地址比较特殊,FF:FF:FF:FF:FF:FF
广播的IP地址也比较特殊,是255.255.255.255或者局域网中最大的IP地址
当交换机收到这样的数据包时,看到目的mac地址是广播地址,则就会将数据包发给局域网中的所有的主机,每台主机都可以收到广播的数据包
链路层:解析目的MAC地址,发现是广播的MAC地址,则允许通过
网络层:解析目的IP地址,发现是广播的IP地址,则也允许通过
只要保证传输层指定的端口号正确,就可以把数据通过广播的方式发送到对应的应用程序了
广播的应用:
ARP协议通过IP地址获取对方的MAC地址,就是通过广播来实现的。
广播的流程
发送者:
创建套接字 socket( ) SOCK_DGRAM
设置为允许发送广播权限 setsockopt( )
填充广播信息结构体 sockaddr_in
发送数据 sendto( )
接收者:
创建套接字 socket( ) SOCK_DGRAM
填充广播信息结构体 sockaddr_in
将套接字与广播信息结构体绑定 bind( )
接收数据 recvfrom( )
设置允许发送广播
int flag = 1;
setsockopt(sockfd, SOL_SOCKET, SO_BROADCAST, &flag, sizeof(flag));
代码实现
接收端代码:
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <stdlib.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <string.h>
#include <arpa/inet.h>
#include <unistd.h>
#define ERRLOG(msg) do{
\
printf("%s:%s:%d\n", __FILE__, __func__, __LINE__);\
perror(msg);\
exit(-1);\
}while(0);
int main(int argc, const char *argv[])
{
//入参合理性检查
if(3 != argc){
printf("Usage : %s <ip> <port>\n", argv[0]);
exit(-1);
}
//创建用户数据报套接字
int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if(-1 == sockfd){
ERRLOG("socket error");
}
//填充服务器广播信息结构体
struct sockaddr_in serveraddr;
memset(&serveraddr, 0, sizeof(serveraddr));
socklen_t serveraddr_len = sizeof(serveraddr);
serveraddr.sin_family = AF_INET;
serveraddr.sin_port = htons(atoi(argv[2]));
//ip地址需要填 当前局域网中最大的IP地址 或者 255.255.255.255
serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
//绑定
if(-1 == bind(sockfd, (struct sockaddr *)&serveraddr, serveraddr_len)){
ERRLOG("bind error");
}
//接收数据
char buff[128] = {
0};
while(1){
memset(buff, 0, 128);
if(-1 == recvfrom(sockfd, buff, 128, 0, NULL, NULL)){
ERRLOG("recvfrom error");
}
printf("buff:[%s]\n", buff);
}
close(sockfd);
return 0;
}
发送端代码:
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <stdlib.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <string.h>
#include <arpa/inet.h>
#include <unistd.h>
#define ERRLOG(msg) do{
\
printf("%s:%s:%d\n", __FILE__, __func__, __LINE__);\
perror(msg);\
exit(-1);\
}while(0);
int main(int argc, const char *argv[])
{
//入参合理性检查
if(3 != argc){
printf("Usage : %s <ip> <port>\n", argv[0]);
exit(-1);
}
//创建用户数据报套接字
int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if(-1 == sockfd){
ERRLOG("socket error");
}
//填充服务器广播信息结构体
struct sockaddr_in serveraddr;
memset(&serveraddr, 0, sizeof(serveraddr));
socklen_t serveraddr_len = sizeof(serveraddr);
serveraddr.sin_family = AF_INET;
serveraddr.sin_port = htons(atoi(argv[2]));
//ip地址需要填 当前局域网中最大的IP地址 或者 255.255.255.255
serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
//设置允许发送广播
int flag = 1;
if(-1 == setsockopt(sockfd, SOL_SOCKET, SO_BROADCAST, &flag, sizeof(flag))){
ERRLOG("setsockopt error");
}
//发送数据
char buff[