TCP的Java支持
协议至关于相互通讯的程序间达成的一种约定,它规定了分组报文的结构、交换方式、包含的意义以及怎样对报文所包含的信息进行解析,TCP/IP协议族有IP协议、TCP协议和UDP协议。如今TCP/IP协议族中的主要socket类型为流套接字(使用TCP协议)和数据报套接字(使用UDP协议)。编程
TCP协议提供面向链接的服务,经过它创建的是可靠地链接。Java为TCP协议提供了两个类:Socket类和ServerSocket类。一个Socket实例表明了TCP链接的一个客户端,而一个ServerSocket实例表明了TCP链接的一个服务器端,通常在TCP Socket编程中,客户端有多个,而服务器端只有一个,客户端TCP向服务器端TCP发送链接请求,服务器端的ServerSocket实例则监听来自客户端的TCP链接请求,并为每一个请求建立新的Socket实例,因为服务端在调用accept()等待客户端的链接请求时会阻塞,直到收到客户端发送的链接请求才会继续往下执行代码,所以要为每一个Socket链接开启一个线程。服务器端要同时处理ServerSocket实例和Socket实例,而客户端只须要使用Socket实例。另外,每一个Socket实例会关联一个InputStream和OutputStream对象,咱们经过将字节写入套接字的OutputStream来发送数据,并经过从InputStream来接收数据。服务器
TCP链接的创建步骤
客户端向服务器端发送链接请求后,就被动地等待服务器的响应。典型的TCP客户端要通过下面三步操做:多线程
一、建立一个Socket实例:构造函数向指定的远程主机和端口创建一个TCP链接;socket
2.经过套接字的I/O流与服务端通讯;ide
三、使用Socket类的close方法关闭链接。函数
服务端的工做是创建一个通讯终端,并被动地等待客户端的链接。典型的TCP服务端执行以下两步操做:this
一、建立一个ServerSocket实例并指定本地端口,用来监听客户端在该端口发送的TCP链接请求;spa
二、重复执行:.net
1)调用ServerSocket的accept()方法以获取客户端链接,并经过其返回值建立一个Socket实例;
2)为返回的Socket实例开启新的线程,并使用返回的Socket实例的I/O流与客户端通讯;
3)通讯完成后,使用Socket类的close()方法关闭该客户端的套接字链接。
TCP Socket Demo
下面给出一个客户端服务端TCP通讯的Demo,该客户端在20006端口请求与服务端创建TCP链接,客户端不断接收键盘输入,并将其发送到服务端,服务端在接收到的数据前面加上“echo”字符串,并将组合后的字符串发回给客户端,如此循环,直到客户端接收到键盘输入“bye”为止。
客户端代码以下:
package zyb.org.client;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintStream;
import java.net.Socket;
import java.net.SocketTimeoutException;
public class Client1 {
public static void main(String[] args) throws IOException {
//客户端请求与本机在20006端口创建TCP链接
Socket client = new Socket("127.0.0.1", 20006);
client.setSoTimeout(10000);
//获取键盘输入
BufferedReader input = new BufferedReader(new InputStreamReader(System.in));
//获取Socket的输出流,用来发送数据到服务端
PrintStream out = new PrintStream(client.getOutputStream());
//获取Socket的输入流,用来接收从服务端发送过来的数据
BufferedReader buf = new BufferedReader(new InputStreamReader(client.getInputStream()));
boolean flag = true;
while(flag){
System.out.print("输入信息:");
String str = input.readLine();
//发送数据到服务端
out.println(str);
if("bye".equals(str)){
flag = false;
}else{
try{
//从服务器端接收数据有个时间限制(系统自设,也能够本身设置),超过了这个时间,便会抛出该异常
String echo = buf.readLine();
System.out.println(echo);
}catch(SocketTimeoutException e){
System.out.println("Time out, No response");
}
}
}
input.close();
if(client != null){
//若是构造函数创建起了链接,则关闭套接字,若是没有创建起链接,天然不用关闭
client.close();//只关闭socket,其关联的输入输出流也会被关闭
}
}
}
服务端须要用到多线程,这里单独写了一个多线程类,代码以下:
package zyb.org.server;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.PrintStream;
import java.net.Socket;
/**
* 该类为多线程类,用于服务端
*/
public class ServerThread implements Runnable {
private Socket client = null;
public ServerThread(Socket client){
this.client = client;
}
@Override
public void run() {
try{
//获取Socket的输出流,用来向客户端发送数据
PrintStream out = new PrintStream(client.getOutputStream());
//获取Socket的输入流,用来接收从客户端发送过来的数据
BufferedReader buf = new BufferedReader(new InputStreamReader(client.getInputStream()));
boolean flag =true;
while(flag){
//接收从客户端发送过来的数据
String str = buf.readLine();
if(str == null || "".equals(str)){
flag = false;
}else{
if("bye".equals(str)){
flag = false;
}else{
//将接收到的字符串前面加上echo,发送到对应的客户端
out.println("echo:" + str);
}
}
}
out.close();
client.close();
}catch(Exception e){
e.printStackTrace();
}
}
}
服务端处理TCP链接请求的代码以下:
package zyb.org.server;
import java.net.ServerSocket;
import java.net.Socket;
public class Server1 {
public static void main(String[] args) throws Exception{
//服务端在20006端口监听客户端请求的TCP链接
ServerSocket server = new ServerSocket(20006);
Socket client = null;
boolean f = true;
while(f){
//等待客户端的链接,若是没有获取链接
client = server.accept();
System.out.println("与客户端链接成功!");
//为每一个客户端链接开启一个线程
new Thread(new ServerThread(client)).start();
}
server.close();
}
}
执行结果截图以下: