广播
1.广播地址
如果用{netid, subnetid, hostid}( {网络ID,子网ID,主机ID})表示IPv4地址。那么有四种类型的广播地址,我们用-1表示所有比特位均为1的字段:
1). 子网广播地址:{netid, subnetid, -1}。这类地址编排指定子网上的所有接口。
2). 全部子网广播地址:{netid, -1, -1}。这类广播地址编排指定网络上的所有子网。
3). 网络广播地址:{netid, -1}。这类地址用于不进行子网划分的网络。
4). 受限广播地址:{-1, -1, -1}。路由器从不转发目的地址为255.255.255.255的IP数据
 
2.单播和广播的比较
单播IP数据报仅有通过目的IP地址指定的单个主机接受,子网上的其他主机均不受影响;
子网上所有未参加相应广播应用的所有机也必须沿协议栈一路向上完整的处理收取UDP广播数据报,直至该数据包经历UDP层时被丢弃为止
 
3.竞争状态
多个进程访问共享数据,但正确结构依赖于进程的执行顺序,这种情况我们称之为竞争状态(race condition)。竞争状态通常是线程编程中始终要注意的一个重要问题,因为在线程中有非常之多的数据需要共享,例如所有的全程变量。
在进行信号处理时,通常会出现各种类型的竞争状态。这是因为在我们的程序执行过程中,内核随时都会递交信号。
 
client.c
#include <netinet/in.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <string.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <net/if.h>
#include <fcntl.h>
#include <unistd.h>
#include <ctype.h>
#define SERVER_PORT 5555
#define CLIENT_PORT 5556
#define MAXLINE 4096 int main(int argc,char *argv[])
{
struct sockaddr_in localaddr;
int confd;
ssize_t len;
char buf[MAXLINE]; //1.创建一个socket
confd=socket(AF_INET,SOCK_DGRAM,);
bzero(&localaddr,sizeof(localaddr));
localaddr.sin_family=AF_INET;
localaddr.sin_addr.s_addr=htonl(INADDR_BROADCAST);
localaddr.sin_port =htons(SERVER_PORT); int opt=-;
setsockopt(confd,SOL_SOCKET,SO_BROADCAST,(char*)&opt,sizeof(opt));
int clientlen=sizeof(struct sockaddr_in); while(){
fgets(buf,sizeof(buf),stdin);
sendto(confd,buf,strlen(buf),,(struct sockaddr*)&localaddr,clientlen);
len=recvfrom(confd,buf,sizeof(buf),,NULL,);
write(STDIN_FILENO,buf,len);
}
close(confd);
return ;
}

server.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <string.h>
#include <netinet/in.h>
#include <ctype.h>
#include <net/if.h> #define SERVER_PORT 5555
#define CLIENT_PORT 5556
#define MAXLINE 1024 int main(void)
{
int sockfd,i;
ssize_t len;
struct sockaddr_in serveraddr,clientaddr;
char buf[MAXLINE];
char ipstr[INET_ADDRSTRLEN];//16个字节
socklen_t clientlen; //构造用于UDP通信的套接字
sockfd=socket(AF_INET,SOCK_DGRAM,);
//绑定广播监听
bzero(&serveraddr,sizeof(serveraddr));
serveraddr.sin_family =AF_INET;
serveraddr.sin_addr.s_addr=htonl(INADDR_ANY);
serveraddr.sin_port=htons(SERVER_PORT);
//printf("%x\n",INADDR_ANY);
int opt=-;
setsockopt(sockfd,SOL_SOCKET,SO_BROADCAST,(char*)&opt,sizeof(opt));
bind(sockfd,(struct sockaddr*)&serveraddr,sizeof(serveraddr)); /*构造client地址 */
bzero(&clientaddr,sizeof(clientaddr));
clientaddr.sin_family=AF_INET;
clientaddr.sin_addr.s_addr=htonl(INADDR_ANY);
clientaddr.sin_port=htons(SERVER_PORT); clientlen=sizeof(struct sockaddr_in);
while(){
memset(buf,,sizeof(buf));
len=recvfrom(sockfd,buf,sizeof(buf),,(struct sockaddr*)&clientaddr,&clientlen);
printf("Rec from client--IP: %s\t PORT %d\n",
inet_ntop(AF_INET,&clientaddr.sin_addr.s_addr,ipstr,sizeof(ipstr)),
ntohs(clientaddr.sin_port));
i=;
while(i<len){
buf[i]=toupper(buf[i]);
i++;
}
buf[i]='\0';
sendto(sockfd,buf,strlen(buf),,(struct sockaddr*)&clientaddr,sizeof(clientaddr));
}
close(sockfd);
return ;
}
多播
1.多播地址
1).IPv4多播地址和IPv6多播地址、IPv6多播地址
IPv4中的D类地址(从224.0.0.0到239.255.255.255)是多播地址。D类地址的低28位构成了多播组ID(group ID),而整个32位地址则称为组地址(group address)。
下面是多播地址映射到以太网地址的方法:

