Java聊天室之使用Socket实现传递对象
作者:小虚竹and掘金 发布时间:2023-02-07 15:28:52
标签:Java,聊天室,Socket,传递对象
一、题目描述
题目实现:使用网络编程时,需要通过Socket传递对象。
二、解题思路
创建一个类:Student,实现序列化
Student类包含两个属性及对应的get()和set()方法
创建一个服务器类:ServerSocketFrame,继承JFrame类
写一个getserver() 方法,实例化Socket对象,启用9527当服务的端口。
创建输入流对象,用来接收客户端信息。
再定义一个getClientInfo()方法,用于接收客户端发送的信息。
对文本框添加一个事件:实现向客户端发磅信息。
创建一个客户端类:ClientSocketFrame,继承JFrame类。
写一个connect() 方法,实例化Socket对象,连接本地服务的9527端口服务。
再定义一个getClientInfo()方法,用于接收服务端发送的信息。
技术重点:
在Java中使用Socket传递对象时,该对象必须是序列化的,在Java中实现Serializable接口的类,创建的对象就是序列化对象,可以通过Socket进行传递,从而实现了使用Socket传递对象的功能。 Serializable接口在javaio包中,该接口没有方法,只是用于标识对象是可序列化的。该接口的定义如下: public interface Serializable 使用Serializable接口,可以创建序列化类。
三、代码详解
Student
package com.xiaoxuzhu;
import java.io.Serializable;
/**
* Description:
*
* @author xiaoxuzhu
* @version 1.0
*
* <pre>
* 修改记录:
* 修改后版本 修改人修改日期修改内容
* 2022/6/4.1 xiaoxuzhu2022/6/4 Create
* </pre>
* @date 2022/6/4
*/
public class Student implements Serializable { // 序列化对象类
private String id; // 类的成员变量
private String name;// 类的成员变量
public String getId() { // 成员变量的getter方法
return id;
}
public void setId(String id) {// 成员变量的setter方法
this.id = id;
}
public String getName() {// 成员变量的getter方法
return name;
}
public void setName(String name) {// 成员变量的setter方法
this.name = name;
}
}
ServerSocketFrame
package com.xiaoxuzhu;
import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.io.*;
import java.net.*;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import javax.swing.JTextField;
/**
* Description:
*
* @author xiaoxuzhu
* @version 1.0
*
* <pre>
* 修改记录:
* 修改后版本 修改人修改日期修改内容
* 2022/6/4.1 xiaoxuzhu2022/6/4 Create
* </pre>
* @date 2022/6/4
*/
public class ServerSocketFrame extends JFrame {
private JTextField tf_name;
private JTextField tf_id;
private JTextArea ta_info;
private ObjectOutputStream out = null; // 创建流对象
private ObjectInputStream in = null; // 创建流对象
private ServerSocket server; // 声明ServerSocket对象
private Socket socket; // 声明Socket对象socket
public void getserver() {
try {
server = new ServerSocket(9527); // 实例化Socket对象
ta_info.append("服务器套接字已经创建成功\n"); // 输出信息
while (true) { // 如果套接字是连接状态
ta_info.append("等待客户机的连接......\n"); // 输出信息
socket = server.accept(); // 实例化Socket对象
ta_info.append("客户机连接成功\n"); // 输出信息
out = new ObjectOutputStream(socket.getOutputStream());
in = new ObjectInputStream(socket.getInputStream());
getClientInfo(); // 调用getClientInfo()方法
}
} catch (Exception e) {
e.printStackTrace(); // 输出异常信息
}
}
private void getClientInfo() {
try {
while (true) { // 如果套接字是连接状态
Student stud = (Student)in.readObject();
if (stud!=null)
ta_info.append("接收到客户机发送的 编号为:" + stud.getId() + " 名称为:" +stud.getName() + "\n"); // 获得客户端信息
}
} catch (Exception e) {
ta_info.append("客户端已退出。\n"); // 输出异常信息
} finally {
try {
if (in != null) {
in.close();// 关闭流
}
if (socket != null) {
socket.close(); // 关闭套接字
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) { // 主方法
ServerSocketFrame frame = new ServerSocketFrame(); // 创建本类对象
frame.setVisible(true);
frame.getserver(); // 调用方法
}
public ServerSocketFrame() {
super();
setTitle("服务器端程序");
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setBounds(100, 100, 379, 260);
final JScrollPane scrollPane = new JScrollPane();
getContentPane().add(scrollPane, BorderLayout.CENTER);
ta_info = new JTextArea();
scrollPane.setViewportView(ta_info);
final JPanel panel = new JPanel();
getContentPane().add(panel, BorderLayout.NORTH);
final JLabel label = new JLabel();
label.setText("编号:");
panel.add(label);
tf_id = new JTextField();
tf_id.setPreferredSize(new Dimension(70,25));
panel.add(tf_id);
final JLabel label_1 = new JLabel();
label_1.setText("名称:");
panel.add(label_1);
tf_name = new JTextField();
tf_name.setPreferredSize(new Dimension(100,25));
panel.add(tf_name);
final JButton button = new JButton();
button.addActionListener(new ActionListener() {
public void actionPerformed(final ActionEvent e) {
Student stud = new Student();
stud.setId(tf_id.getText());
stud.setName(tf_name.getText());
try {
out.writeObject(stud);
} catch (IOException e1) {
e1.printStackTrace();
}
ta_info.append("服务器发送的 编号是:" + tf_id.getText() + " 名称是:"+tf_name.getText()+"\n"); // 将文本框中信息显示在文本域中
tf_id.setText(null); // 将文本框清空
tf_name.setText(null);
}
});
button.setText("发 送");
panel.add(button);
}
}
ClientSocketFrame
package com.xiaoxuzhu;
import java.awt.*;
import java.awt.event.*;
import java.io.*;
import java.net.*;
import javax.swing.*;
/**
* Description:
*
* @author xiaoxuzhu
* @version 1.0
*
* <pre>
* 修改记录:
* 修改后版本 修改人修改日期修改内容
* 2022/6/4.1 xiaoxuzhu2022/6/4 Create
* </pre>
* @date 2022/6/4
*/
public class ClientSocketFrame extends JFrame { // 创建类继承JFrame类
private JButton button;
private JTextField tf_name;
private JLabel label_1;
private JLabel label;
private JPanel panel;
private ObjectInputStream in = null;// 创建流对象
private ObjectOutputStream out = null;// 创建流对象
private Socket socket;// 声明Socket对象
private JTextArea ta_info = new JTextArea();// 创建JtextArea对象
private JTextField tf_id = new JTextField();// 创建JtextField对象
private Container cc;// 声明Container对象
public ClientSocketFrame() { // 构造方法
super(); // 调用父类的构造方法
setTitle("客户端程序");
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setBounds(100, 100, 373, 257);
cc = this.getContentPane(); // 实例化对象
final JScrollPane scrollPane = new JScrollPane();
getContentPane().add(scrollPane, BorderLayout.CENTER);
scrollPane.setViewportView(ta_info);
getContentPane().add(getPanel(), BorderLayout.NORTH);
}
private void connect() { // 连接套接字方法
ta_info.append("尝试连接......\n"); // 文本域中信息信息
try { // 捕捉异常
socket = new Socket("127.0.0.1", 9527); // 实例化Socket对象
while (true){
out = new ObjectOutputStream(socket.getOutputStream());
in = new ObjectInputStream(socket.getInputStream());
ta_info.append("完成连接。\n"); // 文本域中提示信息
getClientInfo();
}
} catch (Exception e) {
e.printStackTrace(); // 输出异常信息
}
}
public static void main(String[] args) { // 主方法
ClientSocketFrame clien = new ClientSocketFrame(); // 创建本例对象
clien.setVisible(true); // 将窗体显示
clien.connect(); // 调用连接方法
}
private void getClientInfo() {
try {
while (true) { // 如果套接字是连接状态
Student stud = (Student)in.readObject();
if (stud!=null)
ta_info.append("接收到服务器发送的 编号为:" + stud.getId() + " 名称为:" +stud.getName() + "\n"); // 获得服务器信息
}
} catch (Exception e) {
e.printStackTrace();
}
finally {
try {
if (in != null) {
in.close();// 关闭流
}
if (socket != null) {
socket.close(); // 关闭套接字
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
/**
* @return
*/
protected JPanel getPanel() {
if (panel == null) {
panel = new JPanel();
panel.add(getLabel());
tf_id.setPreferredSize(new Dimension(70, 25));
panel.add(tf_id);
panel.add(getLabel_1());
panel.add(getTf_name());
panel.add(getButton());
}
return panel;
}
/**
* @return
*/
protected JLabel getLabel() {
if (label == null) {
label = new JLabel();
label.setText("编号:");
}
return label;
}
/**
* @return
*/
protected JLabel getLabel_1() {
if (label_1 == null) {
label_1 = new JLabel();
label_1.setText("名称:");
}
return label_1;
}
/**
* @return
*/
protected JTextField getTf_name() {
if (tf_name == null) {
tf_name = new JTextField();
tf_name.setPreferredSize(new Dimension(100, 25));
}
return tf_name;
}
/**
* @return
*/
protected JButton getButton() {
if (button == null) {
button = new JButton();
button.addActionListener(new ActionListener() {
public void actionPerformed(final ActionEvent e) {
Student stud = new Student();
stud.setId(tf_id.getText());
stud.setName(tf_name.getText());
try {
out.writeObject(stud);
} catch (IOException e1) {
e1.printStackTrace();
}
ta_info.append("客户端发送的 编号是:" + tf_id.getText() + " 名称是:"+tf_name.getText()+"\n"); // 将文本框中信息显示在文本域中
tf_id.setText(null); // 将文本框清空
tf_name.setText(null);
}
});
button.setText("发 送");
}
return button;
}
}
服务器启动
客户端连接
服务器发送学生信息
客户端接收学生信息
客户端发送学生信息
服务端接收学生信息
来源:https://juejin.cn/post/7156811403389567006
0
投稿
猜你喜欢
- 在 Java 中,所有的异常都有一个共同的祖先 Throwable(可抛出)。Throwable 指定代码中可用异常传播机制通过 Java
- 前言这几天听朋友说JPA很好用,根本不用写sql。我在想一个程序员不写sql还能叫程序员?而且越高级的工具封装越多的工具,可拓展性和效率就非
- 一:问题描述 在已经root过的android设备下,app执行一个linux命令,app需要获取su权限,在某些a
- 本文实例实现文件上传的进度显示,我们先看看都有哪些问题我们要解决。1 上传数据的处理进度跟踪2 进度数据在用户页面的显示就这么2个问题,第一
- 用户在注册网站信息的时候基本上都要数据验证码验证。那么图片验证码功能该如何实现呢?大概步骤是:1.在内存中创建缓存图片2.设置背景色3.画边
- 一、前言虽然jdk1.9版本已经问世,但是许多其他的配套设施并不一定支持jdk1.9版本,所以这里仅带领你配置jdk1.8。而jdk1.9的
- 本文主要和大家分享介绍了关于Java JDK * 使用的相关内容,分享出来供大家参考学习,下面来一起看看详细的介绍:前言代理是一种常用的
- 周末这天手痒,正好没事干,想着写一个分页的例子出来给大家分享一下。这个案例分前端和后台两部分,前端使用面向对象的方式写的,里面用到了一些回调
- 本文实例讲述了Java String类简单用法。分享给大家供大家参考,具体如下:一 String类的实例化方式1 代码public clas
- 导读Spring Boot应用可以使用spring-boot-maven-plugin快速打包,构建一个可执行jar。Spring Boot
- 前言在Windows下JAVA用到的环境变量主要有3个,JAVA_HOME、CLASSPATH、PATH,下面来详细的介绍。JAVA_HOM
- 假设下面是你的视频网站链接列表,如果别人想爬取你的数据十分轻松,看规则就知道数据库是序列自增的http://www.xxxx.com/vid
- 前言最近在改进项目的并发功能,但开发起来磕磕碰碰的。看了好多资料,总算加深了认识。于是打算配合查看源代码,总结并发编程的原理。准备从用得最多
- 本文实例讲述了java中struts2实现文件上传下载功能实现方法。分享给大家供大家参考。具体分析如下:1.文件上传首先是jsp页面的代码在
- List接口是Collection接口的子接口,List有一个重要的实现类--ArrayList类,List中的元素是有序排列的而且可重复,
- spring缓存cache的使用在spring配置文件中添加schema和spring对缓存注解的支持:<?xml version=&
- 1概述众所周知,Java支持平台无关性、安全性和网络移动性。而Java平台由Java虚拟机和Java核心类所构成,它为纯Java程序提供了统
- 一、ArrayList类概述什么是集合:提供一种存储空间可变的存储模型,存储的数据容量可以发生改变ArrayList集合的特点:底层是数组实
- 1.导包(1)c3p0 数据库连接池c3p0配置文件加入到src目录下(2)dbutils:对jdbc操作进行了封装it-cast工具包 包
- 什么是NIO?线程在处理数据时,如果线程还处于将数据从channel读到buffer的这段时间内,线程可以去做别的事情,等数据都读到buff