关于socket.shutdownOutput()的理解

文章讨论了在客户端上传图片到服务端时,如果没有正确关闭输出流可能导致服务端阻塞的问题。通过使用`socket.shutdownOutput()`,解决了数据传输结束后服务端等待客户端响应的问题,确保了双方通信的顺畅性。
摘要由CSDN通过智能技术生成

前言:再写一个关于客户端上传图片到服务端的案例时遇到的问题

当客户端发完数据,如果不用socket.shutdownOutput(),或者不使用人为标记,去结束循序

那么服务端可能还会是默认等待接收客户端数据的状态,会发生阻塞状态。

案例:

  1. 编写一个服务器端和客户端

  2. 服务器端在10087端口监听

  3. 客户端连接到服务器端,发送要上传的图片 (字节)

  4. 服务器端接收到客户端发送的数据,上传到本地文件 ,然后向客户端发送上传完毕,最后退出

  5. 客户端接收到服务端发送的信息,输出信息 ,然后退出

代码一:不使用socket.shutdownOutput()的情况

服务端:

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

public class UpLoadServer {
    public static void main(String[] args) throws IOException {
        ServerSocket ss = new ServerSocket(10087);
        Socket s = ss.accept();
        BufferedInputStream bis = new BufferedInputStream(s.getInputStream());
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("src\\upload\\2.jpg"));
        byte[] bys = new byte[1024];
        int len = 0;
        while ((len = bis.read(bys)) != -1){
            bos.write(bys,0,len);
        }
        System.out.println(len);
        //当Client停止发送时,Server还在接收Client的状态
        BufferedOutputStream bosServer = new BufferedOutputStream(s.getOutputStream());
        bosServer.write("上传完毕!".getBytes());
        bosServer.close();
        bos.close();
        bis.close();
        s.close();
    }
}

客户端:

import java.io.*;
import java.net.InetAddress;
import java.net.Socket;

public class UpLoadClient {
    public static void main(String[] args) throws IOException {
        Socket s = new Socket(InetAddress.getLocalHost(),10087);
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream("src\\upload\\1.jpg"));
        BufferedOutputStream bos = new BufferedOutputStream(s.getOutputStream());
        byte[] bys = new byte[1024];
        int len = 0;
        while ((len = bis.read(bys)) != -1){
            bos.write(bys,0,len);
        }
        BufferedInputStream bisClient = new BufferedInputStream(s.getInputStream());//阻塞Server发送
        byte[] bytes = new byte[1024];
        int read = bisClient.read(bytes);
        System.out.println(new String(bytes,0,read));
        bis.close();
        s.close();
    }
}

运行结果:图片上传了,但是

客户端和服务端处于阻塞状态

问题分析

由于结束循环的条件时 len==-1 时,这里我们把len打印出来,判断具体是那一步阻塞了

服务端

        int len = 0;
        while ((len = bis.read(bys)) != -1){
            bos.write(bys,0,len);
            System.out.println(len);
        }
        System.out.println(len);

客户端:客户端传输完文件,len一定会等于1

        int len = 0;
        while ((len = bis.read(bys)) != -1){
            bos.write(bys,0,len);
        }
        System.out.println(len);

运行结果:

这里可以发现:服务端并没有结束循序,客户端结束时,len没有等于-1

因为客户端要接收服务端发来的信息,而服务端还处于接收客户端消息的状态,

两者在互相等待

代码二:使用socket.shutdownOutput()的情况,并继续打印len

服务端:

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

public class UpLoadServer {
    public static void main(String[] args) throws IOException {
        ServerSocket ss = new ServerSocket(10087);
        Socket s = ss.accept();
        BufferedInputStream bis = new BufferedInputStream(s.getInputStream());
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("src\\upload\\2.jpg"));
        byte[] bys = new byte[1024];
        int len = 0;
        while ((len = bis.read(bys)) != -1){
            bos.write(bys,0,len);
            System.out.println(len);
        }
        System.out.println(len);
        //当Client停止发送时,Server还在接收Client的状态
        BufferedOutputStream bosServer = new BufferedOutputStream(s.getOutputStream());
        bosServer.write("上传完毕!".getBytes());
        bosServer.close();
        bos.close();
        bis.close();
        s.close();
    }
}

客户端:

import java.io.*;
import java.net.InetAddress;
import java.net.Socket;