2).多播地址的范围
IPv6多播地址有一个4位的显示范围字段,该字段决定了多播数据报能够游走的范围。IPv6分组也有一个跳限字段,它限制分组被路由器转发的次数。下面是几个已经分配了的范围字段值:
  • 1:节点局部即局部于节点(node-local)
  • 2:链路局部即局部于链路(link-local)
  • 3:网点局部即局部于网点(site-local)
  • 8:组织局部即局部于组织(orgainization-local)
  • 14:全球(global)
其他值或者还没有分配或者已经保留。节点局部数据报禁止从接口输出,链路局部数据报不能被路由器转发,网点和组织的定义由该网点或组织的多播路由器管理员决定。
 
2.多播套接字选项
下面列出部分关于多播的套接字选项

server.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <string.h>
#include <netinet/in.h>
#include <ctype.h>
#include <net/if.h> #define SERVER_PORT 5555
#define CLIENT_PORT 5556
#define MAXLINE 1024
#define GROUP "239.0.0.2" int main(void)
{
int sockfd,i;
ssize_t len;
struct sockaddr_in serveraddr,clientaddr;
char buf[MAXLINE];
char ipstr[INET_ADDRSTRLEN];//16个字节
socklen_t clientlen;
struct ip_mreqn group;//设置组播 //构造用于UDP通信的套接字
sockfd=socket(AF_INET,SOCK_DGRAM,); bzero(&serveraddr,sizeof(serveraddr));
serveraddr.sin_family =AF_INET;
serveraddr.sin_addr.s_addr=htonl(INADDR_ANY);
serveraddr.sin_port=htons(SERVER_PORT);
//printf("%x\n",INADDR_ANY);
bind(sockfd,(struct sockaddr*)&serveraddr,sizeof(serveraddr));
/*设置组地址*/
inet_pton(AF_INET,GROUP,&group.imr_multiaddr);
/*设置本地地址*/
inet_pton(AF_INET,"0.0.0.0",&group.imr_address);
group.imr_ifindex=if_nametoindex("eth0");//将网卡名转换成序号 等价 ip ad setsockopt(sockfd,IPPROTO_IP,IP_MULTICAST_IF,&group,sizeof(group));
/*构造client地址 IP+端口*/
bzero(&clientaddr,sizeof(clientaddr));
clientaddr.sin_family=AF_INET;
inet_pton(AF_INET,GROUP,&clientaddr.sin_addr.s_addr);
clientaddr.sin_port=htons(CLIENT_PORT);
while(){
fgets(buf,sizeof(buf),stdin);
sendto(sockfd,buf,strlen(buf),,(struct sockaddr*)&clientaddr,sizeof(clientaddr));
}
close(sockfd);
return ;
}

client.c

