socket网络通信

本文详细探讨了Java Socket中的关闭方法,包括Socket.close()、shutdownInput()和shutdownOutput()。通过示例代码展示了如何避免服务端和客户端在接收数据时的阻塞问题,以及TCP关闭连接的机制。重点讲解了如何正确使用shutdown方法来通知对方数据发送完毕,同时保持接收数据的能力。
摘要由CSDN通过智能技术生成

http://blog.csdn.net/dabing69221/article/details/17351881

前言:

  在Java Socket中提供了2中关闭方法,其中包括:close() 、shutdownOutput()/shutdownInput() ,前一段时间做项目中用到了Socket,但是没有好好总结。

  现在,我把我遇到的问题分享一下:

  

  在网络协议中,通常明确指定了由谁来发起 “关闭”连接。

  例如:在HTTP协议中,客户端先向服务端发送一个请求,然后服务器响应请求的信息。由于客户端不知道服务端响应信息的大小,因此服务器必须通过关闭套

  接字来指示响应信息结束 —— 这个就是由服务器端发起的关闭连接!


一. Socket.close()方法

   这个方法我们估计都不会陌生,一般使用完Socket后,都会调用Socket.close()  来释放相关的资源。当我们调用Socket.close()方法将同时终止两个方向(输入

   和输出)的数据流。一旦一个终端(客户端或服务端)关闭了套接字,它将无法再发送或接收数据。这就意味着close( )方法只能在调用者完成了通信之后用来

   给另一端发送信号。只要服务端收到了客户端的关闭信号,就立即关闭连接。

  

   1. 考虑一个场景:

   假设你有一个服务器,接收到消息后,在发回给客户端,这种情况下应该是有哪一端来关闭连接呢?

   由于从客户端发送的消息长度是任意的,客户端需要关闭连接以通知服务器消息发送完毕,那么客户端应该什么时候调用close( )方法呢,如果客户端在发送完最后

   一个字节后立即调用close( )方法,它将无法接收服务器发送的消息 (因为: 关闭此套接字也将会关闭该套接字的 InputStream 和 OutputStream);如果客户端在接收

   完所有的数据后在调用close ( )方法,那么服务端 将不知道到底有多少数据要接收(因为:客户端的输出流一直没有关闭,所以,服务端的输入流也就无法到达流末尾返回-1),

   所以,服务器在接收数据时就一直阻塞,而这个时候客户端又要等待接收服务端发送来的消息(即使服务端发送了消息,但因为服务端的输出流一直没有关闭,所以,

   服务端的输入流也就无法到达流末尾返回-1,所以客户端在接收数据时就一直阻塞) —— 以上问题就造成:”服务端和客户端都在接收数据时阻塞"

   

   情况一 :客户端在发送完最后一个字节后立即调用close( )方法

  

[java]  view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. //服务端  
  2. public class Server  
  3. {  
  4.     public static void main(String[] args) throws IOException  
  5.     {  
  6.         System.out.println("------------------------------------");  
  7.         System.out.println("Server start......");  
  8.         System.out.println("------------------------------------");  
  9.   
  10.         ServerSocket server = new ServerSocket(8888);  
  11.   
  12.         while (true)  
  13.         {  
  14.             Socket client = server.accept();  
  15.             OutputStream out = client.getOutputStream();  
  16.             InputStream in = client.getInputStream();  
  17.   
  18.             ByteArrayOutputStream byteArrayOut = new ByteArrayOutputStream();  
  19.             byte[] temp = new byte[1024];  
  20.             int realLen = 0;  
  21.             while ((realLen = in.read(temp)) != -1)  
  22.             {  
  23.                 byteArrayOut.write(temp, 0, realLen);  
  24.             }  
  25.   
  26.             byte[] recv = byteArrayOut.toByteArray();  
  27.   
  28.             // 将接收的消息,发回给客户端  
  29.             out.write(recv);  
  30.             out.flush();  
  31.         }  
  32.     }  
  33. }  
  34.   
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值