概述
网络编程三要素
TCP通信
TCP通信文件上传案例
解决文件阻塞问题(使用多线程)
软件结构
- C/S结构
- B/S结构
网络通信协议
java.net包中提供了2中常用的协议
UDP协议:用户数据报协议
- 无连接通信协议,即数据传输时,数据的发送端和接收端不建立逻辑连接
- 耗资小,效率高(视频会议)
- 偶尔会丢失数据包,不可靠
- 发送数据限制在64K
TCP协议:传输控制协议
- 面向连接的通信协议
- 无差错的可靠的数据传输
- TCP连接中要明确客户端和服务器,每次连接时要创建三次握手
三次握手
- 第一次握手:客户端向服务器发出连接请求,等待服务器确认
- 第二次握手:服务器向客户端回送一个响应,通知客户端接收到了连接请求
- 第三次握手:客户端再次向服务器发送确认信息,确认连接
- 完成三次握手,建立连接后,客户端和服务器就可以建立数据传输了,由于是面向连接的特性,TCP协议保证传输的安全性
网络编程三要素
协议
IP地址
- IPV4:是一个32位的二进制数,但是常用十进制表示
- IPV6:是一个128位的二进制数
端口号
常用命令
查看IP地址
ipconfig
查看网络是否连通
ping 空格 IP地址
特殊的IP地址
- 本机的IP地址:127.0.0.1、localhost
TCP通信协议
TCP通信
面向连接的通信方式,客户端可服务器必须通过三次握手,建立逻辑连接,才能通信(安全)
通信步骤
- 服务器先启动
- 服务器不会主动请求客户端
- 必须客户端请求服务器端
- 客户端和服务器端建立一个逻辑连接
- 而这个连接包含一个对象
- 这个对象就是IO对象
- 客户端和服务器就可以使用IO对象进行通信
- 通信的对象不仅仅是字符,索引O对象是字节流对象
综合案例
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
/**
* TCP通信的客户端:向服务器发送连接请求,给服务器发送数据,接收服务器回写的数据
* 表示客户端的类:
* java.net.Socket;此类实现客户端套接字()。套接字是俩台机器之间通信的节点
* 套接字:包含了IP地址和端口号的网络单位
* 构造方法:
* Socket(String host, int port) 创建一个套接字并将其连接到指定主机的指定端口号
* 参数:
* host:服务器主机名称/IP地址
* port:服务器端口号
* 成员方法:
* OutputStream getOutputStream() 返回此套接字的输出流
* InputStream getInputStream() 返回此套接字的输入流
* 实现步骤:
* 1、创建一个客户端Socket对象,参数绑定服务器的IP地址和端口号
* 2、使用Socket对象中的getOutputStream()获取网络字节输出流的OutputStream对象
* 3、使用网络字节输出流OutputStream对象的write()方法,向服务器发送数据
* 4、使用Socket对象中的方法getInputStream()方法获取网络字节输入流InputStream对象
* 5、使用网络字节输入流的中的read()方法,读取服务器回写的数据
* 6、关闭资源
* 注意:
* 1、客户端和服务器进行交互,必须使用Socket中提供的网络流,不能使用自己创建的流对象
* 2、当我们创建客户端对象Socket的时候,就会去请求服务器和服务器建立三次握手建立通路
* 如果这时候服务器没有启动,就会抛出异常
* 如果服务器已经启动,则进行交互
*/
public class TCPClient {
public static void main(String[] args) throws IOException {
//1、创建Socket对象
Socket socket = new Socket("127.0.0.1", 8888);
//2、使用Socket对象中的getOutputStream()获取网络字节输出流的OutputStream对象
OutputStream os = socket.getOutputStream();
//3、使用网络字节输出流OutputStream对象的write()方法,向服务器发送数据
os.write("你好服务器".getBytes());
//4、使用Socket对象中的方法getInputStream()方法获取网络字节输入流InputStream对象
InputStream is = socket.getInputStream();
//5、使用网络字节输入流的中的read()方法,读取服务器回写的数据
byte[] bytes = new byte[1024];
int len = is.read(bytes);
System.out.println(new String(bytes, 0, len));
//6、关闭资源
socket.close();
}
}
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
/**
* TCP通信的服务器端:接收客户端发送的请求,接收客户端发送的数据,给客户端回写数据
* 表示服务器的类:
* java.net.ServerSocket; 此类实现服务器套接字
* 构造方法:
* ServerSocket(int port)
* 创建绑定到特定端口的服务器套接字。
* 注意:
* 服务器端必须知道哪个客户端请求的服务器
* 所以可以使用accept()方法获取请求的Socket对象
* 特有成员方法
* Socket accept()
* 侦听并接受到此套接字的连接。
* 实现步骤:
* 1、创建服务器ServerSocket对象和系统要指定的端口号
* 2、使用ServerSocket对象的accept()方法,获取请求道的客户端Socket对象
* 3、使用Socket对象中的方法getInputStream()方法获取网络字节输入流InputStream对象
* 4、使用网络字节输入流的中的read()方法,读取客户端发送的数据
* 5、使用Socket对象中的getOutputStream()获取网络字节输出流的OutputStream对象
* 6、使用网络字节输出流OutputStream对象的write()方法,给客户端写数据
* 7、关闭资源(Socket,ServerSocket)
*/
public class TCPServer {
public static void main(String[] args) throws IOException {
//1、创建服务器ServerSocket对象和系统要指定的端口号
ServerSocket server = new ServerSocket(8888);
//2、使用ServerSocket对象的accept()方法,获取请求道的客户端Socket对象
Socket socket = server.accept();
//3、使用Socket对象中的方法getInputStream()方法获取网络字节输入流InputStream对象
InputStream is = socket.getInputStream();
//4、使用网络字节输入流的中的read()方法,读取客户端发送的数据
byte[] bytes = new byte[1024];
int len = is.read(bytes);
System.out.println(new String(bytes, 0, len));
//5、使用Socket对象中的getOutputStream()获取网络字节输出流的OutputStream对象
OutputStream os = socket.getOutputStream();
os.write("收到谢谢".getBytes());
socket.close();
server.close();
}
}
TCP通信文件上传案例
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
/**
* 文件上传案例的客户端:读取本地文件,上传到服务器,读取服务器回写的数据
* 明确:
* 数据源:D:\\a.txt
* 目的地:服务器
* 实现步骤:
* 1、创建一个本地字节输入流FileInputStream对象,构造方法中绑定要读取的数据源
* 2、创建一个客户端Socket对象,构造方法中绑定服务器的IP地址和端口号
* 3、使用Socket中的getOutputStream方法,获取网络字节输出流OutputStream对象
* 4、使用本地字节输入流FileInputStream中的read()方法,读取本地文件
* 5、使用网络字节输出流OutputStream中的方法write(),把读取到的文件上传到服务器
* 6、使用Socket中的getInputStream方法,获取网络字节输出流InputStream对象
* 7、使用网络字节输入流InputStream中的read()方法,读取服务器写回的数据
* 8、释放资源(FileInputStream, socket)
*
*/
public class TCPClient {
public static void main(String[] args) throws IOException {
//1、创建一个本地字节输入流FileInputStream对象,构造方法中绑定要读取的数据源
FileInputStream fis = new FileInputStream("D:\\b.jpg");
//2、创建一个客户端Socket对象,构造方法中绑定服务器的IP地址和端口号
Socket socket = new Socket("127.0.0.1", 8888);
//3、使用Socket中的getOutputStream方法,获取网络字节输出流OutputStream对象
OutputStream os = socket.getOutputStream();
//4、使用本地字节输入流FileInputStream中的read()方法,读取本地文件
int len;
byte[] bytes = new byte[1024];
while ((len = fis.read(bytes)) != -1) {
//5、使用网络字节输出流OutputStream中的方法write(),把读取到的文件上传到服务器
os.write(bytes);
}
/**
* 解决:上传完文件,给文件写一个结束标记
* void shutdownOutput() 禁用此套接字的输出流
* 对于TCP套接字,任何以前写入的数据都将被发送,并且后跟TCP的正常连接终止序列
*/
socket.shutdownOutput();
//6、使用Socket中的getInputStream方法,获取网络字节输出流InputStream对象
InputStream is = socket.getInputStream();
while ((len = is.read(bytes)) != -1) {
//7、使用网络字节输入流InputStream中的read()方法,读取服务器写回的数据
System.out.println(new String(bytes, 0, len));
}
//8、关闭资源
fis.close();
socket.close();
}
}
import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
/**
* 文件上传案例的服务器:读取客户端上传的文件,保存到文件的硬盘,给客户端写回上传成功
* 明确:
* 数据源:客户端上传的 文件
* 目的地:服务器的硬盘 D:\\upload
* 实现步骤:
* * 1、创建服务器ServerSocket对象和系统要指定的端口号
* * 2、使用ServerSocket对象的accept()方法,获取请求道的客户端Socket对象
* * 3、使用Socket对象中的方法getInputStream()方法获取网络字节输入流InputStream对象
* 4、判断目的地文件夹是否存在,不存在则创建
* 5、创建一个本地字节输出流FileOutputStream对象,构造方法中绑定要输出的目的地
* * 6、使用网络字节输入流的中的read()方法,读取客户端上传的文件
* 7、使用本地字节输出流FileOutputStream对象中的write()方法,把读取到的文件保存到服务器的硬盘
* * 8、使用Socket对象中的getOutputStream()获取网络字节输出流的OutputStream对象
* * 9、使用网络字节输出流OutputStream对象的write()方法,给客户端写回上传成功
* * 10、关闭资源(Socket,ServerSocket, Socket)
*/
public class TCPServer {
public static void main(String[] args) throws IOException {
//1、创建服务器ServerSocket对象和系统要指定的端口号
ServerSocket server = new ServerSocket(8888);
//2、使用ServerSocket对象的accept()方法,获取请求道的客户端Socket对象
Socket socket = server.accept();
//3、使用Socket对象中的方法getInputStream()方法获取网络字节输入流InputStream对象
InputStream is = socket.getInputStream();
//4、判断目的地文件夹是否存在,不存在则创建
File file = new File("D:\\upload");
if (!file.exists()) {
file.mkdirs();
}
//5、创建一个本地字节输出流FileOutputStream对象,构造方法中绑定要输出的目的地
FileOutputStream fos = new FileOutputStream(file + "\\1.jpg");
//6、使用网络字节输入流的中的read()方法,读取客户端上传的文件
byte[] bytes = new byte[1024];
int len = 0;
while ((len = is.read(bytes)) != -1) {
//7、使用本地字节输出流FileOutputStream对象中的write()方法,把读取到的文件保存到服务器的硬盘
fos.write(bytes);
}
//8、使用Socket对象中的getOutputStream()获取网络字节输出流的OutputStream对象
OutputStream os = socket.getOutputStream();
//9、使用网络字节输出流OutputStream对象的write()方法,给客户端写回上传成功
os.write("上传成功".getBytes());
//10、关闭资源
socket.close();
fos.close();
server.close();
}
}
解决程序阻塞问题以及优化(使用多线程)
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
/**
* 文件上传案例的客户端:读取本地文件,上传到服务器,读取服务器回写的数据
* 明确:
* 数据源:D:\\a.txt
* 目的地:服务器
* 实现步骤:
* 1、创建一个本地字节输入流FileInputStream对象,构造方法中绑定要读取的数据源
* 2、创建一个客户端Socket对象,构造方法中绑定服务器的IP地址和端口号
* 3、使用Socket中的getOutputStream方法,获取网络字节输出流OutputStream对象
* 4、使用本地字节输入流FileInputStream中的read()方法,读取本地文件
* 5、使用网络字节输出流OutputStream中的方法write(),把读取到的文件上传到服务器
* 6、使用Socket中的getInputStream方法,获取网络字节输出流InputStream对象
* 7、使用网络字节输入流InputStream中的read()方法,读取服务器写回的数据
* 8、释放资源(FileInputStream, socket)
*
*/
public class TCPClient {
public static void main(String[] args) throws IOException {
//1、创建一个本地字节输入流FileInputStream对象,构造方法中绑定要读取的数据源
FileInputStream fis = new FileInputStream("D:\\b.jpg");
//2、创建一个客户端Socket对象,构造方法中绑定服务器的IP地址和端口号
Socket socket = new Socket("127.0.0.1", 8888);
//3、使用Socket中的getOutputStream方法,获取网络字节输出流OutputStream对象
OutputStream os = socket.getOutputStream();
//4、使用本地字节输入流FileInputStream中的read()方法,读取本地文件
int len;
byte[] bytes = new byte[1024];
while ((len = fis.read(bytes)) != -1) {
//5、使用网络字节输出流OutputStream中的方法write(),把读取到的文件上传到服务器
os.write(bytes);
}
/**
* 解决:上传完文件,给文件写一个结束标记
* void shutdownOutput() 禁用此套接字的输出流
* 对于TCP套接字,任何以前写入的数据都将被发送,并且后跟TCP的正常连接终止序列
*/
socket.shutdownOutput();
//6、使用Socket中的getInputStream方法,获取网络字节输出流InputStream对象
InputStream is = socket.getInputStream();
while ((len = is.read(bytes)) != -1) {
//7、使用网络字节输入流InputStream中的read()方法,读取服务器写回的数据
System.out.println(new String(bytes, 0, len));
}
//8、关闭资源
fis.close();
socket.close();
}
}
/**
* 文件上传案例的服务器:读取客户端上传的文件,保存到文件的硬盘,给客户端写回上传成功
* 明确:
* 数据源:客户端上传的 文件
* 目的地:服务器的硬盘 D:\\upload
* 实现步骤:
* 1、创建服务器ServerSocket对象和系统要指定的端口号
* 2、使用ServerSocket对象的accept()方法,获取请求道的客户端Socket对象
* 3、使用Socket对象中的方法getInputStream()方法获取网络字节输入流InputStream对象
* 4、判断目的地文件夹是否存在,不存在则创建
* 5、创建一个本地字节输出流FileOutputStream对象,构造方法中绑定要输出的目的地
* 6、使用网络字节输入流的中的read()方法,读取客户端上传的文件
* 7、使用本地字节输出流FileOutputStream对象中的write()方法,把读取到的文件保存到服务器的硬盘
* 8、使用Socket对象中的getOutputStream()获取网络字节输出流的OutputStream对象
* 9、使用网络字节输出流OutputStream对象的write()方法,给客户端写回上传成功
* 10、关闭资源(Socket,ServerSocket, Socket)
public class TCPServer {
public static void main(String[] args) throws IOException {
//1、创建服务器ServerSocket对象和系统要指定的端口号
ServerSocket server = new ServerSocket(8888);
/**
* 让服务器一直处于监听状态(死循环accept)
* 有一个客户端上传文件,就保存文件
*/
while (true) {
//2、使用ServerSocket对象的accept()方法,获取请求道的客户端Socket对象
Socket socket = server.accept();
/**
* 使用多线程技术,提高程序效率
* 有一个客户端上传文件,就开启一个线程,完成文件的上传
*/
new Thread(new Runnable() {
@Override
public void run() {
try {
//3、使用Socket对象中的方法getInputStream()方法获取网络字节输入流InputStream对象
InputStream is = socket.getInputStream();
//4、判断目的地文件夹是否存在,不存在则创建
File file = new File("D:\\upload");
if (!file.exists()) {
file.mkdirs();
}
/**
* 自定义一个文件的命名规则:防止同名文件被覆盖
* 规则:域名+毫秒值+随机数
*/
String fileName = "itcast" + System.currentTimeMillis() + new Random().nextInt(999999) + ".jpg";
//5、创建一个本地字节输出流FileOutputStream对象,构造方法中绑定要输出的目的地
//FileOutputStream fos = new FileOutputStream(file + "\\1.jpg");
FileOutputStream fos = new FileOutputStream(file + "\\" + fileName);
//6、使用网络字节输入流的中的read()方法,读取客户端上传的文件
byte[] bytes = new byte[1024];
int len = 0;
while ((len = is.read(bytes)) != -1) {
//7、使用本地字节输出流FileOutputStream对象中的write()方法,把读取到的文件保存到服务器的硬盘
fos.write(bytes);
}
//8、使用Socket对象中的getOutputStream()获取网络字节输出流的OutputStream对象
OutputStream os = socket.getOutputStream();
//9、使用网络字节输出流OutputStream对象的write()方法,给客户端写回上传成功
os.write("上传成功".getBytes());
//10、关闭资源
socket.close();
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}).start();
}
//此时服务器不用关闭
//server.close();
}
}