#include <netinet/in.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <string.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <net/if.h>
#include <fcntl.h>
#include <unistd.h>
#include <ctype.h>
#define SERVER_PORT 5555
#define CLIENT_PORT 5556
#define MAXLINE 4096
#define GROUP "224.0.2.0" int main(int argc,char *argv[])
{
struct sockaddr_in serveraddr,localaddr;
int confd;
ssize_t len;
char buf[MAXLINE];
struct ip_mreqn group;//设置组播 //1.创建一个socket
confd=socket(AF_INET,SOCK_DGRAM,);
//2.初始化服务器地址
bzero(&localaddr,sizeof(localaddr));
localaddr.sin_family=AF_INET;
//
inet_pton(AF_INET,"0.0.0.0",&localaddr.sin_addr.s_addr);
localaddr.sin_port =htons(CLIENT_PORT);
bind(confd,(struct sockaddr *)&localaddr,sizeof(localaddr)); /*设置组地址*/
inet_pton(AF_INET,GROUP,&group.imr_multiaddr);
/*设置本地地址*/
inet_pton(AF_INET,"0.0.0.0",&group.imr_address);
group.imr_ifindex=if_nametoindex("eth0");//将网卡名转换成序号 等价 ip ad
/*设置客户端加入多播组*/
setsockopt(confd,IPPROTO_IP,IP_ADD_MEMBERSHIP,&group,sizeof(group)); while(){
len=recvfrom(confd,buf,sizeof(buf),,NULL,);
write(STDIN_FILENO,buf,len);
}
close(confd);
return ;
}
单播、广播、组播的区别小结
1.单播:
  主机之间一对一的通讯模式,网络中的交换机和路由器对数据只进行转发不进行复制。如果10个客户机需要相同的数据,则服务器需要逐一传送,重复10次相同的工作。但由于其能够针对每个客户的及时响应,所以现在的网页浏览全部都是采用单播模式,具体的说就是IP单播协议。网络中的路由器和交换机根据其目标地址选择传输路径,将IP单播数据传送到其指定的目的地。
  单播的优点:
  1)服务器及时响应客户机的请求
  2)服务器针对每个客户不通的请求发送不通的数据,容易实现个性化服务。
        
  单播的缺点:
  1)服务器针对每个客户机发送数据流,服务器流量=客户机数量×客户机流量;在客户数量大、每个客户机流量大的流媒体应用中服务器不堪重负。
  2)现有的网络带宽是金字塔结构,城际省际主干带宽仅仅相当于其所有用户带宽之和的5%。如果全部使用单播协议,将造成网络主干不堪重负。现在的P2P应用就已经使主干经常阻塞。而将主干扩展20倍几乎是不可能。
 
2.广播:
  主机之间一对所有的通讯模式,网络对其中每一台主机发出的信号都进行无条件复制并转发,所有主机都可以接收到所有信息(不管你是否需要),由于其不用路径选择,所以其网络成本可以很低廉。有线电视网就是典型的广播型网络,我们的电视机实际上是接受到所有频道的信号,但只将一个频道的信号还原成画面。在数据网络中也允许广播的存在,但其被限制在二层交换机的局域网范围内,禁止广播数据穿过路由器,防止广播数据影响大面积的主机。
  广播的优点:
  1)网络设备简单,维护简单,布网成本低廉
  2)由于服务器不用向每个客户机单独发送数据,所以服务器流量负载极低。
        
  广播的缺点:
  1)无法针对每个客户的要求和时间及时提供个性化服务。
  2)网络允许服务器提供数据的带宽有限,客户端的最大带宽=服务总带宽。例如有线电视的客户端的线路支持100个频道(如果采用数字压缩技术,理论上可以提供500个频道),即使服务商有更大的财力配置更多的发送设备、改成光纤主干,也无法超过此极限。也就是说无法向众多客户提供更多样化、更加个性化的服务。
  3)广播禁止允许在Internet宽带网上传输。
 
  3.组播:
  主机之间一对一组的通讯模式,也就是加入了同一个组的主机可以接受到此组内的所有数据,网络中的交换机和路由器只向有需求者复制并转发其所需数据。主机可以向路由器请求加入或退出某个组,网络中的路由器和交换机有选择的复制并传输数据,即只将组内数据传输给那些加入组的主机。这样既能一次将数据传输给多个有需要(加入组)的主机,又能保证不影响其他不需要(未加入组)的主机的其他通讯。
  组播的优点:
  1)需要相同数据流的客户端加入相同的组共享一条数据流,节省了服务器的负载。具备广播所具备的优点。
  2)由于组播协议是根据接受者的需要对数据流进行复制转发,所以服务端的服务总带宽不受客户接入端带宽的限制。IP协议允许有2亿6千多万个组播,所以其提供的服务可以非常丰富。  
  3)此协议和单播协议一样允许在Internet宽带网上传输。
  
  组播的缺点:
  1)与单播协议相比没有纠错机制,发生丢包错包后难以弥补,但可以通过一定的容错机制和QOS加以弥补。
  2)现行网络虽然都支持组播的传输,但在客户认证、QOS等方面还需要完善,这些缺点在理论上都有成熟的解决方案,只是需要逐步推广应用到现存网络当中
 

