网络编程(19)—— 多播的编程实现
来源:互联网 发布:淘宝乒乓球店 编辑:程序博客网 时间:2024/04/30 13:15
一、引言
多播,也叫组播,是使用UDP协议传递数据的一种方式。发送数据的主机向一个计算机组发送数据,所有注册在该计算机组中的计算机都能接受到该数据。我们经常进行的视频会议,就是用多播实现的。在网络中,一般通过路由器可实现该功能。首先,发送数据的主机发送一组数据,然后到达支持多播功能的路由器后,路由器会进行复制,将数据包复制后发给本组的其他计算机。
在软件中实现多播,需要将目标地址设置成任意D类ip地址(224.0.0.0~239.255.255.255),然后在发送主机中设置发送的socket选项,设置TTL。TTL是time_live的简称,意为生存时间,它的含义可以理解为数据包可以经过路由器的最大数。它的数值,每经过一个路由器就是减1,当减到0时,该数据包就不会再传递,因此它决定了数据包传送的距离。
TTL设置方法如下:
int send_sock; int time_live=64; ... send_sock=socket(AF_INET,SOCK_DGRAM,0); setsockopt(send_sock,IPPROTO_IP,IP_MULTICAST_TTL,(void*)&time_live,sizeof(time_live));send_sock就是我们要设置的socke对象。我们利用setsockopt函数设置支持多播的socket选项IPPROTO_TP
然后还需要在接收主机中,将主机socket加入对应的接收计算机组。 接收端利用ip_mreq类型的结构体设置多播的地址和主机地址,ip_mreq的imr_multiaddr.s_addr成员接收的是多播的地址,imr_interface.s_addr成员接收的是本地的地址。然后再利用setsockopt()函数设置接收端主机的socket即可。
int recv_sock; struct ip_mreq join_adr; ... recv_sock=socket(AF_INET,SOCK_DGRAM,0); ... join_adr.imr_multiaddr.s_addr="多播组地址信息"; join_adr.imr_interface.s_addr="加入多播组的主机地址信息"; setsockopt(recv_sock,IPPROTO_IP,IP_ADD_MEMBERSHIP,(void*)&join_adr,sizeof(join_adr));
以下示例中,我们将利用多播实现:发送端读取本地文件news.txt的内容,然后通过多播的方式进行发送,这样凡是加入该多播组的所有主机都能接收到发送端发送的数据。
二、发送端
发送端的示例代码如下:
#include<stdio.h>#include<stdlib.h>#include<unistd.h>#include<string.h>#include<sys/socket.h>#include<arpa/inet.h>#define BUF_SIZE 30void error_handling(char* message);int main(int argc,char* argv[]){ int target_sock; struct sockaddr_in target_addr; FILE* fp; char buf[BUF_SIZE]; int time_live=64; if(argc!=3) { printf("Usage %s <address> <port> \n",argv[0]); exit(1); } target_sock=socket(AF_INET,SOCK_DGRAM,0); target_addr.sin_family=AF_INET; target_addr.sin_addr.s_addr=inet_addr(argv[1]); target_addr.sin_port=htons(atoi(argv[2])); fp=fopen("news.txt","r"); if(fp==NULL) { error_handling("fopen error!"); } setsockopt(target_sock,IPPROTO_IP, IP_MULTICAST_TTL,(void*)&time_live,sizeof(time_live)); while(!feof(fp)) { fgets(buf,BUF_SIZE,fp); sendto(target_sock,buf,strlen(buf), 0,(struct sockaddr*)&target_addr,sizeof(target_addr)); sleep(2); } fclose(fp); close(target_sock); return 0;}void error_handling(char* message){ fputs(message,stderr); fputc('\n',stderr); exit(1);}第28行,打开news.txt文件,获得文件指针。
第34行,利用setsockopt设置socket的TTL,而多播组的地址是我们通过main函数的参数传进去的,无需特别的设置,只需指明D类的多播地址即可。
我们可以通过下面的方法调用服务端,233.1.1.1就是多播的地址。
./sender 223.1.1.1 9190
二 接收端
接收端的示例代码如下:#include<stdio.h>#include<stdlib.h>#include<string.h>#include<unistd.h>#include<arpa/inet.h>#include<sys/socket.h>#define BUF_SIZE 30void error_handling(char* message);int main(int argc,char* argv[]){ int recv_sock; struct sockaddr_in recv_addr; char buf[BUF_SIZE]; struct ip_mreq join_adr; int str_len; if(argc!=3) { printf("Usage %s <adress> <port>\n",argv[0]); exit(1); } recv_sock=socket(AF_INET,SOCK_DGRAM,0); recv_addr.sin_family=AF_INET; recv_addr.sin_addr.s_addr=htonl(INADDR_ANY); recv_addr.sin_port=htons(atoi(argv[2])); if(bind(recv_sock,(struct sockaddr*)&recv_addr,sizeof(recv_addr))==-1) error_handling("bind error"); join_adr.imr_multiaddr.s_addr=inet_addr(argv[1]); join_adr.imr_interface.s_addr=htonl(INADDR_ANY); setsockopt(recv_sock,IPPROTO_IP, IP_ADD_MEMBERSHIP,(void*)&join_adr,sizeof(join_adr)); while(1) { str_len=recvfrom(recv_sock,buf,BUF_SIZE-1,MSG_DONTWAIT,NULL,0); printf("接收到 %d 字节。\n",str_len); if(str_len<=0) break; buf[str_len]=0; fputs(buf,stdout); } close(recv_sock); return 0;}void error_handling(char* message){ fputs(message,stderr); fputc('\n',stderr); exit(1);}
第17行,我们声明了一个ip_mreq类型的对象,用来存放多播的组地址和本机的IP地址。
第33、34行分别将多播地址和本机地址传给ip_mreq对象的对应成员。需要注意的是我们这里的多播地址是通过main函数的第二个参数传过来的。
第36行,通过setsockopt()函数使本机加入多播组。
客户端的用法如下:
./recver 223.1.1.1 9190223.1.1.1就是我们要加入的多播组。
Note:
1、因为TCP协议是有连接的,属于点对点通信,多播(组播)只支持UDP协议。
2、多播接收端和发送端的端口号要一致。
Github位置:
https://github.com/HymanLiuTS/NetDevelopment
克隆本项目:
git clone git@github.com:HymanLiuTS/NetDevelopment.git
获取本文源代码:
git checkout NL19
- 网络编程(19)—— 多播的编程实现
- Linux网络编程——多播
- Linux网络编程——多播
- Linux网络编程——多播
- Linux网络编程——多播
- 网络编程多播——Multicast
- java网络编程: UDP 广播与多播的实现
- java网络编程: UDP 广播与多播的实现
- 【网络编程】Linux多播简单实现
- linux网络编程之-----多播(组播)编程
- linux网络编程之-----多播(组播)编程
- linux网络编程之-----多播(组播)编程
- linux网络编程之-----多播(组播)编程
- linux网络编程之-----多播(组播)编程
- linux网络编程之一-----多播(组播)编程
- linux网络编程之-----多播(组播)编程
- linux网络编程之-----多播(组播)编程
- linux网络编程:多播(组播)编程
- weka 3.6.13-SNAPSHOT 过滤器StringToWordVector参数含义解释
- 关于thinkphp的模板转换
- The substring() Method in JDK 6 and JDK 7
- 符号表的基本操作
- 《将博客搬至CSDN》
- 网络编程(19)—— 多播的编程实现
- 【Android开发高级技术】Android 热补丁动态修复框架分析与总结
- Java注解
- LightOJ 1031 Easy Game--区间dp
- 【leetcode】Remove Duplicates from Sorted List
- 机器人(机械臂)动力学建模方法(Newton-Euler equation)
- request
- 如何上传自己的项目源码到github (Git教程)
- 调用动态库导致符号无法解析错误