- 论坛徽章:
- 0
|
工作需要写一段组播的程序,上网下了个例程,编译过后发现几个问题,请高手指点:)
代码如下:
客户端代码
- /*
- * listen.c - An IP multicast client */
- #include <stdio.h>
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <arpa/inet.h>
- #include <netdb.h>
- #include <stdlib.h>
- #include <unistd.h>
- char * host_name = "224.0.0.1";
- /* 多播IP地址 */
- int port = 6789;
- int main(void)
- {
- struct ip_mreq command;
- int loop = 1;
- /* 多播循环 */
- int iter = 0;
- int sin_len;
- char message[256];
- int socket_descriptor;
- struct sockaddr_in sin;
- struct hostent *server_host_name;
- if((server_host_name = gethostbyname(host_name)) == 0)
- {
- perror("gethostbyname");
- exit(EXIT_FAILURE);
- }
- /*bzero(&sin, sizeof(sin));*/
- memset(&sin, 0, sizeof(sin));
- sin.sin_family = AF_INET;
- sin.sin_addr.s_addr = htonl(INADDR_ANY);
- sin.sin_port = htons(port);
- if((socket_descriptor = socket(PF_INET, SOCK_DGRAM, 0)) == -1)
- { perror("socket");
- exit(EXIT_FAILURE);
- }
- /* 调用bind之前,设置套接口选项启用多播IP支持*/
- loop = 1;
- if(setsockopt(socket_descriptor,
- SOL_SOCKET, SO_REUSEADDR,
- &loop, sizeof(loop)) < 0)
- {
- perror("setsockopt:SO_REUSEADDR");
- exit(EXIT_FAILURE);
- }
- if(bind(socket_descriptor,
- (struct sockaddr *)&sin, sizeof(sin)) < 0)
- {
- perror("bind");
- exit(EXIT_FAILURE);
- }
- /* 在同一个主机上进行广播设置套接口,
- 作用是方便单个开发系统上测试多播IP广播 */
- loop = 1;
- if(setsockopt(socket_descriptor,
- IPPROTO_IP, IP_MULTICAST_LOOP,
- &loop, sizeof(loop)) < 0)
- {
- perror("setsockopt:IP_MULTICAST_LOOP");
- exit(EXIT_FAILURE);
- }
- /* 加入一个广播组。进一步告诉Linux内核,
- 特定的套接口即将接受广播数据*/
- command.imr_multiaddr.s_addr = inet_addr("224.0.0.1");
- command.imr_interface.s_addr = htonl(INADDR_ANY);
- if(command.imr_multiaddr.s_addr == -1)
- {
- perror("224.0.0.1 not a legal multicast address");
- exit(EXIT_FAILURE);
- }
- if (setsockopt(socket_descriptor, IPPROTO_IP, IP_ADD_MEMBERSHIP,
- &command, sizeof(command)) < 0)
- {
- perror("setsockopt:IP_ADD_MEMBERSHIP");
- }
- while(iter++ < 8)
- {
- sin_len = sizeof(sin);
- if(recvfrom(socket_descriptor, message, 256, 0,
- (struct sockaddr *)&sin, &sin_len) == -1) {
- perror("recvfrom");
- }
- printf("Response #%-2d from server: %s\n", iter, message);
- sleep(2); }
- /* 接受8个广播后退出 */
- if(setsockopt(socket_descriptor, IPPROTO_IP, IP_DROP_MEMBERSHIP,
- &command, sizeof(command)) < 0) {
- perror("setsockopt:IP_DROP_MEMBERSHIP");
- }
- close(socket_descriptor);
- exit(EXIT_SUCCESS);}
- 服务端代码:
- /*
- * broadcast.c - An IP multicast server
- */
- #include <stdio.h>
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <arpa/inet.h>
- #include <netdb.h>#include <unistd.h>
- #include <stdlib.h>int port = 6789;
- int main(void)
- {
- int socket_descriptor;
- struct sockaddr_in address;
- /* 首先建立套接口 */
- socket_descriptor = socket(AF_INET, SOCK_DGRAM, 0);
- if (socket_descriptor == -1)
- {
- perror("Opening socket");
- exit(EXIT_FAILURE);
- }
- /* 初始化IP多播地址 */
- memset(&address, 0, sizeof(address));
- address.sin_family = AF_INET;
- address.sin_addr.s_addr = inet_addr("224.0.0.1");
- address.sin_port = htons(port);
- /* 开始进行IP多播 */
- while(1) {
- if(sendto(socket_descriptor, "test from broadcast",
- sizeof("test from broadcast"), 0,
- (struct sockaddr *)&address, sizeof(address)) < 0)
- {
- perror("sendto");
- exit(EXIT_FAILURE);
- }
- sleep(2);
- }
- exit(EXIT_SUCCESS);}
复制代码
编译过后会出现几个问题:
第一个 服务器的组地址改变后,比如改成了224.0.1.1,只有本地能受到信息 ,而局网内的另一台pc收不到(他们的组地址也改成224.0.1.1)。
第二个 服务器的组地址和本地的客户端的组地址不一样,也可以受到信息。
上述两个问题彻底扰乱了我的思维,请高手指点一下,谢谢:)
[ 本帖最后由 jamesman_xu 于 2007-9-15 17:58 编辑 ] |
|