网络编程--广播&组播的更多相关文章

  1. Ztack学习笔记(6)-广播组播点播

    Zigbee网络中进行数据通信主要有三种类型:单播.组播.广播.那这三种方式如何设置呢,在哪里设置呢, 一. 广播 当应用程序需要将数据包发送给网络的每一个设备时,使用这种模式.广播的短地址有三种 0 ...

  2. Linux IPC socket 广播,组播

    getsockopt()/setsockopt() //获得sockfd指向的socket的属性 int getsockopt(int sockfd, int level, int optname, ...

  3. Linux网络编程组播测试代码

    Linux网络编程组播测试代码 (转载)   组播客户端代码如下: #include <sys/types.h>#include <sys/socket.h>#include ...

  4. [C# 网络编程系列]专题七:UDP编程补充——UDP广播程序的实现

    转自:http://www.cnblogs.com/zhili/archive/2012/09/03/2666974.html 上次因为时间的关系,所以把上一个专题遗留下的一个问题在本专题中和大家分享 ...

  5. IPv4组播通信原理

    2011-05-08 21:21:14 标签:组播 vin_do,vin_do学习笔记,笔记 休闲 职场 摘自网络,感谢原作者 摘要: 本文试图成为学习TCP/IP网络组播技术的入门材料.文中介绍了组 ...

  6. IP组播技术介绍及实现例子

    引 言 近年来,随着Internet的迅速普及和爆炸性发展,在Internet上产生了许多新的应用,其中不少是高带宽的多媒体应用,譬如网 络视频会议.网络音频/视频广播.AOD/VOD.股市行情发布. ...

  7. day34 Pyhton 网络编程

    一今日内容 # 函数 # 面向对象 # 进阶 # 网络编程 4 # 并发编程 6-7 # 概念 # 网络基础 # 局域网的概念 # 交换机和路由器的工作流程 # ip地址 # mac地址 # 子网掩码 ...

  8. Linux 系统编程 学习:06-基于socket的网络编程1:有关概念

    Linux 系统编程 学习:006-基于socket的网络编程1:有关概念 背景 上一讲 进程间通信:System V IPC(2)中,我们介绍了System IPC中关于信号量的概念,以及如何使用. ...

  9. linux网络编程(一)

    ============================================================== 第一天:基本概念.TCP.FTP: =================== ...

随机推荐

  1. 继承FileInputFormat类来理解 FileInputFormat类

    import java.io.IOException; import java.util.ArrayList; import java.util.List; import org.apache.had ...

  2. 字符串缓冲区StringBuffer类,正则表达式

    StringBuffer类 StringBuffer又称为可变字符序列,它是一个类似于 String 的字符串缓冲区,通过某些方法调用可以改变该序列的长度和内容.它是一个容器,容器中可以装很多字符串. ...

  3. mybatis-generator的maven插件使用异常(mybatis-generator-maven-plugin):generate failed: Exception getting JDBC Driver

    使用mybatis的代码生成工具:mybatis-generator,在父model中引入了maven插件的依赖,如下: <!-- Mybatis.generator插件 --> < ...

  4. 页面跳转,A跳到B,B再返回A时自动定位到离开A时的位置

    <template> <div class="hello" @scroll="scrollLoad" id="myScrollBox ...

  5. mac系统的几种u盘启动制作方式

    先拿一个U盘,格式化好(guid分区表之类的选项弄好) 1.通过终端制作: 1>下载好自己要安装的系统镜像,最新的在App Store上下,以前的可以去pc 苹果等地方下载 2>在终端输入 ...

  6. 日常运维管理技巧一(查看负载 W)

    日常运维管理技巧一(查看负载 W) 今天针对Linux系统管理做一个专题的记录,以后会用的几率也是很大的,只要掌握必备的基础知识,做初级系统管理员是不成问题的. 作为一个运维工程师.系统管理员,如果对 ...

  7. Erwin 简单使用

    1. 物理设计:汉译英过程 ① Logical 中操作:Tools-Names-Edit Naming Standards…-Glossary选项import,导入内容为编辑好的CSV文件(只包含中文 ...

  8. map集合修改其中元素 去除Map集合中所有具有相同值的元素 Properties长久保存的流操作 两种用map记录单词或字母个数的方法

    package com.swift.lianxi; import java.util.HashMap; import java.util.Iterator; import java.util.Map; ...

  9. 关于discuz 不能全文搜索的问题

    这个问题客服反馈很多次了,以为discuz 默认搜索只能搜标题,除非配置了sphinx全文搜索引擎. 但是之前比较老的员工说以前能用的,也就是discuz老版本. 今天突然想到是不是discuz纵横搜 ...

  10. egg- 配置

    1. model module.exports = app => { const { INTEGER, STRING, TEXT } = app.Sequelize; const User = ...