Java网络编程(2)

news2024/11/15 7:19:25

关于网络编程上一章内容,可以参考:
https://blog.csdn.net/Raine_Yang/article/details/128697335?spm=1001.2014.3001.5501

使用服务器处理多个客户端

一般来说,同一服务器要持续运行处理多个客户端的请求。我们可以为每一个客户端请求分配一个单独的线程进行处理,从而实现处理多客户端

服务器端

package networking;

import java.io.*;
import java.net.*;


public class MultiThreadServer implements Runnable{
	
	private int clientNo = 0;
	
	private ServerSocket serverSocket;
	
	public MultiThreadServer(int port) {
		clientNo = 0;
		try {
			serverSocket = new ServerSocket(port);
		} catch (IOException e) {
			e.printStackTrace();
		}
	}
	
	public void run() {
		try {
			while (true) {
				Socket socket = serverSocket.accept();
				clientNo++;
				InetAddress inetAddress = socket.getInetAddress();
				System.out.println(inetAddress.getHostName() + " " + inetAddress.getHostAddress());
				new Thread(new HandleClient(socket)).start();
			}
		} catch (IOException e) {
			System.err.println(e);
		}
	}

	public static void main(String[] args) {
		// TODO Auto-generated method stub

		Thread server = new Thread(new MultiThreadServer(8000));
		server.start();
	}

}

1

public MultiThreadServer(int port) {
	clientNo = 0;
	try {
		serverSocket = new ServerSocket(port);
	} catch (IOException e) {
		e.printStackTrace();
	}
}

在指定port上新建服务器socket

2

while (true) {
	Socket socket = serverSocket.accept();
	clientNo++;
	InetAddress inetAddress = socket.getInetAddress();
	System.out.println(inetAddress.getHostName() + " " + inetAddress.getHostAddress());
	new Thread(new HandleClient(socket)).start();
}

无限循环接受客户端请求。每当接收到客户端请求建立连接后创建新线程运行HandleClient

HandleClient程序:

package networking;

import java.net.*;
import java.io.*;

public class HandleClient implements Runnable{
	
	private Socket socket;
	
	public HandleClient(Socket socket) {
		this.socket = socket;
	}
	
	public void run() {
		try {
			DataInputStream inputFromClient = new DataInputStream(socket.getInputStream());
			DataOutputStream outputToClient = new DataOutputStream(socket.getOutputStream());
			
			while (true) {
				double radius = inputFromClient.readDouble();
				double area = radius * radius * Math.PI;
				outputToClient.writeDouble(area);
			}
		} catch(IOException e) {
			e.printStackTrace();
		}
	}

	public static void main(String[] args) {
		// TODO Auto-generated method stub

	}

}

该线程处理单个客户端请求,从客户端读入输入半径,返回圆的面积(和上一章同一示例)

客户端程序:

package networking;

import java.io.*;
import java.net.*;


public class Client implements Runnable{
	
	private String hostName;
	private int port;
	
	private double radius;
	
	public Client(String hostName, int port, double radius) {
		this.hostName = hostName;
		this.port = port;
		this.radius = radius;
	}
	
	public void run() {
		try {
			Socket socket = new Socket(hostName, port);
			DataOutputStream toServer = new DataOutputStream(socket.getOutputStream());
			DataInputStream fromServer = new DataInputStream(socket.getInputStream());
			while(true) {
				try {
					toServer.writeDouble(radius);
					toServer.flush();

					double area = fromServer.readDouble();
					System.out.println("Area: " + area);
					Thread.sleep(1000);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
			}
		} catch (IOException ex) {
			System.out.println(ex.toString());
		}
	}


	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Thread client1 = new Thread(new Client("localhost", 8000, 5));
		Thread client2 = new Thread(new Client("localhost", 8000, 6));
		Thread client3 = new Thread(new Client("localhost", 8000, 7));
		Thread client4 = new Thread(new Client("localhost", 8000, 8));
		client1.start();
		client2.start();
		client3.start();
		client4.start();
	}

}

这里我们创建了4个客户端同时向服务器发送请求,执行效果如下:
在这里插入图片描述
可以看到服务器同时处理多个客户端请求

网络传输对象

在之前的示例中,我们一直在使用DataInputStream和DataOutputStream传输基础数据类型。要在客户端和服务器间传输对象我们需要使用ObjectInputStream和ObjectOutputStream实现

下一示例中客户端向服务器发送一个StudentAddress类对象,服务器得到对象后写入到一个名为student.dat的文件中

StudentAddress类:

package networking;

public class StudentAddress implements java.io.Serializable {
	
	private String name;
	private String street;
	private String city;
	
	public StudentAddress(String name, String street, String city) {
		this.name = name;
		this.street = street;
		this.city = city;
	}
	