public class UpLoadClient {
    public static void main(String[] args) throws IOException {
        Socket s = new Socket(InetAddress.getLocalHost(),10087);
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream("src\\upload\\1.jpg"));
        BufferedOutputStream bos = new BufferedOutputStream(s.getOutputStream());
        byte[] bys = new byte[1024];
        int len = 0;
        while ((len = bis.read(bys)) != -1){
            bos.write(bys,0,len);
        }
        s.shutdownOutput();// 返回给Server一个-1
        System.out.println(len);
        BufferedInputStream bisClient = new BufferedInputStream(s.getInputStream());//阻塞Server发送
        byte[] bytes = new byte[1024];
        int read = bisClient.read(bytes);
        System.out.println(new String(bytes,0,read));
        bis.close();
        s.close();
    }
}

运行结果:

从这里可以发现,使用socket.shutdownOutput()让结束了服务端一直处于循环的状态,结束了两者一直在阻塞的状态,其中可以看出关键就是这个-1,使用socket.shutdownOutput()后,服务端接收到了一个数据-1

注意点

并且这与socket.close()同理,如果客户端不接收服务端发来的信息,而是循环完后直接socket.close(),服务端也会结束并且最后打印出来的len值为-1

总结

 调用Socket.shutdownOutput()后,禁用此套接字的输出流,对于 TCP 套接字,任何以前写入的数据都将被发送,并且后跟 TCP 的正常连接终止序列(即-1),之后,从另一端TCP套接字的输入流中读取数据时,如果到达输入流末尾而不再有数据可用,则返回 -1。

简单理解就是告诉服务端,我没有数据要传了

  • 4
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: b'socket.shutdownoutput()'是一个方法,用于关闭套接字的输出流。这个方法会阻止所有的输出数据的发送,并发送一个EOF(End of File)信号,告诉接收方数据已经发送完毕。此时,只能读取数据,不能发送数据。 ### 回答2: socket.shutdownoutput()是一个方法,用于关闭Socket的输出流。在网络编程中,Socket是实现网络通信的一种机制,通过Socket可以建立客户端和服务器之间的连接,并进行数据的发送和接收。 当调用socket.shutdownoutput()方法时,Socket的输出流将会被关闭。这意味着,客户端将无法再向服务器发送数据,但是仍然可以接收服务器发送的数据。此时,服务器可以继续向客户端发送数据,直到所有数据发送完毕。 关闭Socket的输出流通常在以下情况下使用: 1. 客户端向服务器发送完所有的数据后,可以调用socket.shutdownoutput()关闭输出流,以通知服务器数据发送完毕,服务器也可以根据此信号判断客户端是否结束通信。 2. 服务器在处理完客户端的请求后,可以调用socket.shutdownoutput()关闭输出流,以通知客户端服务器已经完成了对请求的处理,为了节省资源,可以关闭输出流。 需要注意的是,socket.shutdownoutput()只会关闭Socket的输出流,不会关闭Socket的输入流。如果希望关闭Socket的输入流,可以调用socket.shutdowninput()方法。同时,关闭Socket的输出流或输入流之后,Socket本身并不会被关闭,可以继续使用Socket进行其他操作。 总之,socket.shutdownoutput()用于关闭Socket的输出流,用于通知对方数据发送完毕或处理完毕。 ### 回答3: socket.shutdownoutput()是一个用于关闭socket输出流的方法。当我们调用这个方法时,socket连接的一方将无法再向对方发送数据,但仍然可以接收对方发送的数据。这个方法在网络编程中非常有用,特别是在实现半双工通信时。 在一些情况下,我们可能需要在数据传输完成后关闭输出流,比如在客户端发送完请求后希望关闭输出流,以便服务器端知道数据传输完成并开始处理请求。此时我们可以使用socket.shutdownoutput()方法来实现。 此外,当我们希望在通信过程中关闭一个方向的数据流时(单向通信),也可以使用这个方法。比如在客户端向服务器端发送请求后,关闭客户端的输出流,从而实现数据的单向传输。这对于一些需要服务器端向客户端发送实时数据的应用非常有用。 需要注意的是,一旦调用了socket.shutdownoutput()方法关闭了输出流,就无法再次启用输出流。如果需要发送更多数据,需要重新建立一个新的socket连接。 总之,socket.shutdownoutput()方法是一个用于关闭socket输出流的方法,它在网络编程中非常有用,可以用于实现数据传输完成后关闭输出流,或者实现单向数据传输等功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值