1、网络编程基础
1.1 为什么需要网络编程?
1.2 什么是网络编程
当然,我们只要满足进程不同就行;所以即便是同一个主机,只要是不同进程,基于网络来传输数据,也属于网络编程。特殊的,对于开发来说,在条件有限的情况下,一般也都是在一个主机中运行多个进程来完成网络编程。但是,我们一定要明确,我们的目的是提供网络上不同主机,基于网络来传输数据资源:
- 进程A:编程来获取网络资源
- 进程B:编程来提供网络资源
1.3 网络编程中的基本概念
- 发送端和接收端
在一次网络数据传输时:发送端 :数据的 发送方进程 ,称为发送端。发送端主机即网络通信中的源主机。接收端 :数据的 接收方进程 ,称为接收端。接收端主机即网络通信中的目的主机。收发端 :发送端和接收端两端,也简称为收发端。注意:发送端和接收端只是相对的,只是一次网络数据传输产生数据流向后的概念。
- 请求和响应
第一次:请求数据的发送
第二次:响应数据的发送。
- 客户端和服务端
客户端保存资源在服务端
好比在银行办事:银行提供存款服务:用户(客户端)保存资源(现金)在银行(服务端)。银行提供取款服务:用户(客户端)获取服务端资源(银行替用户保管的现金)
1.4 常见的客户端服务端模型
1. 客户端先发送请求到服务端2. 服务端根据请求数据,执行相应的业务处理3. 服务端返回响应:发送业务处理结果4. 客户端根据响应数据,展示处理结果(展示获取的资源,或提示保存资源的处理结果)
2、Socket套接字
2.1 概念
2.2 分类
- 流套接字:使用传输层TCP协议
有连接可靠传输面向字节流有接收缓冲区,也有发送缓冲区大小不限
- 数据报套接字:使用传输层UDP协议
无连接不可靠传输面向数据报有接收缓冲区,无发送缓冲区大小受限:一次最多传输64k
- 原始套接字
2.3 Java数据报套接字通信模型
以上只是一次发送端的UDP数据报发送,及接收端的数据报接收,并没有返回的数据。也就是只有请求,没有响应。对于一个服务端来说,重要的是提供多个客户端的请求处理及响应,流程如下:
2.4 Java流套接字通信模型
2.5 Socket编程注意事项
1. 客户端和服务端:开发时,经常是基于一个主机开启两个进程作为客户端和服务端,但真实的场 景,一般都是不同主机。
2. 注意目的IP和目的端口号,标识了一次数据传输时要发送数据的终点主机和进程
3. Socket编程我们是使用流套接字和数据报套接字,基于传输层的TCP或UDP协议,但应用层议,也需要考虑,这块我们在后续来说明如何设计应用层协议。
4. 关于端口被占用的问题:如果一个进程A已经绑定了一个端口,再启动一个进程B绑定该端口,就会报错,这种情况也叫端口被占用。对于java进程来说,端口被占用的常见报错信息如下:
此时需要检查进程B绑定的是哪个端口,再查看该端口被哪个进程占用。以下为通过端口号查进程 的方式:
- 在cmd输入 netstat -ano | findstr 端口号 ,则可以显示对应进程的pid。如以下命令显示了8888进程的pid
- 在任务管理器中,通过pid查找进程
解决端口被占用的问题:如果占用端口的进程A不需要运行,就可以关闭A后,再启动需要绑定该端口的进程B。如果需要运行A进程,则可以修改进程B的绑定端口,换为其他没有使用的端口。
3、UDP数据报套接字编程
3.1 DatagramSocket API
DatagramSocket 方法:
3.2 DatagramPacket API
DatagramPacket 方法:
构造UDP发送的数据报时,需要传入 SocketAddress ,该对象可以使用 InetSocketAddress 来创建。
3.3 InetSocketAddress API
3.4 示例一:一发一收(无响应)
package org.example.udp.demo1;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.util.Arrays;
public class UdpServer {
//服务器socket要绑定固定的端口
private static final int PORT = 8888;
public static void main(String[] args) throws IOException {
// 1.创建服务端DatagramSocket,指定端口,可以发送及接收UDP数据报
DatagramSocket socket = new DatagramSocket(PORT);
//不停的接收客户端udp数据报
while (true){
// 2.创建数据报,用于接收客户端发送的数据
byte[] bytes = new byte[1024];//1m=1024kb, 1kb=1024byte, UDP最多
64k(包含UDP首部8byte)
DatagramPacket packet = new DatagramPacket(bytes, bytes.length);
System.out.println("------------------------------------------------
---");
System.out.println("等待接收UDP数据报...");
// 3.等待接收客户端发送的UDP数据报,该方法在接收到数据报之前会一直阻塞,接收到数
据报以后,DatagramPacket对象,包含数据(bytes)和客户端ip、端口号
socket.receive(packet);
System.out.printf("客户端IP:%s%n",
packet.getAddress().getHostAddress());
System.out.printf("客户端端口号:%s%n", packet.getPort());
System.out.printf("客户端发送的原生数据为:%s%n",
Arrays.toString(packet.getData()));
System.out.printf("客户端发送的文本数据为:%s%n", new
String(packet.getData()));
}
}
}
---------------------------------------------------等待接收 UDP 数据报 ...
package org.example.udp.demo1;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
public class UdpClient {
// 服务端socket地址,包含域名或IP,及端口号
private static final SocketAddress ADDRESS = new
InetSocketAddress("localhost", 8888);
public static void main(String[] args) throws IOException {
// 4.创建客户端DatagramSocket,开启随机端口就行,可以发送及接收UDP数据报
DatagramSocket socket = new DatagramSocket();
// 5-1.准备要发送的数据
byte[] bytes = "hello world!".getBytes();
// 5-2.组装要发送的UDP数据报,包含数据,及发送的服务端信息(服务器IP+端口号)
DatagramPacket packet = new DatagramPacket(bytes, bytes.length,
ADDRESS);
// 6.发送UDP数据报
socket.send(packet);
}
}
---------------------------------------------------等待接收 UDP 数据报 ...客户端 IP : 127 . 0.0 . 1客户端端口号: 57910客户端发送的原生数据为: [ 104 , 101 , 108 , 108 , 111 , 32 , 119 , 111 , 114 , 108 , 100 , 33 ,0 , 0 , 0 , ... 此处省略很多 0 ]客户端发送的文本数据为: hello world !---------------------------------------------------等待接收 UDP 数据报 ...
- 1. 如果N>M,则接收的byte[]字节数组中会有很多初始化byte[]的初始值0,转换为字符串就是空白字符;
- 2. 如果N<M,则会发生数据部分丢失(可以自己尝试,把接收的字节数组长度指定为比发送的字节数组长度更短)。
要解决以上问题,就需要发送端和接收端双方约定好一致的协议,如规定好结束的标识或整个数据的长度。
3.5 示例二:请求响应
以下为服务端和客户端代码:
package org.example.udp.demo2;
import java.io.File;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.nio.charset.StandardCharsets;
public class UdpServer {
//服务器socket要绑定固定的端口
private static final int PORT = 8888;
//本地文件目录要展示的根路径
private static final String BASE_PATH = "E:/TMP";
public static void main(String[] args) throws IOException {
// 1.创建服务端DatagramSocket,指定端口,可以发送及接收UDP数据报
DatagramSocket socket = new DatagramSocket(PORT);
//不停的接收客户端udp数据报
while (true){
// 2.创建数据报,用于接收客户端发送的数据
byte[] requestData = new byte[1024];//1m=1024kb, 1kb=1024byte, UDP最 多64k(包含UDP首部8byte)
DatagramPacket requestPacket = new DatagramPacket(requestData,
requestData.length);
System.out.println("------------------------------------------------
---");
System.out.println("等待接收UDP数据报...");
// 3.等待接收客户端发送的UDP数据报,该方法在接收到数据报之前会一直阻塞,接收到数
据报以后,DatagramPacket对象,包含数据(bytes)和客户端ip、端口号
socket.receive(requestPacket);
System.out.printf("客户端IP:%s%n",
requestPacket.getAddress().getHostAddress());
System.out.printf("客户端端口号:%s%n", requestPacket.getPort());
// 7.接收到的数据作为请求,根据请求数据执行业务,并返回响应
for (int i = 0; i < requestData.length; i++) {
byte b = requestData[i];
if(b == '\3') {
// 7-1.读取请求的数据:读取到约定好的结束符(\3),取结束符之前的内容
String request = new String(requestData, 0, i);
// 7-2.根据请求处理业务:本地目录根路径+请求路径,作为要展示的目录,列
出下一级子文件
//请求的文件列表目录
System.out.printf("客户端请求的文件列表路径为:%s%n", BASE_PATH +
request);
File dir = new File(BASE_PATH + request);
//获取下一级子文件,子文件夹
File[] children = dir.listFiles();
// 7-3.构造要返回的响应内容:每个文件及目录名称为一行
StringBuilder response = new StringBuilder();
if(children != null){
for (File child : children) {
response.append(child.getName()+"\n");
}
}
//响应也要约定结束符
response.append("\3");
byte[] responseData =
response.toString().getBytes(StandardCharsets.UTF_8);
// 7-4.构造返回响应的数据报DatagramPacket,注意接收的客户端数据报包
含IP和端口号,要设置到响应的数据报中
DatagramPacket responsePacket = new
DatagramPacket(responseData, responseData.length,
requestPacket.getSocketAddress());
// 7-5.发送返回响应的数据报
socket.send(responsePacket);
break;
}
}
}
}
}
---------------------------------------------------等待接收 UDP 数据报 ...
package org.example.udp.demo2;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.nio.charset.StandardCharsets;
import java.util.Scanner;
public class UdpClient {
// 服务端socket地址,包含域名或IP,及端口号
private static final SocketAddress ADDRESS = new
InetSocketAddress("localhost", 8888);
public static void main(String[] args) throws IOException {
// 4.创建客户端DatagramSocket,开启随机端口就行,可以发送及接收UDP数据报
DatagramSocket socket = new DatagramSocket();
// 5-1.准备要发送的数据:这里调整为键盘输入作为发送的内容
Scanner scanner = new Scanner(System.in);
while(true){
System.out.println("------------------------------------------------
---");
System.out.println("请输入要展示的目录:");
// 5-2.每输入新行(回车),就作为UDP发送的数据报,为了接收端获取有效的内容(去除
空字符串),约定\3为结束
String request = scanner.nextLine() + "\3";
byte[] requestData = request.getBytes(StandardCharsets.UTF_8);
// 5-3.组装要发送的UDP数据报,包含数据,及发送的服务端信息(服务器IP+端口号)
DatagramPacket requestPacket = new DatagramPacket(requestData,
requestData.length, ADDRESS);
// 6.发送UDP数据报
socket.send(requestPacket);
// 8.接收服务端响应的数据报,并根据响应内容决定下个步骤(我们这里简单的打印即可)
// 8-1.创建数据报,用于接收服务端返回(发送)的响应
byte[] responseData = new byte[1024];
DatagramPacket responsePacket = new DatagramPacket(responseData,
responseData.length);
// 8-2.接收响应数据报
socket.receive(responsePacket);
System.out.println("该目录下的文件列表为:");
// byte[]下次解析的起始位置
int next = 0;
for (int i = 0; i < responseData.length; i++) {
byte b = responseData[i];
if(b == '\3')//结束符退出
break;
if(b == '\n'){//换行符时进行解析
//起始位置到换行符前一个索引位置为要解析的内容
String fileName = new String(responseData, next, i-next);
System.out.println(fileName);
//下次解析从换行符后一个索引开始
next = i+1;
}
}
}
}
}
---------------------------------------------------请输入要展示的目录:
---------------------------------------------------请输入要展示的目录:/该目录下的文件列表为:1260441 b1b8a74be3695ccc0d9706938158f25103 aa249707ee4ab17635142cd0e---------------------------------------------------请输入要展示的目录:
---------------------------------------------------等待接收 UDP 数据报 ...客户端 IP : 127 . 0.0 . 1客户端端口号: 57297客户端请求的文件列表路径为: E : / TMP /---------------------------------------------------等待接收 UDP 数据报 ...
4、TCP流套接字编程
4.1 ServerSocket API
ServerSocket 方法:
4.2 Socket API
Socket 方法:
4.3 TCP中的长短连接
- 短连接:每次接收到数据并返回响应后,都关闭连接,即是短连接。也就是说,短连接只能一次收发数据。
- 长连接:不关闭连接,一直保持连接状态,双方不停的收发数据,即是长连接。也就是说,长连接可以多次收发数据。
- 建立连接、关闭连接的耗时:短连接每次请求、响应都需要建立连接,关闭连接;而长连接只需要第一次建立连接,之后的请求、响应都可以直接传输。相对来说建立连接,关闭连接也是要耗时的,长连接效率更高。
- 主动发送请求不同:短连接一般是客户端主动向服务端发送请求;而长连接可以是客户端主动发送请求,也可以是服务端主动发。
- 两者的使用场景有不同:短连接适用于客户端请求频率不高的场景,如浏览网页等。长连接适用于客户端与服务端通信频繁的场景,如聊天室,实时游戏等。
4.4 示例一:一发一收(短连接)
package org.example.tcp.demo1;
import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
public class TcpServer {
//服务器socket要绑定固定的端口
private static final int PORT = 8888;
public static void main(String[] args) throws IOException {
// 1.创建一个服务端ServerSocket,用于收发TCP报文
ServerSocket server = new ServerSocket(PORT);
// 不停的等待客户端连接
while(true) {
System.out.println("------------------------------------------------
---");
System.out.println("等待客户端建立TCP连接...");
// 2.等待客户端连接,注意该方法为阻塞方法
Socket client = server.accept();
System.out.printf("客户端IP:%s%n",
client.getInetAddress().getHostAddress());
System.out.printf("客户端端口号:%s%n", client.getPort());
// 5.接收客户端的数据,需要从客户端Socket中的输入流获取
System.out.println("接收到客户端请求:");
InputStream is = client.getInputStream();
// 为了方便获取字符串内容,可以将以上字节流包装为字符流
BufferedReader br = new BufferedReader(new InputStreamReader(is,
"UTF-8"));
String line;
// 一直读取到流结束:TCP是基于流的数据传输,一定要客户端关闭Socket输出流才表示服
务端接收IO输入流结束
while ((line = br.readLine()) != null) {
System.out.println(line);
}
// 6.双方关闭连接:服务端是关闭客户端socket连接
client.close();
}
}
}
---------------------------------------------------等待客户端建立 TCP 连接 ...
package org.example.tcp.demo1;
import java.io.*;
import java.net.Socket;
public class TcpClient {
//服务端IP或域名
private static final String SERVER_HOST = "localhost";
//服务端Socket进程的端口号
private static final int SERVER_PORT = 8888;
public static void main(String[] args) throws IOException {
// 3.创建一个客户端流套接字Socket,并与对应IP的主机上,对应端口的进程建立连接
Socket client = new Socket(SERVER_HOST, SERVER_PORT);
// 4.发送TCP数据,是通过socket中的输出流进行发送
OutputStream os = client.getOutputStream();
// 为了方便输出字符串作为发送的内容,可以将以上字节流包装为字符流
PrintWriter pw = new PrintWriter(new OutputStreamWriter(os, "UTF-8"));
// 4-1.发送数据:
pw.println("hello world!");
// 4-2.有缓冲区的IO操作,真正传输数据,需要刷新缓冲区
pw.flush();
// 7.双方关闭连接:客户端关闭socket连接
client.close();
}
}
---------------------------------------------------等待客户端建立 TCP 连接 ...客户端 IP : 127 . 0.0 . 1客户端端口号: 51118接收到客户端请求:hello world !---------------------------------------------------等待客户端建立 TCP 连接...
4.5 示例二:请求响应(短连接)
package org.example.tcp.demo2;
import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
public class TcpServer {
//服务器socket要绑定固定的端口
private static final int PORT = 8888;
//本地文件目录要展示的根路径
private static final String BASE_PATH = "E:/TMP";
public static void main(String[] args) throws IOException {
// 1.创建一个服务端ServerSocket,用于收发TCP报文
ServerSocket server = new ServerSocket(PORT);
// 不停的等待客户端连接
while(true) {
System.out.println("------------------------------------------------
---");
System.out.println("等待客户端建立TCP连接...");
// 2.等待客户端连接,注意该方法为阻塞方法
Socket socket = server.accept();
System.out.printf("客户端IP:%s%n",
socket.getInetAddress().getHostAddress());
System.out.printf("客户端端口号:%s%n", socket.getPort());
// 5.接收客户端的数据,需要从客户端Socket中的输入流获取
InputStream is = socket.getInputStream();
// 为了方便获取字符串内容,可以将以上字节流包装为字符流
BufferedReader br = new BufferedReader(new InputStreamReader(is,
"UTF-8"));
// 客户端请求只发送一行数据,我们也只需要读取一行
String request = br.readLine();
// 6.根据请求处理业务:本地目录根路径+请求路径,作为要展示的目录,列出下一级子文
件
//请求的文件列表目录
System.out.printf("客户端请求的文件列表路径为:%s%n", BASE_PATH +
request);
File dir = new File(BASE_PATH + request);
//获取下一级子文件,子文件夹
File[] children = dir.listFiles();
// 7.返回响应给客户端:通过客户端socket中的输出流发送响应数据
OutputStream os = socket.getOutputStream();
// 为了方便输出字符串作为发送的内容,可以将以上字节流包装为字符流
PrintWriter pw = new PrintWriter(new OutputStreamWriter(os, "UTF-
8"));
// 7-1.返回的响应内容:每个文件及目录名称为一行
if(children != null){
for (File child : children) {
pw.println(child.getName());
}
}
// 7-2.有缓冲区的IO操作,真正传输数据,需要刷新缓冲区
pw.flush();
// 7-3.双方关闭连接:服务端是关闭客户端socket连接
socket.close();
}
}
}
---------------------------------------------------等待客户端建立 TCP 连接 ...
package org.example.tcp.demo2;
import java.io.*;
import java.net.Socket;
import java.util.Scanner;
public class TcpClient {
//服务端IP或域名
private static final String SERVER_HOST = "localhost";
//服务端Socket进程的端口号
private static final int SERVER_PORT = 8888;
public static void main(String[] args) throws IOException {
// 准备要发送的数据:这里调整为键盘输入作为发送的内容
Scanner scanner = new Scanner(System.in);
while(true) {
System.out.println("------------------------------------------------
---");
System.out.println("请输入要展示的目录:");
// 每输入新行(回车),就作为发送的TCP请求报文
String request = scanner.nextLine();
// 3.创建一个客户端流套接字Socket,并与对应IP的主机上,对应端口的进程建立连接
Socket socket = new Socket(SERVER_HOST, SERVER_PORT);
// 4.发送TCP数据,是通过socket中的输出流进行发送
OutputStream os = socket.getOutputStream();
// 为了方便输出字符串作为发送的内容,可以将以上字节流包装为字符流
PrintWriter pw = new PrintWriter(new OutputStreamWriter(os, "UTF-
8"));
// 4-1.发送数据:
pw.println(request);
// 4-2.有缓冲区的IO操作,真正传输数据,需要刷新缓冲区
pw.flush();
// 8.接收返回的响应数据:通过socket中的输入流获取
System.out.println("接收到服务端响应:");
InputStream is = socket.getInputStream();
// 为了方便获取字符串内容,可以将以上字节流包装为字符流
BufferedReader br = new BufferedReader(new InputStreamReader(is,
"UTF-8"));
String line;
// 一直读取到流结束:TCP是基于流的数据传输,一定要服务端关闭Socket输出流才表示客
户端接收的IO输入流结束
while ((line = br.readLine()) != null) {
System.out.println(line);
}
// 9.双方关闭连接:客户端关闭socket连接
socket.close();
}
}
}
---------------------------------------------------请输入要展示的目录:
---------------------------------------------------请输入要展示的目录:/接收到服务端响应:1260441 b1b8a74be3695ccc0d9706938158f25103 aa249707ee4ab17635142cd0e---------------------------------------------------请输入要展示的目录:
---------------------------------------------------等待客户端建立 TCP 连接 ...客户端 IP : 127 . 0.0 . 1客户端端口号: 52493客户端请求的文件列表路径为: E : / TMP /---------------------------------------------------等待客户端建立 TCP 连接 ...
- 对于服务端来说,处理一次请求并返回响应后,才能再次处理下一次请求和响应,效率是比较低的。这个问题比较好解决:可以使用多线程,每次的请求与响应都在线程中处理。这样多个客户端请求的话,可以在多个线程中并发并行的执行。
- 服务端解析请求,是只读取了一行,而客户端解析响应,是一直读取到流结束。可以想想为什么解析请求时,没有读取到流结束?
4.6 再谈协议
- 客户端发送请求和服务端解析请求要使用相同的数据格式。
- 服务端返回响应和客户端解析响应也要使用相同的数据格式。
- 请求格式和响应格式可以相同,也可以不同。
- 约定相同的数据格式,主要目的是为了让接收端在解析的时候明确如何解析数据中的各个字段。
- 可以使用知名协议(广泛使用的协议格式),如果想自己约定数据格式,就属于自定义协议。
- 如果是使用知名协议,这个动作也称为封装
- 如果是使用小众协议(包括自定义协议),这个动作也称为序列化,一般是将程序中的对象转换为特定的数据格式。
- 如果是使用知名协议,这个动作也称为分用
- 如果是使用小众协议(包括自定义协议),这个动作也称为反序列化,一般是基于接收数据特定的格式,转换为程序中的对象
- 可以基于长度约定,如int字段,约定好4个字节即可
- 可以约定字段之间的间隔符,或最后一个字段的结束符,如换行符间隔,\3符号结束等等
- 除了该字段“数据”本身,再加一个长度字段,用来标识该“数据”长度;即总共使用两个字段:
4.7 示例三:多线程+自定义协议
- 提供多种操作:展示目录下文件列表,文件重命名,删除文件,上传文件,下载文件
- 在不同的操作中,需要抽象出请求和响应的字段,也即是说,要约定客户端服务端统一的请求协议,同时也要约定服务端与客户端统一的响应协议
请求类型操作的文件或目录路径数据\3
- 以上总共包含3个字段,前2个字段需要按换行符读取,最后一个字段需要按结束符读取
- 请求类型标识是什么操作:展示目录下文件列表,文件重命名,删除文件,上传文件,下载文件
- 重命名、上传文件操作,需要“数据”字段,其他操作可以置为空字符串
- “数据”字段为最后一个字段,使用\3结束符,这样在数据本身有换行符也能正确处理
状态码(标识是否操作成功)数据(展示列表时,返回目录下的文件列表,或下载文件的数据)\3
请求数据格式如下:
1/\3
200\1\2\3\1.txt\2.txt\3
4/1package org.example;public class Main {…… 略}\3
200\3
执行流程
代码实现如下:
- 每个字段为一个属性:操作类型,操作路径,数据
- 完成服务端解析请求封装:按约定的方式读,先按行读取前2个字段,再按结束符读第3个字段
- 完成客户端发送请求封装:按约定的方式写,前2个字段按行输出,第3个字段以\3结束
package org.example.tcp.demo3;
import java.io.*;
import java.util.ArrayList;
import java.util.List;
public class Request {
//操作类型:1(展示目录文件列表),2(文件重命名),3(删除文件),4(上传文件),5(下载
文件)
private Integer type;
//操作的目录路径
private String url;
//数据
private String data;
//服务端解析请求时:根据约定好的格式来解析
public static Request serverParse(InputStream is) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(is, "UTF-
8"));
Request request = new Request();
//前2行分别为操作类型和操作路径
request.type = Integer.parseInt(br.readLine());
request.url = br.readLine();
//使用list保存字符
List<Character> list = new ArrayList<>();
//数据:循环读取
while(true){
//一个字符一个字符的读
char c = (char) br.read();
//一直读取到结束符\3
if(c == '\3')
break;
list.add(c);
}
//拼接数据
StringBuilder sb = new StringBuilder();
for (char c : list){
sb.append(c);
}
request.data = sb.toString();
return request;
}
//客户端发送请求到服务端
public void clientWrite(OutputStream os) throws IOException {
PrintWriter pw = new PrintWriter(os);
pw.println(type);
pw.println(url);
pw.write(data+"\3");
// 4-2.有缓冲区的IO操作,真正传输数据,需要刷新缓冲区
pw.flush();
}
@Override
public String toString() {
return "Request{" +
"type=" + type +
", url='" + url + '\'' +
", data='" + data + '\'' +
'}';
}
public Integer getType() {
return type;
}
public void setType(Integer type) {
this.type = type;
}
public String getUrl() {
return url;
}
public void setUrl(String url) {
this.url = url;
}
public String getData() {
return data;
}
public void setData(String data) {
this.data = data;
}
}
- 每个字段为一个属性:响应状态码
- 完成客户端解析响应封装:按约定的方式读,先按行读取第1个字段,再按结束符读第2个字段
- 完成服务端发送响应封装:按约定的方式写,第1个字段按行输出,第2个字段以\3结束
package org.example.tcp.demo3;
import java.io.*;
import java.util.ArrayList;
import java.util.List;
public class Response {
//响应的状态码,200表示操作成功,404表示没有找到该路径的文件或目录
private int status;
//数据
private String data;
//客户端解析服务端返回的响应数据
public static Response clientParse(InputStream is) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(is, "UTF-
8"));
Response response = new Response();
response.status = Integer.parseInt(br.readLine());
//使用list保存字符
List<Character> list = new ArrayList<>();
//数据:循环读取
while(true){
//一个字符一个字符的读
char c = (char) br.read();
//一直读取到结束符\3
if(c == '\3')
break;
list.add(c);
}
//拼接数据
StringBuilder sb = new StringBuilder();
for (char c : list){
sb.append(c);
}
response.data = sb.toString();
return response;
}
//服务端返回响应给客户端
public void serverWrite(OutputStream os) throws IOException {
PrintWriter pw = new PrintWriter(new OutputStreamWriter(os, "UTF-8"));
pw.println(status);
pw.write(data+"\3");
// 4-2.有缓冲区的IO操作,真正传输数据,需要刷新缓冲区
pw.flush();
}
@Override
public String toString() {
return "Response{" +
"status=" + status +
", data='" + data + '\'' +
'}';
}
public int getStatus() {
return status;
}
public void setStatus(int status) {
this.status = status;
}
public String getData() {
return data;
}
public void setData(String data) {
this.data = data;
}
}
- ServerSocket.accept() 为建立客户端服务端连接的方法,为提高效率,使用多线程
- 先要解析请求数据,即 Request 已封装好的服务端解析请求,返回 Request 对象
- 返回响应数据,需要根据不同的请求字段,做不同的业务处理,并返回对应的响应内容
package org.example.tcp.demo3;
import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
import java.nio.file.Files;
import java.util.UUID;
public class TcpServer {
//服务器socket要绑定固定的端口
private static final int PORT = 8888;
//本地文件目录要展示的根路径
private static final String BASE_PATH = "E:/TMP";
public static void main(String[] args) throws IOException {
// 1.创建一个服务端ServerSocket,用于收发TCP报文
ServerSocket server = new ServerSocket(PORT);
// 不停的等待客户端连接
while(true) {
// 2.等待客户端连接,注意该方法为阻塞方法
Socket socket = server.accept();
new Thread(new Runnable() {
@Override
public void run() {
try {
System.out.println("------------------------------------
---------------");
System.out.println("客户端建立TCP连接...");
System.out.printf("客户端IP:%s%n",
socket.getInetAddress().getHostAddress());
System.out.printf("客户端端口号:%s%n", socket.getPort());
// 5.接收客户端的数据,需要从客户端Socket中的输入流获取
InputStream is = socket.getInputStream();
// 解析为请求对象
Request request = Request.serverParse(is);
System.out.println("服务端收到请求:"+request);
// 6.根据请求处理业务:处理完成返回响应对象
Response response = build(request);
// 7.返回响应给客户端:通过客户端socket中的输出流发送响应数据
OutputStream os = socket.getOutputStream();
// 7-1.返回的响应内容:按照约定格式输出响应对象中的内容
System.out.println("服务端返回响应:"+response);
response.serverWrite(os);
// 7-3.双方关闭连接:服务端是关闭客户端socket连接
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}).start();
}
}
//根据请求处理业务,返回响应对象
public static Response build(Request request){
Response response = new Response();
response.setStatus(200);
File url = new File(BASE_PATH+request.getUrl());
//该路径的文件或目录不存在
if(!url.exists()){
response.setStatus(404);
response.setData("");
return response;
}
try {
switch (request.getType()){
//1展示目录文件列表
case 1: {
File[] children = url.listFiles();
if(children == null){
response.setData("");
}else{
//拼接要返回的数据:文件列表
StringBuilder sb = new StringBuilder();
for (int i = 0; i < children.length; i++) {
File child = children[i];
//文件路径截取掉服务端本地路径前缀
sb.append(child.getAbsolutePath().substring(BASE_PATH.length())+"\n");
}
response.setData(sb.toString());
}
break;
}
//2文件重命名
case 2: {
url.renameTo(new
File(url.getParent()+File.separator+request.getData()));
break;
}
//3删除文件
case 3: {
url.delete();
break;
}
//上传文件
case 4: {
//上传到请求的操作路径目录下,保存的文件名简单的以随机字符串uuid生成即
可
FileWriter upload = new
FileWriter(url.getAbsolutePath()+File.separator+ UUID.randomUUID());
upload.write(request.getData());
upload.flush();
upload.close();
break;
}
//下载文件
case 5: {
String data = new String(Files.readAllBytes(url.toPath()));
response.setData(data);
break;
}
}
} catch (IOException e) {
e.printStackTrace();
}
return response;
}
}
- 先要建立和服务端的连接,连接服务端的IP和端口
- 根据输入来构建请求数据:
- 解析响应数据,并根据响应来执行相应的业务,我们这里暂时简单的解析为 Response 对象,并打印即可
package org.example.tcp.demo3;
import java.io.*;
import java.net.Socket;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.Scanner;
public class TcpClient {
//服务端IP或域名
private static final String SERVER_HOST = "localhost";
//服务端Socket进程的端口号
private static final int SERVER_PORT = 8888;
public static void main(String[] args) throws IOException {
// 准备要发送的数据:这里调整为键盘输入作为发送的内容
Scanner scanner = new Scanner(System.in);
while(true) {
//根据键盘输入构造一个请求对象,包含操作类型,操作路径,长度和数据
Request request = build(scanner);
// 3.创建一个客户端流套接字Socket,并与对应IP的主机上,对应端口的进程建立连接
Socket socket = new Socket(SERVER_HOST, SERVER_PORT);
// 4.发送TCP数据,是通过socket中的输出流进行发送
OutputStream os = socket.getOutputStream();
// 4-1.发送请求数据:按照约定的格式输出请求对象中的内容
System.out.println("客户端发送请求:"+request);
request.clientWrite(os);
// 8.接收返回的响应数据:通过socket中的输入流获取
InputStream is = socket.getInputStream();
// 根据约定的格式获取响应数据
Response response = Response.clientParse(is);
System.out.println("客户端收到响应:"+response);
// 9.双方关闭连接:客户端关闭socket连接
socket.close();
}
}
//客户端发送请求时,根据键盘输入构造一个请求对象
public static Request build(Scanner scanner) throws IOException {
System.out.println("---------------------------------------------------
");
System.out.println("请输入要操作的类型:1(展示目录文件列表),2(文件重命名),
3(删除文件),4(上传文件),5(下载文件)");
Request request = new Request();
int type = Integer.parseInt(scanner.nextLine());
System.out.println("请输入要操作的路径:");
String url = scanner.nextLine();
String data = "";
//只需要操作类型和操作路径的请求,长度和数据构造为空的
if(type == 2){//重命名操作,需要输入重命名的名称
System.out.println("请输入要重命名的名称:");
data = scanner.nextLine();
}else if(type == 4){//文件上传,需要提供上传的文件路径
System.out.println("请输入要上传的文件路径:");
String upload = scanner.nextLine();
data = new String(Files.readAllBytes(Paths.get(upload)));
}else if(type != 1 && type !=3 && type!=5){
System.out.println("只能输入1-5的数字,请重新输入");
return build(scanner);
}
request.setType(type);
request.setUrl(url);
request.setData(data);
return request;
}
}