组播实验

在csdn上看的代码在xp,win2003系统自己可以互相通信,但和win10不可以。请高人给个能协调IGMP版本的通用,可靠,稳定的代码。
http://blog.csdn.net/grove6lin/article/details/8252043

注意开启mt(多线程)

#include <iostream>
#include <winsock2.h> //注意这里的include文件顺序
#include <Ws2tcpip.h>
#include <process.h> //_beginthread要求

#pragma comment(lib, "ws2_32.lib")

using namespace std;

const char* MULTICAST_IP = "224.0.0.99"; //多播组地址
const int MULTICAST_PORT = 2002; //多播组端口

const int BUFFER_SIZE = 1024;

void do_send(void* arg); //读取用户输入并发送到多播组线程函数
void do_read(void* arg); //读物多播组数据函数

int main()
{
    WSAData wsaData;

    if( WSAStartup(MAKEWORD(2,2), &wsaData) != 0 )
    {
        cout<<"Error in WSAStartup"<<endl;
        return 0;
    }

    SOCKET server;
    server = socket(AF_INET, SOCK_DGRAM, 0); //创建一个UDP套接口
    cout<<"create socket: "<<server<<endl;

    int ret ;

    const int on = 1; //允许程序的多个实例运行在同一台机器上
    ret = setsockopt(server, SOL_SOCKET, SO_REUSEADDR, (char *)&on, sizeof(on));
    if( ret == SOCKET_ERROR )
    {
        WSACleanup();

        cout<<"Error in setsockopt(SO_REUSEADDR): "<<WSAGetLastError()<<endl;
        return 0;
    }

    const int routenum = 10;
    ret = setsockopt(server,IPPROTO_IP,IP_MULTICAST_TTL,\
        (char*)&routenum,sizeof(routenum));
    if( ret == SOCKET_ERROR )
    {
        WSACleanup();

        cout<<"Error in setsockopt(IP_MULTICAST_TTL): "<<WSAGetLastError()<<endl;
        return 0;
    }

//  const int loopback = 0; //禁止回馈
        const int loopback = 1; //
    ret = setsockopt(server,IPPROTO_IP,IP_MULTICAST_LOOP,\
        (char*)&loopback,sizeof(loopback));
    if( ret == SOCKET_ERROR )
    {
        WSACleanup();

        cout<<"Error in setsockopt(IP_MULTICAST_LOOP): "<<WSAGetLastError()<<endl;
        return 0;
    }

    sockaddr_in local;
    memset(&local, 0, sizeof(local));
    local.sin_family = AF_INET;
    local.sin_port = htons(MULTICAST_PORT);
    local.sin_addr.S_un.S_addr = INADDR_ANY;

    ret = bind(server, (sockaddr*)(&local), sizeof(local));
    if( ret == SOCKET_ERROR )
    {
        WSACleanup();

        cout<<"Error in bind: "<<WSAGetLastError()<<endl;
        return 0;
    }

    ip_mreq mreq;
    memset(&mreq, 0, sizeof(mreq));
    mreq.imr_interface.S_un.S_addr = INADDR_ANY;
    mreq.imr_multiaddr.S_un.S_addr = inet_addr(MULTICAST_IP);

    //加入一个多播组
    ret = setsockopt(server,IPPROTO_IP,IP_ADD_MEMBERSHIP,\
        (char*)&mreq,sizeof(mreq));
    if( ret == SOCKET_ERROR )
    {
        WSACleanup();

        cout<<"Error in setsockopt(IP_ADD_MEMBERSHIP): "<<WSAGetLastError()<<endl;
        return 0;
    }

    //创建了两个线程,一个读用户输入并发送,一个读多播组数据
    HANDLE hHandle[2];
    hHandle[0] = (HANDLE)_beginthread(do_send,0,(void*)server);
    hHandle[1] = (HANDLE)_beginthread(do_read,0,(void*)server);

    //如果用户输入结束,程序就终止了
    WaitForSingleObject(hHandle[0], INFINITE);

    WSACleanup();
    getchar();
    return 0;
}

