基于Android平台利用UDP广播/多播数据传输研究

前言

很久没更新技术文章了,先更新一篇对于UDP多播研究的文章。

UDP广播与多播

在之前学习计算机网络课程时,了解到了TCP网络以及UDP网络,TCP是需要进行三次握手的可靠网络,而UDP则是不可靠的传输方式。但是不可靠并不意味着UDP就是没有用的传输方式,例如腾讯的QQ客户端传输以及一些网络游戏中的数据传输都是用了UDP协议。因为不需要建立连接,客户端可以直接向服务器发送信息,客户端只需要给出服务器的IP地址以及端口号,然后将信息封装到报文当中,就不需要再管其他,不像TCP那样需要等待服务器返回报文。服务器可能不存在或者能否收到相应的报文,UDP协议中客户端根本不用管。

之前学习了UDP实践过一对一的单播程序,但是鲜有接触一对多的服务:广播(broadcast)、多播(multicast)。经过查阅相关资料以及研究,发现广播中网络中所有主机都会接受一份数据副本,而多播只会发送到一个多播地址,由地址转发给需要接受该数据的主机。

UDP广播

UDP广播与单播的最大区别在于IP地址不同,广播使用的预留的255.255.255.255地址,通过这一广播地址消息会被发送到该网络上的所有网络主机上。并且通过查阅资料,其实广播地址分为以下四种:

  • 有限广播,有限广播的地址设为255.255.255.255。并且使用了该广播地址的是不会被路由器转发的,因为如果路由器转发了广播信息,那么网络上的所有器都会进行链式传播,引发网络瘫痪。在指定给本地网络的广播数据包时,目的地址的网络标识部分和主机标识部分全都是1(255.255.255.255)。在任何情况下,路由器都不转发目的地址为有限广播地址的数据报,这样的数据报仅出现在本地网络中
  • 非定向广播,这种地址的形式为“netid.255.255.255。”如126.255.255.255。网络使用非定向广播向特定网段上的所有主机发送数据包!
  • 子网定向广播,在划分为子网的网络中,子网定向广播地址限于表示特定子网上的主机。
  • 全部子网定向广播,在划分为子网的internet网络中,网络设备可以使用全部子网定向广播地址向所有子网的主机发送广播消息。这一类型的地址现在已经基本不使用了,而由D类组播地址所取代 。

我理解的是广播多应用于网络游戏中处于同一本地网络客户端之间的信息交流,意思就是游戏中常用的大喇叭、或者是对所有人说话,但是还是需要指定主机的端口号,不可能主机的所有端口都会监听。

UDP多播

多播和广播类似,指定接受者的端口号,并且地址范围是224.0.0.0至239.255.255.255,由于其和广播类似,接下来例子就以UDP多播来实现Android平台上的发送与接收。

经过研究与查阅资料,发现Java上实现多播需要用到MulticastSocket类,其实该类就是DatagramSocket的子类,在使用时除了多播自己的一些特性外,把它当做DatagramSocket类使用就可以了。

首先是利用多播发送数据:

public class MultiCastClientActivity extends AppCompatActivity {

    private static final String TAG = "MultiCastClientActivity";

    private MulticastSocket mSocket;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_multi_cast_client);
        startService(new Intent(this, MultiCastService.class));
        startService(new Intent(this, MulticastServiceB.class));
        try {
            mSocket = new MulticastSocket(UDPConstant.PORT);
            mSocket.setTimeToLive(UDPConstant.TTLTIME);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public void sendMulticast(View view) {
        new SendThread().start();
    }

    private class SendThread extends Thread {
        @Override
        public void run() {
            DatagramPacket datagramPacket = null;
            byte[] data = "hello world!".getBytes();
            try {
                InetAddress address = InetAddress.getByName(UDPConstant.IP_ADDRESS);
                if (!address.isMulticastAddress()) {
                    throw new NoMulticastException();
                }
                datagramPacket = new DatagramPacket(data, data.length, address, UDPConstant.PORT);
                mSocket.send(datagramPacket);
//                mSocket.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

大概步骤是:

  1. 建立多播的socket
  2. 设置端口以及TTL(避免延时导致等待循环)
  3. 当需要发送的packet时,建立一个DatagramPacket并且设置好相应的多播地址以及数据,然后利用socket进行发送数据。

我在这里建立了两个Service充当网络上的主机,就拿其中一个service拿演示:

public class MultiCastService extends Service {

    private static final String TAG = "MultiCastService";

    private MulticastSocket mSocket;
    private InetAddress mAddress;

    @Override
    public void onCreate() {
        super.onCreate();
        try {
            mAddress = InetAddress.getByName(UDPConstant.IP_ADDRESS);

            if (!mAddress.isMulticastAddress()) {
                throw new NoMulticastException();
            }

            mSocket = new MulticastSocket(UDPConstant.PORT);
            mSocket.setTimeToLive(UDPConstant.TTLTIME);
            mSocket.joinGroup(mAddress);
            new WorkThread().start();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    private class WorkThread extends Thread {

        @Override
        public void run() {
            byte[] buffer = new byte[1024];
            DatagramPacket datagramPacket = new DatagramPacket(buffer, 1024);
            while (true) {
                try {
                    mSocket.receive(datagramPacket);
                    String result = new String(buffer,0,datagramPacket.getLength());
                    Log.i(TAG, "run: " + result);
                } catch (IOException e) {
                    e.printStackTrace();
                }

            }

        }
    }
}

基本流程如下:

  1. 同样是建立一个MulticastSocket来进行监听
  2. 然后让其加入到多播网段组当中。
  3. 让其开启对packet的监听,当接收到多播时,便可拿到数据并打印出来。

当触发发送事件时,得到的结论与预期是一致的,结果如下:

02-11 11:42:36.811 24526-24565/com.nickming.tcpudpdemo I/MulticastServiceB: run: hello world!
02-11 11:42:36.811 24526-24564/com.nickming.tcpudpdemo I/MultiCastService: run: hello world!

两个service都能够收到相应的发送的数据。证明这个多播的方案是可行的,由于多播具有广播所有的优点,并且使用起来更方便,所以UDP多播作为Android上的数据传输还是很方便的。

需要说到的是广播的话只需要将其中的地址改为255.255.255.255便可改为广播方式发送数据。

小结

利用UDP多播,我们可以轻松的在Android平台上实现局域网内的文件传输、视频聊天、广播聊天等功能。例如像实现IPC通信,在同一局域网的环境下就可以利用TCP或者UDP来实现,比用AIDL等方式要方便些。还有系统里面也有一些功能也是利用多播实现的,例如像WifiManager.MulticastLock就可以发送packet来改变wifi的状态。总而言之,UDP多播在Android平台上还是有很多用途的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270

推荐阅读更多精彩内容

  • 12.1 引言 在第1章中我们提到有三种IP地址:单播地址、广播地址和多播地址。本章将更详细地介绍广播和多播。 广...
    张芳涛阅读 744评论 0 4
  • 1.这篇文章不是本人原创的,只是个人为了对这部分知识做一个整理和系统的输出而编辑成的,在此郑重地向本文所引用文章的...
    SOMCENT阅读 12,979评论 6 174
  • 个人认为,Goodboy1881先生的TCP /IP 协议详解学习博客系列博客是一部非常精彩的学习笔记,这虽然只是...
    贰零壹柒_fc10阅读 5,019评论 0 8
  • 名词延伸 通俗的说,域名就相当于一个家庭的门牌号码,别人通过这个号码可以很容易的找到你。如果把IP地址比作一间房子...
    杨大虾阅读 20,391评论 2 57
  • 11.1 引言 UDP是一个简单的面向数据报的运输层协议:进程的每个输出操作都正好产生一个UDP数据报,并组装成一...
    张芳涛阅读 2,684评论 1 6