	public String getName() {
		return name;
	}
	
	public String getStreet() {
		return street;
	}
	
	public String getCity() {
		return city;
	}

}

服务器端:

package networking;

import java.io.*;
import java.net.*;

public class StudentServer implements Runnable{
	
	public void run() {
		ObjectInputStream inputFromClient = null;
		ObjectOutputStream outputToFile = null;
		
		
		try {
			ServerSocket serverSocket = new ServerSocket(8000);
			outputToFile = new ObjectOutputStream(new FileOutputStream("student.dat", true));
			
			while (true) {
				Socket socket = serverSocket.accept();
				inputFromClient = new ObjectInputStream(socket.getInputStream());
				Object object = inputFromClient.readObject();
				outputToFile.writeObject(object);
			}
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			try {
				inputFromClient.close();
				outputToFile.close();
			} catch (Exception ex) {
				ex.printStackTrace();
			}
		}
	}
	
	

	public static void main(String[] args) {
		// TODO Auto-generated method stub

		Thread server = new Thread(new StudentServer());
		server.start();
	}

}

1

outputToFile = new ObjectOutputStream(new FileOutputStream("student.dat", true));

创建输出对象数据流,将数据写入student.dat文件

2

inputFromClient = new ObjectInputStream(socket.getInputStream());
Object object = inputFromClient.readObject();
outputToFile.writeObject(object);

创建对象输入数据流,从socket得到输入数据。利用readObject()方法获取对象,并通过writeObject()方法将对象输入文件

3

finally {
	try {
		inputFromClient.close();
		outputToFile.close();
	} catch (Exception ex) {
		ex.printStackTrace();
	}
}

这里要使用finally方法的目的是保证无论前面获取对象时是否抛出异常,输入流和输出流都可以正常关闭。

客户端:

package networking;

import java.net.*;
import java.io.*;

public class StudentClient implements Runnable {
	
	public void run() {
		try {
			StudentAddress s1 = new StudentAddress("a", "a", "a");
			Socket socket = new Socket("localhost", 8000);
			ObjectOutputStream toServer = new ObjectOutputStream(socket.getOutputStream());
			toServer.writeObject(s1);
		} catch (IOException ex) {
			ex.printStackTrace();
		}
	}

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		
		Thread client = new Thread(new StudentClient());
		client.start();
		
	}

}

1

ObjectOutputStream toServer = new ObjectOutputStream(socket.getOutputStream());
toServer.writeObject(s1);

在客户端创建数据输出流,并发送StudentAddress对象

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/172009.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

@Scope 用法

参考:Scope注解 详细讲解及示例 官方文档 When used as a type-level annotation in conjunction with Component, Scope indicates the name of a scope to use for instances of the annotated type. When used as a method-level annotation in conjunction w…

【NI Multisim 14.0虚拟仪器设计——放置虚拟仪器仪表(示波器)】

目录 序言 🍍放置虚拟仪器仪表 🍉示波器 🍊🍊1.“时基”选项组 (1)标度 (2)X轴位移 (3)显示方式选择 🍊🍊2.“通道”选项组 …

函数指针与回调函数详解

目录1.函数指针2.函数指针数组3.指向函数指针数组的指针4.回调函数1.函数指针 前面我们学的: 整形指针是指向整形的指针字符指针是指向字符的指针数组指针是指向数组的指针 所以函数指针就是指向函数的指针 假如有一个int类型变量a,要取它的地址就是…

【Linux】Linux软件包管理器与Linux编辑器

文章目录🎪 Linux软件包管理器🚀 1.yum基本介绍🚀 2.yum基本命令🚀 3.关于rzsz工具的安装与使用⭐3.1 rzsz工具介绍⭐3.2 rzsz工具安装⭐3.3 rzsz工具使用🎪 Linux编辑器🚀 1.vim常用三种模式🚀…

事务管理-spring

什么是事务 - 事务是由N步数据库操作序列组成的逻辑执行单元,这系列操作要么全执行,要么全放弃执行。 • 事务的特性(ACID) - 原子性(Atomicity):事务是应用中不可再分的最小执行体。 - 一致…

MySQL使用索引的最佳指南

MySQL使用索引的最佳指南1.选择合适的字段创建索引2.尽可能的考虑建立联合索引而不是单列索引3.注意避免冗余索引4.考虑在字符串类型的字段上使用前缀索引代替普通索引5.索引失效的情况1.选择合适的字段创建索引 不为 NULL 的字段 :索引字段的数据应该尽量不为 NUL…

java面向对象,全是对象,这么多对象2023015