void do_send(void* arg)
{
    SOCKET server = (SOCKET)arg;

    char sendline[BUFFER_SIZE+1];

    sockaddr_in remote;
    memset(&remote, 0, sizeof(remote));
    remote.sin_addr.s_addr = inet_addr ( MULTICAST_IP );
    remote.sin_family = AF_INET ;
    remote.sin_port = htons(MULTICAST_PORT);

    for(;;) //读取用户输入知道用户输入"end"
    {
        cin.getline(sendline, BUFFER_SIZE);

        if(strncmp(sendline,"end",3)==0)
            break;

        //发送用户输入的数据到多播组
        sendto(server, sendline, strlen(sendline), 0, (sockaddr*)(&remote), sizeof(remote)); 
    }

    cout<<"do_send end..."<<endl;
}

void do_read(void* arg)
{
    SOCKET server = (SOCKET)arg;

    char buf[BUFFER_SIZE+1];
    int ret;

    sockaddr_in client;
    int clientLen;

    for(;;) //一直读取知道主线程终止
    {
        clientLen = sizeof(client);
        memset(&client, 0, clientLen);

        ret = recvfrom(server, buf, BUFFER_SIZE, 0, (sockaddr*)(&clientLen), &clientLen);
        if ( ret == 0) //do_read在用户直接回车发送了一个空字符串
        {
            continue;
        }
        else if( ret == SOCKET_ERROR )
        {
            if( WSAGetLastError() == WSAEINTR ) //主线程终止recvfrom返回的错
                break;

            cout<<"Error in recvfrom: "<<WSAGetLastError()<<endl;
            break ;
        }
        buf[ret] = '\0';
        cout<<"received: "<<buf<<endl;
    }

    cout<<"do_read end..."<<endl;
}

http://blog.csdn.net/langeldep/article/details/6167137

http://wk.baidu.com/view/7f42bb19fc4ffe473368ab2a?pn=-2&pu=
上边连接中的代码如下,也仅仅能在winxp,win2003间通信。网上有人讲修改注册表调剂版本,不过我没有成功过。
在vs2013下要用多字节,平台xp,多线程。
receive

// testMulticast.cpp : 定义控制台应用程序的入口点。
//receive

#include "stdafx.h"
#include <winsock2.h>
#include <ws2tcpip.h>
#include <stdio.h>
#include <stdlib.h>
#pragma  comment(lib,"ws2_32.lib")
#define MCASTADDR "233.0.0.1" //本例使用的多播组地址。
#define MCASTPORT 5150 //绑定的本地端口号。
#define BUFSIZE 1024 //接收数据缓冲大小。
int main(int argc,char ** argv)
{
    WSADATA wsd;
    struct sockaddr_in local,remote,from;
    SOCKET sock,sockM;
    TCHAR recvbuf[BUFSIZE];
    /*struct ip_mreq mcast; // Winsock1.0 */
    int len = sizeof(struct sockaddr_in);
    int ret;
    //初始化WinSock2.2
    if (WSAStartup(MAKEWORD(2,2),&wsd) != 0)
    {
        printf("WSAStartup() failed\n");
        return -1;
    }
    /*
    创建一个SOCK_DGRAM类型的SOCKET
    其中,WSA_FLAG_MULTIPOINT_C_LEAF表示IP多播在控制面层上属于"无根"类型;
    WSA_FLAG_MULTIPOINT_D_LEAF表示IP多播在数据面层上属于"无根",有关控制面层和 数据面层有关概念请参阅MSDN说明。
    */
    if ((sock = WSASocket(AF_INET,SOCK_DGRAM,0,NULL,0,
        WSA_FLAG_MULTIPOINT_C_LEAF | WSA_FLAG_MULTIPOINT_D_LEAF |
        WSA_FLAG_OVERLAPPED)) == INVALID_SOCKET)
    {
        printf("socket failed with:%d\n",WSAGetLastError());
        WSACleanup();
        return -1;
    }
    //将sock绑定到本机某端口上。
    local.sin_family = AF_INET;
    local.sin_port = htons(MCASTPORT);
    local.sin_addr.s_addr = INADDR_ANY;
    if (bind(sock,(struct sockaddr*)&local,sizeof(local)) == SOCKET_ERROR) {
        printf("bind failed with:%d \n",WSAGetLastError());
        closesocket(sock);
        WSACleanup();
        return -1;
    }
    //加入多播组
    remote.sin_family = AF_INET;
    remote.sin_port = htons(MCASTPORT);
    remote.sin_addr.s_addr = inet_addr(MCASTADDR);
    /* Winsock1.0 */
    /*
    mcast.imr_multiaddr.s_addr = inet_addr(MCASTADDR);
    mcast.imr_interface.s_addr = INADDR_ANY;
    if( setsockopt(sockM,IPPROTO_IP,IP_ADD_MEMBERSHIP,
    (char*)&mcast,sizeof(mcast)) == SOCKET_ERROR)
    {
    printf("setsockopt(IP_ADD_MEMBERSHIP) failed:%d\n",WSAGetLastError()); closesocket(sockM);
    WSACleanup();
    return -1;
    }
    */
    /* Winsock2.0*/
    if ((sockM = WSAJoinLeaf(sock,(SOCKADDR*)&remote,sizeof(remote),
        NULL,NULL,NULL,NULL,
        JL_BOTH)) == INVALID_SOCKET)
    {

        printf("WSAJoinLeaf() failed:%d\n",WSAGetLastError());
        closesocket(sock);
        WSACleanup();
        return -1;
    }
    //接收多播数据,当接收到的数据为"QUIT"时退出。
    while (1)
    {
        if ((ret = recvfrom(sock,recvbuf,BUFSIZE,0,
            (struct sockaddr*)&from,&len)) == SOCKET_ERROR)
        {
            printf("recvfrom failed with:%d\n",WSAGetLastError());
            closesocket(sockM);
            closesocket(sock);
            WSACleanup();
            return -1;
        }
        if (strcmp(recvbuf,"QUIT") == 0) break;
        else {
            recvbuf[ret] = '\0';
            printf("RECV:' %s ' FROM <%s> \n",recvbuf,inet_ntoa(from.sin_addr));
        }
    }
    closesocket(sockM);
    closesocket(sock);
    WSACleanup();
    return 0;
}

