开发思路:
服务端两个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. javascript-初级-day01-属性操作、图片切换、短信发送模拟

    大多数js就是操作一些css和html的技巧,如果你会html和css学习js更加轻松哦! js中如何获取元素: 通过id名称来获取元素; document get element by id 'li ...

  2. CSS动画--让div动起来

    CSS动画 今天在写代码时候,遇到了css动画效果如何实现的问题,经过查阅和实践,总结出一下结论. transition transition 指定动画变化的对应属性 以及动画的执行时间. 例如:tr ...

  3. uniapp增加自定义埋点功能

    起因 首先来说,uniapp其实是自带系统埋点统计功能的.基本也算是面面俱到. 但是一些未知原因,貌似数据有所丢失,再加上没有一些重要的定制化功能,以及最重要的数据安全方面的考虑,还是决定接入公司的埋 ...

  4. freeswitch APR库哈希表

    概述 freeswitch的核心源代码是基于apr库开发的,在不同的系统上有很好的移植性. 哈希表在开发中应用的非常广泛,主要场景是对查询效率要求较高的逻辑,是典型的空间换时间的数据结构实现. 大多数 ...

  5. vcstool是什么?

    为什么会去了解vcstool,在想要手动编译并且获取ROS源码的时候,有一个Get ROS 2 code的章节中使用到了这个工具. mkdir -p ~/ros2_foxy/src cd ~/ros2 ...

  6. CF611F New Year and Cleaning

    题意 CF611F New Year and Cleaning 想法 这个题是\(NOIP2020\)的弱化版.. 我们把所有在二维上的点都一起考虑,那么所有点对于一个步骤的移动是相当于这些所有点所组 ...

  7. NOIP 2020 游记

    第一次写比赛游记,请多多指教! I. 考前 由于最近参加了太多太多比赛了,所以没有敲模板题: 考前一周:主要是在做 AtCoder 的题和 xjoi 的模拟赛,相当于恶补了一些套路吧! 考前一天:上午 ...

  8. 快速傅里叶变换(FFT)随笔

    终于学会了FFT,水一篇随笔记录一下 前置知识网上一大堆,这里就不多赘述了,直接切入正题 01 介绍FFT 这里仅指出FFT在竞赛中的一般应用,即优化多项式乘法 一般情况下,计算两个规模为$n$的多项 ...

  9. [linux] 常用命令及参数-2

    sort 1 sort是把结果输出到标准输出,因此需要输出重定向将结果写入文件 2 sort seq.txt > file.txt 3 sort -u seq.txt 输出去重重复后的行 4 s ...

  10. SpringBoot集成Kafka的实战用法大全

    本文是SpringBoot+Kafka的实战讲解,如果对kafka的架构原理还不了解的读者,建议先看一下<大白话kafka架构原理>.<秒懂kafka HA(高可用)>两篇文章 ...