面向对象(一遍一遍的领悟) Java支持面向对象的三大特征:封装、继承和多态, Java提供 了private、protected和public三个访问控制修饰符来实现良好的封装,提供了extends关键字来让子类继承父类,子类继承父类…

人工智能图像形状检测算法

博主简介 博主是一名大二学生,主攻人工智能研究。感谢让我们在CSDN相遇,博主致力于在这里分享关于人工智能,c,Python,爬虫等方面知识的分享。 如果有需要的小伙伴可以关注博主,博主会继续更新的&#xff0c…

Redis下载安装与配置(linux)

一、Redis下载与安装 1.下载安装包 官网下载地址:Download | Redis 点击"Download 7.0.7",即可进行下载。 2.将安装包上传至服务器 2.1将安装包上传至/usr/local目录并解压 cd /usr/local lstar -zxvf redis-7.0.7.tar.gz2.2删除安装包 r…

第二天总结 之 商品类型管理界面的实现 之 添加和修改操作 的实现

添加和修改操作 页面跳转问题 点击修改按钮时 跳转的路径 如下 点击添加按钮时 跳转的路径如下 通过这两张图片 不难发现 跳转的是同一个jsp 但是添加操作 是不带id跳转 而修改操作是带着id跳转 所以在其 跳转的页面add_goods_type.jsp页面中 有一个这样的判断 如果没有id…

Linux常用命令——tmux命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) tmux Tmux是一个优秀的终端复用软件,类似GNU Screen,但来自于OpenBSD,采用BSD授权。 补充说明 使用它最直观的好处就是,通过一个终端登录远程主机并运行tmux后&a…

2022年HarmonyOS/OpenHarmony生态观察

一、鸿蒙生态世界快速构建升级中 HarmonyOS鸿蒙2019年正式面世,当时消费者只能在华为的智慧屏上体验;2020年,鸿蒙智联-华为面向智能硬件生态伙伴全新品牌和开放平台发布;2021年,智能手机等多种终端全面搭载HarmonyOS2…

UE4 RenderDoc笔记

1.Meh Viewer:当前DrawCall的Mesh信息,可以查看每个点的输入和输出 可以看到该DrawCall的Mesh顶点数量为510(该材质ID的Mesh三角面数)170*3,第一个顶点ID为3637 2.Texture Viewer:查看该次事件所调用的输入、输出缓…

【进阶】Spring Boot创建和使用

努力经营当下,直至未来明朗! 文章目录一、Spring Boot 概述二、Spring Boot优点三、Spring Boot项目创建1. 使用IDEA创建(社区版)2. 网页版创建(了解)四、项目目录介绍和运行1. 项目目录介绍2. 输出hello w…

Win10 Hyper-V 固定虚拟机IP地址的方法

Windows10系统Hyper-V中存在着一个名为“Default Switch”的缺省虚拟交换机,其本质上是一块虚拟网卡,其所连接的虚拟网络的类型为"Internal"(有关Hyper-V三种网络类型的含义,可参看《Hyper-V三种虚拟网络类型的理解_bol…

“深度学习”学习日记。误差反向传播法--算法实现

2023.1.18 经过学习了计算图、链式法则、加法层、乘法层、激活函数层、Affine层、Softmax层的反向传播的实现。今天来学习反向传播法的算法实现,做一次总结; 实现的思路(“学习”的步骤): 一,前提 神经…

4.Java的基础语法

小伙伴们,本篇内容让我们一起来总结学习Java的基础语法吧!😉 文章目录一、注释二、关键字三、字面量(也被叫做:常量/字面值常量)四、一些特殊字面量的书写五、变量(1)变量的定义格式:(2)输出打印变量:(3)变量的基本用法:(4)变量的注意事项:(5)变量的练习总结一、注释…

【第二章 Excel数据格式】

Excel数据格式1.Excel数据格式2.更改单元格格式2.1数值型数据的更改2.2文本型数据的更改2.3日期型数据的更改2.4日期型数据、数值型数据->文本型数据1.Excel数据格式 数字、文本和日期是最常用的三种数据格式, 数字一般右对齐,方便观测数据位数&am…

mac ganache安装以及在metamask创建ganache网络和账户导入

在做区块链本地测试时,需要测试网络、测试账户以及测试币,可以使用ganache来启动本地网络以及生成账户进行测试。 一、下载及安装ganache 首先下载ganache, 网址是这个https://trufflesuite.com/ganache/ 下载好后进行安装。 安装好以后使用…

Allegro如何让BUS线以粗线形式显示操作指导

Allegro如何让BUS线以粗线形式显示操作指导 在评估PCB布线的时候,设置好Bus线对于评估非常有帮助,Allegro不仅可以支持设置Bus组,还可以让Bus线以粗线形式显示,如下图 具体操作如下 选择Edit-PropertyFind选择nets