sender

// sendersock2.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"


#include <winsock2.h>
#include <ws2tcpip.h>
#include <stdio.h>
#include <stdlib.h>
#pragma  comment(lib,"ws2_32.lib")
#define MCASTADDR "233.0.0.1" //本例使用的多播组地址。
#define MCASTPORT 5150 //本地端口号。
#define BUFSIZE 1024 //发送数据缓冲大小。
int main(int argc,char ** argv)
{
    WSADATA wsd;
    struct sockaddr_in remote;
    SOCKET sock,sockM;
    TCHAR sendbuf[BUFSIZE];
    int len = sizeof(struct sockaddr_in);
    //初始化WinSock2.2
    if (WSAStartup(MAKEWORD(2,2),&wsd) != 0)
    {
        printf("WSAStartup() failed\n");
        return -1;
    }
    if ((sock = WSASocket(AF_INET,SOCK_DGRAM,0,NULL,0,
        WSA_FLAG_MULTIPOINT_C_LEAF | WSA_FLAG_MULTIPOINT_D_LEAF |
        WSA_FLAG_OVERLAPPED)) == INVALID_SOCKET)
    {
        printf("socket failed with:%d\n",WSAGetLastError());
        WSACleanup();
        return -1;
    }
    //加入多播组
    remote.sin_family = AF_INET;
    remote.sin_port = htons(MCASTPORT);
    remote.sin_addr.s_addr = inet_addr(MCASTADDR);
    if ((sockM = WSAJoinLeaf(sock,(SOCKADDR*)&remote,
        sizeof(remote),NULL,NULL,NULL,NULL,
        JL_BOTH)) == INVALID_SOCKET)
    {
        printf("WSAJoinLeaf() failed:%d\n",WSAGetLastError());
        closesocket(sock);
        WSACleanup();
        return -1;
    }
    //发送多播数据,当用户在控制台输入"QUIT"时退出。
    while (1)
    {
        printf("SEND : ");
        scanf("%s",sendbuf);
        if (sendto(sockM,(char*)sendbuf,strlen(sendbuf),0,
            (struct sockaddr*)&remote,sizeof(remote)) == SOCKET_ERROR)
        {
            printf("sendto failed with: %d\n",WSAGetLastError());
            closesocket(sockM);
            closesocket(sock);
            WSACleanup();
            return -1;
        }
        if (strcmp(sendbuf,"QUIT") == 0) break;
        Sleep(500);
    }
    closesocket(sockM);
    closesocket(sock);
    WSACleanup();
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值