Java實(shí)現(xiàn)Udp網(wǎng)絡(luò)編程

在看到本文之前睡腿,如果讀者沒(méi)看過(guò)筆者的上一個(gè)系列 Java實(shí)現(xiàn)Socket網(wǎng)絡(luò)編程熬拒,建議先翻閱爷光。

筆者將在上期Demo的基礎(chǔ)上,進(jìn)一步修改和擴(kuò)展澎粟,達(dá)到本次Demo的運(yùn)行效果蛀序。

首先展示Demo的演示效果:
初始狀態(tài):1個(gè)服務(wù)器欢瞪,2個(gè)客戶端

Paste_Image.png

檢測(cè)通信正常:

Paste_Image.png

斷開(kāi)服務(wù)器,再次檢測(cè)通信正常:

Paste_Image.png

服務(wù)器重新啟動(dòng)徐裸,自動(dòng)刷新:

Paste_Image.png

添加客戶端:

Paste_Image.png

關(guān)于 C(客戶端)和 S(服務(wù)器)之間的TCP通信遣鼓,以及 C 檢測(cè) S 狀態(tài),自動(dòng)重連等機(jī)制重贺,筆者在上期Demo的實(shí)現(xiàn)過(guò)程中已詳細(xì)闡述骑祟,此處就不再贅述。

我們來(lái)看看本次案例的實(shí)現(xiàn)需求:
1檬姥、服務(wù)器支持多客戶端訪問(wèn)
2曾我、C和S之間使用TCP連接
3粉怕、C和C之間使用UDP直接通信

由于案例需求的步驟1健民、2已實(shí)現(xiàn),我們對(duì)步驟3作如下設(shè)計(jì)思路:
1贫贝、客戶端創(chuàng)建監(jiān)聽(tīng)線程秉犹,建立UDP監(jiān)聽(tīng)端口,并發(fā)消息告訴服務(wù)器稚晚,指定自己的服務(wù)端口崇堵。
2、服務(wù)器得知客戶端的服務(wù)端口后客燕,廣播通知其他客戶端鸳劳,現(xiàn)已登錄的客戶端服務(wù)端口列表。
3也搓、客戶端之間直接通過(guò)UDP赏廓,向指定服務(wù)端口發(fā)送消息。

值得注意的是傍妒,C與C之間要求直接通信幔摸,所以必須滿足“在服務(wù)器關(guān)閉的情況下,C與C之間仍能通信”的情況颤练,而不是借助服務(wù)器完成間接通信

首先既忆,我們創(chuàng)建客戶端監(jiān)聽(tīng)線程,并發(fā)消息告訴服務(wù)器

public void run() {

        try {

            DatagramSocket server = new DatagramSocket(0);// 隨機(jī)分配一個(gè)端口號(hào)

            // 向服務(wù)器發(fā)送接收客戶端的DatagramSocket的端口號(hào)

            String message = Common.SPECIAL;

            String t = "" + server.getLocalPort();

            ClientMain.frame.setTitle("client " + t);
            String c = "" + t.length();
            if (c.length() < 2) {
                c = "000" + c;
            } else if (c.length() < 3) {
                c = "00" + c;
            } else if (c.length() < 4) {
                c = "0" + c;
            }
            message += c + t;

            OutputStreamWriter outstream = null;

            // 將信息發(fā)送給服務(wù)器
            try {
                outstream = new OutputStreamWriter(mSocket.getOutputStream(),
                        "GBK");
                outstream.write(message);
                outstream.flush();

            } catch (IOException e1) {
                ClientMain.jlConnect.setText("Out Of Connect.");
                ClientMain.isConnected = false;
                if (outstream != null)
                    try {
                        outstream.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                e1.printStackTrace();
            }

            while (true) {

                byte[] recvBuf = new byte[1024];// 定義接收消息的緩沖區(qū)
                DatagramPacket recvPacket = new DatagramPacket(recvBuf,
                        recvBuf.length);// 數(shù)據(jù)包
                server.receive(recvPacket);

                // 接收到的消息
                String recvStr = new String(recvPacket.getData(), 0,
                        recvPacket.getLength());

                ClientMain.jtaReceivedMessage.append(recvStr + "\n");
                // 滾動(dòng)到底端
                ClientMain.jtaReceivedMessage
                        .setCaretPosition(ClientMain.jtaReceivedMessage
                                .getText().length());
            }

        } catch (Exception e) {

            e.printStackTrace();
        }
}

服務(wù)器得知客戶端的服務(wù)端口后嗦玖,廣播通知其他客戶端

else if (s.startsWith(Common.SPECIAL) && s.length() > 10
                            && count == Integer.parseInt((s.substring(6, 10)))) {
                        // 存儲(chǔ)客戶端監(jiān)聽(tīng)端口
                        /**
                         * 一定要注意使用前初始化患雇,否則在IDE在這里檢測(cè)不到空指針錯(cuò)誤
                         */
                        HashMap<Socket, String> map = new HashMap<Socket, String>();
                        map.put(mSocket, s.substring(10));
                        ServerMain.clientMonitorPortList.add(map);
                        // 發(fā)送更新列表信息給客戶端
                        sendUpdateToClient();
                        count = -10;
                        s = "";
                    }

sendUpdateToClient方法如下:

// 發(fā)送更新列表信息給所有客戶端
    private void sendUpdateToClient() {
        String message = Common.SEND_TO_CLIENT;
        String t = "";

        for (int i = 0; i < ServerMain.clientMonitorPortList.size(); i++) {
            HashMap<Socket, String> map = ServerMain.clientMonitorPortList
                    .get(i);
            Iterator iter1 = map.entrySet().iterator();
            Map.Entry entry = (Map.Entry) iter1.next();
            Socket key = (Socket) entry.getKey();
            int localPort = key.getPort();
            String port = (String) entry.getValue();
            if (i != ServerMain.clientMonitorPortList.size() - 1)
                t += localPort + " " + port + " ";
            else
                t += localPort + " " + port;
        }

        String c = "" + t.length();
        if (c.length() < 2) {
            c = "000" + c;
        } else if (c.length() < 3) {
            c = "00" + c;
        } else if (c.length() < 4) {
            c = "0" + c;
        }
        message += c + t;

        OutputStreamWriter outstream = null;

        // 將信息發(fā)送給每個(gè)客戶端
        for (int i = 0; i < ListenThread.clientSockets.size(); i++) {
            try {
                HashMap<Socket, Boolean> map = ListenThread.clientSockets
                        .get(i);
                // 用迭代器獲取HashMap的Key,即所選中的Socket
                Iterator iter = map.entrySet().iterator();
                Map.Entry<Socket, Boolean> entry = (Entry<Socket, Boolean>) iter
                        .next();

                Socket key = (Socket) entry.getKey();
                outstream = new OutputStreamWriter(key.getOutputStream(), "GBK");
                outstream.write(message);
                outstream.flush();

            } catch (IOException e1) {
                if (outstream != null)
                    try {
                        outstream.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                e1.printStackTrace();
            }
        }
    }

最后宇挫,客戶端通過(guò)UDP向指定服務(wù)端口發(fā)送消息
當(dāng)選中JList的項(xiàng)時(shí)庆亡,向選中的項(xiàng)發(fā)送消息,如果沒(méi)有選中項(xiàng)捞稿,則向服務(wù)器發(fā)送消息

// 設(shè)置監(jiān)聽(tīng)
        jbSendMessage.addActionListener(new ActionListener() {

            @Override
            public void actionPerformed(ActionEvent e) {

                if (jtaSendMessage.getText().equals("")) {
                    JOptionPane.showMessageDialog(null, "發(fā)送內(nèi)容不能為空!");
                    return;
                }

                // 取得要發(fā)送的消息
                String message = Common.SIMPLE;

                String t = "client " + Common.IP + ":" + mSocket.getLocalPort()
                        + " " + jtaSendMessage.getText();
                String c = "" + t.length();
                if (c.length() < 2) {
                    c = "000" + c;
                } else if (c.length() < 3) {
                    c = "00" + c;
                } else if (c.length() < 4) {
                    c = "0" + c;
                }
                message += c + t;

                OutputStreamWriter outstream = null;

                // 如果沒(méi)有選中又谋,則向服務(wù)器發(fā)送消息
                if (selecteds == null || selecteds.length == 0) {
                    try {
                        outstream = new OutputStreamWriter(mSocket
                                .getOutputStream(), "GBK");
                        outstream.write(message);
                        outstream.flush();
                    } catch (IOException e1) {
                        if (outstream != null)
                            try {
                                outstream.close();
                            } catch (IOException e2) {
                                e2.printStackTrace();
                            }
                        e1.printStackTrace();
                    }
                } else {
                    String sendPort = "";

                    // 檢測(cè)現(xiàn)在進(jìn)行發(fā)送行為的是哪個(gè)客戶端
                    for (int i = 0; i < clientPortList.size(); i++) {
                        HashMap<String, String> map = (HashMap<String, String>) clientPortList
                                .get(i);
                        Iterator iter1 = map.entrySet().iterator();
                        Map.Entry entry = (Map.Entry) iter1.next();
                        String sendSocketPort = (String) entry.getKey();
                        // mSocket.getLocalPort()是int類型拼缝,要注意加""
                        if (sendSocketPort.equals(mSocket.getLocalPort() + "")) {
                            sendPort = (String) entry.getValue();
                        }
                    }

                    // 向選中的客戶端發(fā)送消息
                    for (int i = 0; i < selecteds.length; i++) {
                        // 獲取選中的端口
                        HashMap<String, String> map = (HashMap<String, String>) clientPortList
                                .get(selecteds[i]);
                        Iterator iter1 = map.entrySet().iterator();
                        Map.Entry entry = (Map.Entry) iter1.next();
                        String port = (String) entry.getValue();

                        try {
                            // 生成一個(gè)臨時(shí)發(fā)送端口
                            DatagramSocket client = new DatagramSocket(0);
                            // 要發(fā)送的數(shù)據(jù)
                            String sendMessage = "client " + Common.IP + ":"
                                    + sendPort + " " + jtaSendMessage.getText();
                            byte[] buf = sendMessage.getBytes();
                            // 定義發(fā)送信息的目的地
                            InetAddress destination = InetAddress
                                    .getByName(Common.IP);
                            // 生成數(shù)據(jù)包
                            DatagramPacket dp = new DatagramPacket(buf,
                                    buf.length, destination, Integer
                                            .valueOf(port));
                            client.send(dp);
                        } catch (Exception e1) {
                            e1.printStackTrace();
                        }
                    }
                }

                // 清空文本
                jtaSendMessage.setText(null);

            }

        });

本次實(shí)驗(yàn)步驟看似簡(jiǎn)單,但也有幾個(gè)不得不注意的地方:
1彰亥、在讀寫(xiě)數(shù)據(jù)的循環(huán)里咧七,是檢測(cè)不到空指針錯(cuò)誤的,只會(huì)檢測(cè)到讀寫(xiě)錯(cuò)誤后不斷嘗試重連任斋。讀者在開(kāi)發(fā)過(guò)程中一定要注意把相應(yīng)的控件初始化继阻,而發(fā)現(xiàn)不斷重連,重復(fù)讀寫(xiě)時(shí)废酷,應(yīng)首先考慮是否在讀寫(xiě)循環(huán)里引用了未初始化的控件瘟檩。
2、mSocket.getLocalPort()方法返回的是int類型澈蟆,使用equals比較時(shí)要注意加雙引號(hào)""墨辛,以轉(zhuǎn)換成String類型,否則IDE不會(huì)編譯報(bào)錯(cuò)趴俘,但結(jié)果并未如意睹簇。
3、使用UDP端口容易混亂:讀者在開(kāi)發(fā)過(guò)程中應(yīng)盡量避免更新UI時(shí)整體刪除再添加剩余項(xiàng)寥闪,而改用“只刪除關(guān)閉項(xiàng)太惠,只增加新增項(xiàng)”,前種方法在開(kāi)發(fā)過(guò)程中容易造成端口混亂疲憋。同時(shí)凿渊,筆者建議讀者在涉及JList操作時(shí),多用ArrayList替代HashMap存儲(chǔ)缚柳,因?yàn)锳rrayList是插入有序的埃脏,能減少混亂的發(fā)生。
4喂击、注意在視圖model中刪除了項(xiàng)剂癌,也要同時(shí)在列表List中刪除對(duì)應(yīng)項(xiàng),以做到真正的刪除翰绊,而不是假刪除佩谷。
5、刪除List中的所有項(xiàng):
for(int i=0;i<list.size();)list.remove(i);
注意监嗜!這里不能添加i++谐檀,因?yàn)槊看蝦emove后,list.size()會(huì)自動(dòng)減小裁奇,如果添加了i++桐猬,則不能完全刪除List中的元素,從而導(dǎo)致二次混亂

最后刽肠,筆者在github上給出了兩次實(shí)驗(yàn)的Demo源碼溃肪,供讀者學(xué)習(xí)和思考免胃,感謝關(guān)注!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末惫撰,一起剝皮案震驚了整個(gè)濱河市羔沙,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌厨钻,老刑警劉巖扼雏,帶你破解...
    沈念sama閱讀 216,919評(píng)論 6 502
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異夯膀,居然都是意外死亡诗充,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,567評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén)诱建,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)蝴蜓,“玉大人,你說(shuō)我怎么就攤上這事涂佃±恚” “怎么了蜈敢?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,316評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵辜荠,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我抓狭,道長(zhǎng)伯病,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,294評(píng)論 1 292
  • 正文 為了忘掉前任否过,我火速辦了婚禮午笛,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘苗桂。我一直安慰自己药磺,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,318評(píng)論 6 390
  • 文/花漫 我一把揭開(kāi)白布煤伟。 她就那樣靜靜地躺著癌佩,像睡著了一般。 火紅的嫁衣襯著肌膚如雪便锨。 梳的紋絲不亂的頭發(fā)上围辙,一...
    開(kāi)封第一講書(shū)人閱讀 51,245評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音放案,去河邊找鬼姚建。 笑死,一個(gè)胖子當(dāng)著我的面吹牛吱殉,可吹牛的內(nèi)容都是我干的掸冤。 我是一名探鬼主播厘托,決...
    沈念sama閱讀 40,120評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼稿湿!你這毒婦竟也來(lái)了催烘?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 38,964評(píng)論 0 275
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤缎罢,失蹤者是張志新(化名)和其女友劉穎伊群,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體策精,經(jīng)...
    沈念sama閱讀 45,376評(píng)論 1 313
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡舰始,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,592評(píng)論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了咽袜。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片丸卷。...
    茶點(diǎn)故事閱讀 39,764評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖询刹,靈堂內(nèi)的尸體忽然破棺而出谜嫉,到底是詐尸還是另有隱情,我是刑警寧澤凹联,帶...
    沈念sama閱讀 35,460評(píng)論 5 344
  • 正文 年R本政府宣布沐兰,位于F島的核電站,受9級(jí)特大地震影響蔽挠,放射性物質(zhì)發(fā)生泄漏住闯。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,070評(píng)論 3 327
  • 文/蒙蒙 一澳淑、第九天 我趴在偏房一處隱蔽的房頂上張望比原。 院中可真熱鬧,春花似錦杠巡、人聲如沸量窘。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,697評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)蚌铜。三九已至,卻和暖如春兄一,著一層夾襖步出監(jiān)牢的瞬間厘线,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,846評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工出革, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留造壮,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,819評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像耳璧,于是被迫代替她去往敵國(guó)和親成箫。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,665評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理旨枯,服務(wù)發(fā)現(xiàn)蹬昌,斷路器,智...
    卡卡羅2017閱讀 134,654評(píng)論 18 139
  • 計(jì)算機(jī)網(wǎng)絡(luò)概述 網(wǎng)絡(luò)編程的實(shí)質(zhì)就是兩個(gè)(或多個(gè))設(shè)備(例如計(jì)算機(jī))之間的數(shù)據(jù)傳輸攀隔。 按照計(jì)算機(jī)網(wǎng)絡(luò)的定義皂贩,通過(guò)一定...
    蛋炒飯_By閱讀 1,224評(píng)論 0 10
  • 網(wǎng)絡(luò)編程 網(wǎng)絡(luò)編程對(duì)于很多的初學(xué)者來(lái)說(shuō),都是很向往的一種編程技能昆汹,但是很多的初學(xué)者卻因?yàn)楹荛L(zhǎng)一段時(shí)間無(wú)法進(jìn)入網(wǎng)絡(luò)編...
    程序員歐陽(yáng)閱讀 2,014評(píng)論 1 37
  • 前幾天紫雨老師和錦明老師今天同時(shí)來(lái)群里給我們大家做輔導(dǎo)明刷,雖然被紫雨老師批評(píng)我也沒(méi)有怨言,我知道紫雨老師是被我...
    苦咖啡_0a98閱讀 372評(píng)論 5 4
  • 二零一五年八月的最后一天满粗,我和他相遇在高二十六班的教室里辈末。那時(shí)的我和他雖然在同一個(gè)學(xué)校待了一年,但我和他還...
    瓊晴閱讀 202評(píng)論 0 0