一、 概述
TCP通信能实现两台计算机之间的数据交互,通信的两端,要严格区分为客户端(Client)与服务端(Server)。
两端通信时步骤:
- 服务端程序,需要事先启动,等待客户端的连接。
- 客户端主动连接服务器端,连接成功才能通信。服务端不可以主动连接客户端。
在Java中,提供了两个类用于实现TCP通信程序:
- 客户端:
java.net.Socket
类表示。创建Socket
对象,向服务端发出连接请求,服务端响应请求,两者建立连接开始通信。 - 服务端:
java.net.ServerSocket
类表示。创建ServerSocket
对象,相当于开启一个服务,并等待客户端的连接。
二、Socket类
Socket
类:该类实现客户端套接字,套接字指的是两台设备之间通讯的端点。
构造方法
-
public Socket(String host, int port)
:创建套接字对象并将其连接到指定主机上的指定端口号。如果指定的host是null ,则相当于指定地址为回送地址。小贴士:回送地址(127.x.x.x) 是本机回送地址(Loopback Address),主要用于网络软件测试以及本地机进程间通信,无论什么程序,一旦使用回送地址发送数据,立即返回,不进行任何网络传输。
构造举例,代码如下:
Socket client = new Socket("127.0.0.1", 7777);
成员方法
-
public InputStream getInputStream()
: 返回此套接字的输入流。- 如果此Scoket具有相关联的通道,则生成的InputStream 的所有操作也关联该通道。
- 关闭生成的InputStream也将关闭相关的Socket。
-
public OutputStream getOutputStream()
: 返回此套接字的输出流。- 如果此Scoket具有相关联的通道,则生成的OutputStream 的所有操作也关联该通道。
- 关闭生成的OutputStream也将关闭相关的Socket。
-
public void close()
:关闭此套接字。- 一旦一个socket被关闭,它不可再使用。
- 关闭此socket也将关闭相关的InputStream和OutputStream 。
-
public void shutdownOutput()
: 禁用此套接字的输出流。- 任何先前写出的数据将被发送,随后终止输出流。
三、 ServerSocket类
ServerSocket
类:这个类实现了服务器套接字,该对象等待通过网络的请求。
构造方法
public ServerSocket(int port)
:使用该构造方法在创建ServerSocket对象时,就可以将其绑定到一个指定的端口号上,参数port就是端口号。
构造举例,代码如下:
ServerSocket server = new ServerSocket(7777);
成员方法
public Socket accept()
:侦听并接受连接,返回一个新的Socket对象,用于和客户端实现通信。该方法会一直阻塞直到建立连接。
四、 简单的TCP网络程序
TCP通信分析图解
- 【服务端】启动,创建ServerSocket对象,等待连接。
- 【客户端】启动,创建Socket对象,请求连接。
- 【服务端】接收连接,调用accept方法,并返回一个Socket对象。
- 【客户端】Socket对象,获取OutputStream,向服务端写出数据。
- 【服务端】Scoket对象,获取InputStream,读取客户端发送的数据。
到此,客户端向服务端发送数据成功。
自此,服务端向客户端回写数据。
- 【服务端】Socket对象,获取OutputStream,向客户端回写数据。
- 【客户端】Scoket对象,获取InputStream,解析回写数据。
- 【客户端】释放资源,断开连接。
代码如下:
package test0811.demo1;
import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
/**
* 服务器端
*/
public class Server {
public static void main(String[] args) throws IOException {
//第一步:实例化Socket服务器,并指定端口通讯
ServerSocket ss = new ServerSocket(9001);
System.out.println("服务器被创建");
System.out.println("服务器已经启动,并开始监听9001端口,等待客服端连接中...");
System.out.println("accept阻塞,一直到客服端连接上,才会往后执行");
//第二步:监听客服端连接,并同意客户端连接,并建立通讯管道,也就是获得通讯的socket
Socket socket = ss.accept();
System.out.println(socket);
//第三步:socket管道中拿输入输出流,并读写出数据
InputStream is = socket.getInputStream();
//把字节流转换为字符流
BufferedReader br = new BufferedReader(
new InputStreamReader(is));
String s = br.readLine();
s += ",我也不知道我是谁!";
OutputStream os = socket.getOutputStream();
os.write(s.getBytes());
//第四步:关闭资源
os.close();
br.close();
is.close();
socket.close();
ss.close();
}
}
package test0811.demo1;
import java.io.*;
import java.net.Socket;
/**
* 客户端
*/
public class Client {
public static void main(String[] args) throws IOException {
//第一步:实例化客户端socket对象,并建立和服务器的通信管理
Socket socket = new Socket("localhost",9001);
//第二步:通过socket拿流,来传输数据
OutputStream os = socket.getOutputStream();
String s = "我是谁?";
//打印流
PrintStream pw = new PrintStream(os,true);
pw.println(s);
InputStream is = socket.getInputStream();
BufferedReader br = new BufferedReader(
new InputStreamReader(is));
s = br.readLine();
System.out.println(s);
//第四步:关闭资源
br.close();
is.close();
pw.close();
os.close();
socket.close();
}
}
运行结果如下: