开发思路:
服务端两个Socket:
  1. UDP的DatagramSocket接收客户端消息。
  2. 广播multicastSocket 发送广播消息

客户端一个Socket:

  1. multicastSocket 接收服务端发来的广播 (不允许客户端直接发广播),同时发送点对点数据到UDP服务端DatagramSocket 。
  2. javax.swing.* 提供客户端图形化界面

生疏知识点:

图形化界面按钮监控键盘

addActionListener(AbstractAction接口实现类);监听器。
topTextField.getInputMap().put(KeyStroke.getKeyStroke('\n'),"confirm_username"); 键盘事件和名字关联;
topTextField.getActionMap().put("confirm_username",confirmListener); 事件别名和监听器关联
服务器端程序:
package tcpandudp.udpexample;

import java.io.IOException;
import java.net.*; /**
* @ClassName MulticastServerScoket
* @projectName: object1
* @author: Zhangmingda
* @description: 设计思想:为避免所有人直接发广播,导致消息内容无环节把控,消息发送的时候,发送到非广播端口(作为一个服务端)再由此服务端向广播地址发送消息,
* date: 2021/5/13.
*/
public class MulticastServerScoket {
/**
* 广播Socket端口
*/
private static final int BROAD_PORT = 30000;
/**
* 广播IP地址 :224.0.0.0 到 239.255.255.255
*/
private static final String BROAD_IP = "230.3.3.3";
/**
* 服务端口
*/
private static final int SERVER_PORT = 8888;
private void start(){
try (
/**
* 广播用套接字
*/
MulticastSocket multicastSocket = new MulticastSocket(BROAD_PORT);
/**
* 服务端套接字
*/
DatagramSocket datagramSocket = new DatagramSocket(SERVER_PORT)
){
/**
* 广播IP地址
*/
InetAddress multi_addr = InetAddress.getByName(BROAD_IP);
/**
* 广播套接字配置接收广播的IP地址
*/
multicastSocket.joinGroup(multi_addr);
/**
* 构建接收UDP数据的包
*/
byte[] inBuff = new byte[4096];
DatagramPacket inPacket = new DatagramPacket(inBuff,inBuff.length);
/**
* 接收广播,并广播出去
*/
while (true){
//普通套接字收广播
datagramSocket.receive(inPacket); String content = new String(inPacket.getData(),0,inPacket.getLength());
System.out.println(content);
//广播套接字发广播
DatagramPacket outPacket = new DatagramPacket(inPacket.getData(),inPacket.getLength(),multi_addr,BROAD_PORT);
multicastSocket.send(outPacket);
}
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* 程序入口
*/
public static void main(String[] args) {
new MulticastServerScoket().start();
}
}

客户端

package tcpandudp.udpexample;

import javax.swing.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.IOException;
import java.net.*;
import java.util.Scanner;
import java.util.concurrent.atomic.AtomicReference; /**
* @ClassName MulticastClientSocket
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/5/13.
*/
public class MulticastClientSocket {
/**
* 广播Socket端口
*/
private static final int BROAD_PORT = 30000;
/**
* 广播IP地址 :224.0.0.0 到 239.255.255.255
*/
private static final String BROAD_IP = "230.3.3.3"; /**
* 要发送数据包的服务端IP地址和 端口
*/
private static final String SERVER_IP = "127.0.0.1";
private static final int SERVER_PORT = 8888; private void go() {
JFrame jFrame = new JFrame("广播聊天客户端");
//上部
JPanel topPanel = new JPanel();
JTextField topTextField = new JTextField(30);
JButton topButton = new JButton("确认用户名");
topPanel.add(topTextField);
topPanel.add(topButton);
jFrame.add(topPanel, BorderLayout.NORTH);
//中部
JTextArea centerArea = new JTextArea(10,40);
JScrollPane centerPane = new JScrollPane(centerArea);
jFrame.add(centerPane);
//下
JPanel bottomPanel = new JPanel();
JTextField bottomTextField = new JTextField(35);
JButton sendButton = new JButton("发送");
bottomPanel.add(bottomTextField);
bottomPanel.add(sendButton);
jFrame.add(bottomPanel, BorderLayout.SOUTH); jFrame.setLocation(400,300);
jFrame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
jFrame.pack();
jFrame.setVisible(true);
AtomicReference<String> username = new AtomicReference<>("");
//确认用户名事件监听器
Action confirmListener = new AbstractAction() {
@Override
public void actionPerformed(ActionEvent actionEvent) {
String topTextContent = topTextField.getText();
if (topTextContent != null && ! "".equals(topTextContent)){
username.set(topTextContent);
System.out.println("用户名设置成功");
}
}
};
//绑定事件监听器
topButton.addActionListener(confirmListener);
topTextField.getInputMap().put(KeyStroke.getKeyStroke('\n'),"confirm_username");
topTextField.getActionMap().put("confirm_username",confirmListener);
/**
* 发送&接收用socket
*/
try{
MulticastSocket mClinetSocket = new MulticastSocket(BROAD_PORT);
InetAddress multi_addr = InetAddress.getByName(BROAD_IP);
/**
* 接收端
*/
new Thread(){
@Override
public void run() {
byte[] inBuff = new byte[4096];
/**
* 构建接收包
*/
DatagramPacket inPacket = new DatagramPacket(inBuff,0,inBuff.length);
while (true){
/**
* 接收数据放包里,转为字符串打印出来
*/
try {
mClinetSocket.receive(inPacket);
} catch (IOException e) {
e.printStackTrace();
}
String content = new String(inPacket.getData(),0, inPacket.getLength());
System.out.println(content);
centerArea.append(content + "\n"); //向文本框加入文本
}
}
}.start();
/**
* 套接字加入广播地址
*/
mClinetSocket.joinGroup(multi_addr); /**
* 服务端地址构建
*/
InetAddress server_addr = InetAddress.getByName(SERVER_IP);
/**
* 构建发出包
*/
DatagramPacket outPacket = new DatagramPacket(new byte[0],0,server_addr,SERVER_PORT);
/**
* 发送按钮监听器
*/
Action sendListener = new AbstractAction() {
@Override
public void actionPerformed(ActionEvent actionEvent) {
if (username.get().equals("")){
System.err.println("请输入用户名");
}else {
String sendContent = bottomTextField.getText();
if (sendContent != null && ! "".equals(sendContent)){
sendContent = username + ":" + sendContent;
//输入字节存入包,
outPacket.setData(sendContent.getBytes());
try {
mClinetSocket.send(outPacket);
bottomTextField.setText(null);
} catch (IOException ex) {
ex.printStackTrace();
}
}
}
}
};
/**
* 发送按钮绑定监听器
*/
sendButton.addActionListener(sendListener);
//内容框监听键盘
bottomTextField.getInputMap().put(KeyStroke.getKeyStroke('\n'),"send");
bottomTextField.getActionMap().put("send",sendListener);
} catch (IOException e) {
e.printStackTrace();
} } public static void main(String[] args) {
new MulticastClientSocket().go();
}
}

java 网络编程基础 UDP协议DatagramSocket,MulticastSocket 实现局域网聊天软件(Server + Client模式)的更多相关文章

  1. java 网络编程基础 UDP协议的Socket:DatagramSocket;广播Socket:MulticastSocket

    什么是UDP协议: UDP协议是一种不可靠的网络协议,它在通信实例的两端各建立一个Socket 但这两个 Socket之间并没有虚拟链路,这两个Socket只是发送.接收数据报的对象.Java 提供了 ...

  2. java:网络编程(UDP (DatagramSocket和DatagramPacket)正则表达式)

    java:网络编程(UDP (DatagramSocket和DatagramPacket)正则表达式) * TCP* 特点:面向连接,点对点的通信,效率较低,但安全可靠* UDP:用户数据报协议,类似 ...

  3. Java网络编程和NIO详解开篇:Java网络编程基础

    Java网络编程和NIO详解开篇:Java网络编程基础 计算机网络编程基础 转自:https://mp.weixin.qq.com/s/XXMz5uAFSsPdg38bth2jAA 我们是幸运的,因为 ...

  4. Java网络编程基础(Netty预备知识)

    今天在家休息,闲来无事,写篇博客,陶冶下情操~~~ =================我是分割线================ 最近在重新学习Java网络编程基础,以便后续进行Netty的学习. 整 ...

  5. java网络编程基础——TCP网络编程一

    基于TCP协议的网络编程 TCP/IP协议是一种可靠的网络协议,它的通信的两端各自建立一个Socket,从而在通信的两端之间形成网络虚拟链路. Java使用Socket对象来代表两端的通信端口,并通过 ...

  6. java网络编程基础——网络基础

    java网络编程 网络编程基础 1.常用的网络拓扑结构: 星型网络.总线网络.环线网络.树形网络.星型环线网络 2.通信协议的组成 通信协议通常由3部分组成: 语义部分:用于决定通信双方对话类型 语法 ...

  7. JAVA基础知识之网络编程——-基于UDP协议的通信例子

    UDP是一种不可靠的协议,它在通信两端各建立一个socket,这两个socket不会建立持久的通信连接,只会单方面向对方发送数据,不检查发送结果. java中基于UDP协议的通信使用DatagramS ...

  8. Java学习笔记52(网络编程:UDP协议案例)

    InetAddress类: 表示互联网中的IP地址,示例: package demo; import java.net.InetAddress; import java.net.UnknownHost ...

  9. java 网络编程基础 TCP/IP协议:服务端ServerSocket;客户端Socket; 采用多线程方式处理网络请求

    1.Java中客户端和服务器端通信的简单实例 Java中能接收其他通信实体连接请求的类是ServerSocket,ServerSocket对象用于监听来自客户端的Socket连接,如果没有连接,它将一 ...

随机推荐

  1. Apache ShardingSphere 5.0.0 内核优化及升级指南

    经过近两年时间的优化和打磨,Apache ShardingSphere 5.0.0 GA 版终于在本月正式发布,相比于 4.1.1 GA 版,5.0.0 GA 版在内核层面进行了大量的优化.首先,基于 ...

  2. CF#581 (div2)题解

    CF#581 题解 A BowWow and the Timetable 如果不是4幂次方直接看位数除以二向上取整,否则再减一 #include<iostream> #include< ...

  3. R语言与医学统计图形-【30】流行病学数据可视化

    sjPlot包适用于社会科学.流行病学中调查数据可视化,且能和SPSS数据无缝对接(流行病学问卷调查录入Epidata软件后,都会转成SPSS格式或EXCEL格式保存). 辅助包sjmisc进行数据转 ...

  4. 关于AnnotationHub的一些应用

    AnnotationHub是一个包含大量注释信息的数据库,里面有很多物种,以及来源于很多数据库的注释信息. 1,安装这个包 source("https://bioconductor.org/ ...

  5. chmod文件权限分配问题

    一. 文件(文件夹)的权限问题 一个文件或者文件夹,使用它的人有三类:root.当前用户和其他用户,例如,我们可以通过 ls -l xxx.xxx 来查看文件 "xxx.xxx" ...

  6. Azure Key Vault(二)- 入门简介

    一,引言 在介绍 Azure Key Vault 之前,先简单介绍一下 HSM(硬件安全模块). -------------------- 我是分割线 -------------------- 1,什 ...

  7. adjust, administer

    adjust to just, exact. In measurement technology and metrology [度量衡学], calibration [校准] is the compa ...

  8. day06 视图层

    day06 视图层 今日内容 视图层 小白必会三板斧 JsonResponse form表单发送文件 FBV与CBV FBV基于函数的视图 CBV基于类的视图 模板层 模板语法的传值 模板语法之过滤器 ...

  9. 位运算符在JS中的妙用

    正文 位运算 JavaScript 中最臭名昭著的 Bug 就是 0.1 + 0.2 !== 0.3,因为精度的问题,导致所有的浮点运算都是不安全的,具体原因可详见<0.1 + 0.2不等于0. ...

  10. OpenStack之之一: 快速添加计算节点

    根据需求创建脚本,可以快速添加节点#:初始化node节点 [root@node2 ~]# systemctl disable NetworkManager [root@node2 ~]# vim /e ...