客户端:
public class TCPClient {
public static void main(String[] args) throws IOException {
FileInputStream fis=new FileInputStream("E:\\Trump.jpg");
Socket socket=new Socket("127.0.0.1",8888);
OutputStream os=socket.getOutputStream();
byte[] bytes=new byte[1024];
int len;
while((len=fis.read(bytes))!=-1){
//将读取到的数据传输到客户端与服务器端之间的IO流
os.write(bytes,0,len);
}
//读服务器回写的数据
InputStream is=socket.getInputStream();
len=is.read(bytes);
System.out.println(new String(bytes,0,len));
socket.close();
fis.close();
}
}
服务器端:
public class TCPServer {
public static void main(String[] args) throws IOException {
ServerSocket server=new ServerSocket(8888);
Socket socket=server.accept();
InputStream is=socket.getInputStream();
FileOutputStream fos=new FileOutputStream("E:\\Test\\Trump.jpg");
int len;
byte[] bytes=new byte[1024];
while ((len=is.read(bytes))!=-1) {
fos.write(bytes,0,len);
}
OutputStream os=socket.getOutputStream();
os.write("文件上传完毕!".getBytes());
fos.close();
socket.close();
}
}
这段代码执行以后会发现server类 read()方法发生了阻塞,经过查找资料发现read()是一个阻塞函数(阻塞函数就是当这个函数不执行完,函数所在线程就一直停止在这里不动。)。
在输入数据可用、检测到流末尾或者抛出异常前,read方法一直阻塞。如果客户端没有声明断开outputStream那么它就会认为客户端仍旧可能发送数据,像read()这种阻塞读取函数还有BufferedReader类种的 readLine()、DataInputStream种的readUTF()等。
解决方案:
Socket任意一端在调用完write()方法时调用shutdownOutput()方法关闭输出流,这样服务器端的inputStream上的read操作就会返回-1, 这里我们要注意下不能调用socket.getInputStream().close()。因为它会导致socket直接被关闭。 当然如果不需要继续在socket上进行读操作,也可以直接关闭socket。但是这个方法不能用于通信双方需要多次交互的情况。