java使用MulticastSocket實(shí)現(xiàn)多點(diǎn)廣播
DatagramSocket只允許數(shù)據(jù)報(bào)發(fā)送給指定的目標(biāo)地址,而MulticastSocket可以將數(shù)據(jù)報(bào)以廣播方式發(fā)送到數(shù)量不等的多個(gè)客戶端。
若要使用多點(diǎn)廣播時(shí),則需要讓一個(gè)數(shù)據(jù)報(bào)標(biāo)有一組目標(biāo)主機(jī)地址,當(dāng)數(shù)據(jù)報(bào)發(fā)出后,整個(gè)組的所有主機(jī)都能收到該數(shù)據(jù)報(bào)。IP多點(diǎn)廣播實(shí)現(xiàn)了將單一信息發(fā)送到多個(gè)接收者的廣播,其思想是設(shè)置一組特殊網(wǎng)絡(luò)地址作為廣播地址,每個(gè)多點(diǎn)廣播地址都被看做一個(gè)組,當(dāng)客戶端主要發(fā)送、接收信息時(shí),加入到該組即可。
IP協(xié)議為多點(diǎn)廣播提供了這批特殊的IP地址,這些地址的IP地址范圍是224.0.0.0至239.255.255.255。
通過(guò)Java實(shí)現(xiàn)多點(diǎn)廣播時(shí),MulticastSocket類是實(shí)現(xiàn)這一功能的關(guān)鍵,當(dāng)MulticastSocket把一個(gè)DatagramPacket發(fā)送到多點(diǎn)廣播的IP地址,該數(shù)據(jù)報(bào)將被自動(dòng)廣播到加入該地址的所有MulticastSocket類既可以發(fā)送數(shù)據(jù)報(bào)到多點(diǎn)廣播地址,也可以接受其他主機(jī)的廣播信息。
MulticastSocket有點(diǎn)像DatagramSocket,事實(shí)上MulticastSocket是特殊的DatagramSocket。若要發(fā)送一個(gè)數(shù)據(jù)報(bào)時(shí),可使用隨機(jī)端口段間MulticastSocket,也可以指定端口來(lái)創(chuàng)建MulticastSocket。
MulticastSocket提供了如下三個(gè)構(gòu)造器
public MulticastSocket():使用本機(jī)默認(rèn)地址、隨機(jī)端口來(lái)創(chuàng)建一個(gè)MulticastSocket對(duì)象。
public MulticastSocket(int number):使用本機(jī)默認(rèn)地址、指定端口來(lái)創(chuàng)建一個(gè)MulticastSocket對(duì)象。
public MulticastSocket(SocketAddress bindaddr):使用本機(jī)指定IP地址、指定端口來(lái)創(chuàng)建一個(gè)MulticastSocket對(duì)象。
創(chuàng)建一個(gè)MulticastSocket對(duì)象后,還需要將該MulticastSocket加入到指定的多點(diǎn)廣播地址,MulticastSocket使用joinGroup()方法來(lái)加入指定組;使用leaveGroup()方法脫離一個(gè)組。
joinGroup(InetAddress multicastAddr):將該MulticastSocket加入指定的多點(diǎn)廣播地址
leaveGroup(InetAddress multicastAddr):讓該MulticastSocket離開(kāi)指定的多點(diǎn)廣播地址。
在某些系統(tǒng)中,可能有多個(gè)網(wǎng)絡(luò)接口。這可能會(huì)對(duì)多點(diǎn)廣播帶來(lái)問(wèn)題,這時(shí)候程序需要在一個(gè)指定的網(wǎng)絡(luò)接口上監(jiān)聽(tīng),通過(guò)調(diào)用setInterface可選擇MulticastSocket所使用的網(wǎng)絡(luò)接口;頁(yè)可以使用getInterface方法查詢MulticastSocket監(jiān)聽(tīng)的網(wǎng)絡(luò)接口。
如果創(chuàng)建僅用于發(fā)送數(shù)據(jù)報(bào)的MulticastSocket對(duì)象,則使用默認(rèn)地址、隨機(jī)端口即可。但如果創(chuàng)建接收用的MulticastSocket對(duì)象,則該MulticastSocket對(duì)象必須有指定端口,否則發(fā)送方無(wú)法確定發(fā)送的數(shù)據(jù)報(bào)的目標(biāo)端口。
MulticastSocket用于發(fā)送,接收數(shù)據(jù)報(bào)的方法與DatagramSocket的完全一樣。但MulticastSocket比DatagramSocket多一個(gè)setTimeToLive(int ttl)的方法,該ttl參數(shù)設(shè)置數(shù)據(jù)報(bào)最多可以跨過(guò)幾個(gè)網(wǎng)絡(luò),當(dāng)ttl為0時(shí),指定數(shù)據(jù)報(bào)應(yīng)停留在本地主機(jī);當(dāng)ttl的值為1時(shí),指定數(shù)據(jù)報(bào)發(fā)送到本地局域網(wǎng);當(dāng)ttl的值為32時(shí),意味著只能發(fā)送到本站點(diǎn)的網(wǎng)絡(luò)上;當(dāng)ttl為64時(shí),意味著數(shù)據(jù)報(bào)應(yīng)保留在本地區(qū)。當(dāng)ttl為128時(shí),意味著數(shù)據(jù)報(bào)應(yīng)保留在本大洲。當(dāng)ttl為255時(shí),意味著數(shù)據(jù)報(bào)可發(fā)送到所有地方。默認(rèn)情況下,該ttl的值為1。
使用MulticastSocket進(jìn)行多點(diǎn)廣播時(shí)所有通信實(shí)體都是平等的,他們都將自己的數(shù)據(jù)報(bào)發(fā)送到多點(diǎn)廣播IP地址,并使用MulticastSocket接收其他人發(fā)送的廣播數(shù)據(jù)報(bào)。
import java.io.IOException; import java.net.DatagramPacket; import java.net.InetAddress; import java.net.MulticastSocket; import java.util.Scanner; //讓該類實(shí)現(xiàn)Runnable接口,該類的實(shí)例可作為線程的target public class Test implements Runnable { // 使用常量作為本程序的多點(diǎn)廣播IP地址 private static final String BROADCAST_IP = "230.0.0.1"; // 使用常量作為本程序的多點(diǎn)廣播目的的端口 public static final int BROADCAST_PORT = 30000; // 定義每個(gè)數(shù)據(jù)報(bào)的最大大小為4K private static final int DATA_LEN = 4096; // 定義本程序的MulticastSocket實(shí)例 private MulticastSocket socket = null; private InetAddress broadcastAddress = null; private Scanner scan = null; // 定義接收網(wǎng)絡(luò)數(shù)據(jù)的字節(jié)數(shù)組 byte[] inBuff = new byte[DATA_LEN]; // 以指定字節(jié)數(shù)組創(chuàng)建準(zhǔn)備接受數(shù)據(jù)的DatagramPacket對(duì)象 private DatagramPacket inPacket = new DatagramPacket(inBuff, inBuff.length); // 定義一個(gè)用于發(fā)送的DatagramPacket對(duì)象 private DatagramPacket outPacket = null; public void init() throws IOException { try { // 創(chuàng)建用于發(fā)送、接收數(shù)據(jù)的MulticastSocket對(duì)象 // 因?yàn)樵揗ulticastSocket對(duì)象需要接收,所以有指定端口 socket = new MulticastSocket(BROADCAST_PORT); broadcastAddress = InetAddress.getByName(BROADCAST_IP); // 將該socket加入指定的多點(diǎn)廣播地址 socket.joinGroup(broadcastAddress); // 設(shè)置本MulticastSocket發(fā)送的數(shù)據(jù)報(bào)被回送到自身 socket.setLoopbackMode(false); // 初始化發(fā)送用的DatagramSocket,它包含一個(gè)長(zhǎng)度為0的字節(jié)數(shù)組 outPacket = new DatagramPacket(new byte[0], 0, broadcastAddress, BROADCAST_PORT); // 啟動(dòng)以本實(shí)例的run()方法作為線程體的線程 new Thread(this).start(); // 創(chuàng)建鍵盤(pán)輸入流 scan = new Scanner(System.in); // 不斷讀取鍵盤(pán)輸入 while (scan.hasNextLine()) { // 將鍵盤(pán)輸入的一行字符串轉(zhuǎn)換字節(jié)數(shù)組 byte[] buff = scan.nextLine().getBytes(); // 設(shè)置發(fā)送用的DatagramPacket里的字節(jié)數(shù)據(jù) outPacket.setData(buff); // 發(fā)送數(shù)據(jù)報(bào) socket.send(outPacket); } } finally { socket.close(); } } public void run() { try { while (true) { // 讀取Socket中的數(shù)據(jù),讀到的數(shù)據(jù)放在inPacket所封裝的字節(jié)數(shù)組里。 socket.receive(inPacket); // 打印輸出從socket中讀取的內(nèi)容 System.out.println("聊天信息:" + new String(inBuff, 0, inPacket.getLength())); } } // 捕捉異常 catch (IOException ex) { ex.printStackTrace(); try { if (socket != null) { // 讓該Socket離開(kāi)該多點(diǎn)IP廣播地址 socket.leaveGroup(broadcastAddress); // 關(guān)閉該Socket對(duì)象 socket.close(); } System.exit(1); } catch (IOException e) { e.printStackTrace(); } } } public static void main(String[] args) throws IOException { new Test().init(); } }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java畢業(yè)設(shè)計(jì)實(shí)戰(zhàn)之校園一卡通系統(tǒng)的實(shí)現(xiàn)
這是一個(gè)使用了java+Springboot+Maven+mybatis+Vue+mysql+wd開(kāi)發(fā)的校園一卡通系統(tǒng),是一個(gè)畢業(yè)設(shè)計(jì)的實(shí)戰(zhàn)練習(xí),具有校園一卡通系統(tǒng)該有的所有功能,感興趣的朋友快來(lái)看看吧2022-01-01Java實(shí)現(xiàn)五子棋網(wǎng)絡(luò)版
這篇文章主要為大家詳細(xì)介紹了基于Java編寫(xiě)的網(wǎng)絡(luò)五子棋,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-03-03Java消息隊(duì)列的簡(jiǎn)單實(shí)現(xiàn)代碼
本篇文章主要介紹了Java消息隊(duì)列的簡(jiǎn)單實(shí)現(xiàn)代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-07-07Java基礎(chǔ)教程之基本類型數(shù)據(jù)類型、包裝類及自動(dòng)拆裝箱
這篇文章主要給大家介紹了關(guān)于Java基礎(chǔ)教程之基本類型數(shù)據(jù)類型、包裝類及自動(dòng)拆裝箱的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06IDEA中thymeleaf語(yǔ)法沒(méi)有提示的問(wèn)題及解決
這篇文章主要介紹了IDEA中thymeleaf語(yǔ)法沒(méi)有提示的問(wèn)題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05Spring Boot實(shí)現(xiàn)對(duì)文件進(jìn)行壓縮下載功能
在Web應(yīng)用中,文件下載功能是一個(gè)常見(jiàn)的需求,特別是當(dāng)你需要提供用戶下載各種類型的文件時(shí),本文將演示如何使用Spring Boot框架來(lái)實(shí)現(xiàn)一個(gè)簡(jiǎn)單而強(qiáng)大的文件下載功能,需要的朋友跟隨小編一起學(xué)習(xí)吧2023-09-09SpringBoot?整合RabbitMq?自定義消息監(jiān)聽(tīng)容器來(lái)實(shí)現(xiàn)消息批量處理
Spring Boot中提供了默認(rèn)的監(jiān)聽(tīng)器容器,但是有時(shí)候我們需要自定義監(jiān)聽(tīng)器容器,來(lái)滿足一些特殊的需求,比如批量獲取數(shù)據(jù),這篇文章主要介紹了SpringBoot?整合RabbitMq?自定義消息監(jiān)聽(tīng)容器來(lái)實(shí)現(xiàn)消息批量處理,需要的朋友可以參考下2023-04-04淺談mybatis mapper.xml文件中$和#的區(qū)別
這篇文章主要介紹了淺談mybatis mapper.xml文件中$和#的區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-11-11