Java socket中关闭IO流后,发生什么事?(以关闭输出流为例)
声明:该博文以socket中,关闭输出流为例进行说明。
为了方便讲解,我们把DataOutputstream dout = new DataOutputStream(new BufferedOutputStream(mySocket.getOutputStream()));中的dout做为Socket输出流的代言。同样的,din是输入流的代言。
可以造成dout被关闭的操作有:
1、调用dout.close();或din.close();因为使用这种流关闭,会造成socket被关闭,所以输入输出流都将不可再用。
2、调用socket.close();
3、调用socket.shutdownOutputStream();单方面关闭dout,此时din还可正常使用。
以下,我将对socket中关闭输出流进行3个测试
输出流关闭测试一:socket关闭吗?
输出流关闭测试二:该流是否可以重新开启?
输出流关闭测试三:输出缓冲区里的数据是丢弃,还是发送?
测试结果如下:
测试一:dout.close();会造成socket被关闭,但socket.shutdownOutputStream()不会。
测试二:不可以,会抛出异常!
测试三:丢弃
客户端程序:
package com.test2;
import java.io.*;
import java.net.*;
/**
* @ClassName: SocketTest
* @Description: 测试Socket中,流关闭后,socket是否关闭?是否可重开流?输出缓存区的数据是发送出去,还是丢弃?
* @author 慢跑学Android
* @date 2011-11-12 上午11:15:21
*
*/
public class SocketTest {
Socket mySocket;
DataOutputStream dout;
public static void main(String[] args){
new SocketTest();
} public SocketTest(){
// 输出流关闭的测试一:socket关闭吗?
test1();
// 输出流关闭测试二:该流是否可以重新开启?
test2();
// 输出流关闭测试三:输出缓冲区里的数据是丢弃,还是发送?
test3();
} private void test1() {
// 输出流关闭的测试一:socket关闭吗?
System.out.println("\n****2种方式关闭输出流,Socket是否关闭?***\n");
try {
mySocket = new Socket("27.154.122.233",9999);
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} try {
dout = new DataOutputStream(new BufferedOutputStream(mySocket.getOutputStream()));
//下面这一句主要是用来证明socket确实处于开启状态
System.out.println("输出流刚打开,Socket是否关闭?" + mySocket.isClosed());
mySocket.shutdownOutput();
System.out.println("使用shutdownOutput关闭输出流,Socket是否关闭?" + mySocket.isClosed());
dout.close();
System.out.println("使用close关闭输出流,Socket是否关闭?" + mySocket.isClosed());
} catch (IOException e) {
e.printStackTrace();
}
} private void test2() {
// 输出流关闭测试二:使用shutdownOutputStream后,输出流是否可以重新开启?
System.out.println("\n****使用shutdownOutputStream后,输出流是否可以重新开启?***\n");
try {
mySocket = new Socket("27.154.122.233",9999);
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} try {
dout = new DataOutputStream(new BufferedOutputStream(mySocket.getOutputStream()));
mySocket.shutdownOutput();
// 重开输出流
dout = new DataOutputStream(mySocket.getOutputStream());
dout.writeUTF("是否允许我重开?");
// 清空输出缓存,确保当dout通道没问题时,消息可以到达服务器
dout.flush();
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
mySocket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
} private void test3(){
// 输出流关闭测试三:输出缓冲区里的数据是丢弃,还是发送?
System.out.println("\n***输出缓冲区里的数据是丢弃,还是发送?****\n");
try {
mySocket = new Socket("27.154.122.233",9999);
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} try {
dout = new DataOutputStream(new BufferedOutputStream(mySocket.getOutputStream()));
dout.writeUTF("shutdownOutput后,数据发得得出去吗?");
mySocket.shutdownOutput();
} catch (IOException e) {
e.printStackTrace();
}
}
}
服务器端程序:
/**
* @Title: ServerSocketTest.java
* @Package com.test1
* @Description: TODO(该文件为”Socket中,流关闭后,发生什么事“的Sever测试端)
* @author 慢跑学Android
* @date 2011-11-12 上午11:31:05
* @version V1.0
*/
package com.test1; import java.io.*;
import java.net.*; public class ServerSocketTest extends Thread{
private ServerSocket myServerSocket;
private final int PORT = 9999;
public static void main(String[] args){
ServerSocketTest sst = new ServerSocketTest();
sst.start();
} public ServerSocketTest(){
// 初始化一个ServeSocket端
try {
myServerSocket = new ServerSocket(PORT);
} catch (IOException e) {
e.printStackTrace();
}
} public void run(){
while(true){
System.out.println("我是服务器,我在9999端口监听....");
try {
Socket socket = myServerSocket.accept();
DataInputStream din = new DataInputStream(new BufferedInputStream(socket.getInputStream()));
String msgIn = din.readUTF();
System.out.println(msgIn.trim());
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
说明一点:
在test3()中,因为dout = new DataOutputStream(newBufferedOutputStream(mySocket.getOutputStream()));使用了Buffered,所以在dout.writeUTF()方法后,如果没有使用dout.flush();数据会存在输出缓存中,不会发送出去的。
如果我们队dout的声明是,dout = new DataOutputStream(mySocket.getOutputStream());那么,数据会立即发送出去。(除非,对方没有调用read()来读取数据,且数据量极大,超过了对方的输入缓存。不过,此时dout.writeUTF();这里会堵塞。)
以下是程序运行后,客户端与服务器各自的控制台输出情况:
----------------------------------客户端--------------------------
java.net.SocketException: Socket output is shutdown
at java.net.Socket.getOutputStream(Unknown Source)
at com.test2.SocketTest.test2(SocketTest.java:66)
at com.test2.SocketTest.<init>(SocketTest.java:22)
at com.test2.SocketTest.main(SocketTest.java:15)
****2种方式关闭输出流,Socket是否关闭?***
输出流刚打开,Socket是否关闭?false
使用shutdownOutput关闭输出流,Socket是否关闭?false
使用close关闭输出流,Socket是否关闭?true
****使用shutdownOutputStream后,输出流是否可以重新开启?***
***输出缓冲区里的数据是丢弃,还是发送?****
---------------------------------服务器------------------------------
我是服务器,我在9999端口监听....
我是服务器,我在9999端口监听....
java.io.EOFException
at java.io.DataInputStream.readUnsignedShort(Unknown Source)
at java.io.DataInputStream.readUTF(Unknown Source)
at java.io.DataInputStream.readUTF(Unknown Source)
at com.test1.ServerSocketTest.run(ServerSocketTest.java:37)
java.io.EOFException
at java.io.DataInputStream.readUnsignedShort(Unknown Source)
at java.io.DataInputStream.readUTF(Unknown Source)
at java.io.DataInputStream.readUTF(Unknown Source)
at com.test1.ServerSocketTest.run(ServerSocketTest.java:37)
java.io.EOFException
at java.io.DataInputStream.readUnsignedShort(Unknown Source)
我是服务器,我在9999端口监听....
at java.io.DataInputStream.readUTF(Unknown Source)
at java.io.DataInputStream.readUTF(Unknown Source)
at com.test1.ServerSocketTest.run(ServerSocketTest.java:37)
Reference:
Java socket中关闭IO流后,发生什么事?(以关闭输出流为例)
Java socket中关闭IO流后,发生什么事?(以关闭输出流为例)的更多相关文章
- Android—Socket中关闭IO流后导致Socket关闭不能再收发数据的解决办法
以Socket发送数据为例: 发送数据时候要声明:DataOutputStream os = new DataOutputStream(socket.getOutputStream()); 最近开发遇 ...
- Java中的IO流(二)
上一篇<Java中的IO流(一)>把学习IO流的字符流作了一下记录,本篇把字节流记录一下. 一,Java中的字节流 Java中的字节流的操作方式与字符流的操作方式大致相同,连方法名都是类似 ...
- Java中的IO流之输出流|乐字节
大家好,乐字节小乐又来了.上一篇给大家带来的是:Java中的IO流之输入流|乐字节,本文将继续讲述IO流之输出流. 一.输出流 1.抽象类:OutputStream 和 Writer Output ...
- Java中的IO流(一)
一,IO流的分类 A,按照操作的对象不同可分为字节流与字符流 B,按照流的方向不同可分为输入(指从外部存储设备拿文件到内存)流也叫读与输出(指从内存把文件写入到外部存储设备)流也叫写 注:字节流可以操 ...
- Java中的IO流,Input和Output的用法,字节流和字符流的区别
Java中的IO流:就是内存与设备之间的输入和输出操作就成为IO操作,也就是IO流.内存中的数据持久化到设备上-------->输出(Output).把 硬盘上的数据读取到内存中,这种操作 成为 ...
- Java中的IO流(五)
上一篇<Java中的IO流(四)>记录了一下Properties类,此类不属于IO流,它属于集合框架.接下来说一下IO流中的其它流 一,打印流PrintStream PrintStream ...
- Java中的IO流 - 入门篇
前言 大家好啊,我是汤圆,今天给大家带来的是<Java中的IO流-入门篇>,希望对大家有帮助,谢谢 由于Java的IO类有很多,这就导致我刚开始学的时候,感觉很乱,每次用到都是上网搜,结果 ...
- Java中的IO流总结
Java中的IO流总结 1. 流的继承关系,以及字节流和字符流. 2. 节点流FileOutputStream和FileInputStream和处理流BufferedInputStream和Buffe ...
- Java中的IO流大体介绍
由于Java中的IO流是在是知识点繁多,所以我大约花了1周的时间将其整理起来.但是整理起来后并不是将完事了,我还是要分字节流和字符流来讲述.然后字节流和字符流中还有是否带有缓冲流. 讲述完IO流后我将 ...
随机推荐
- ThinkPHP的cookide保存二维数组的方法
ThinkPHP中的cookie是不支持二维数组的. 如果要保存二维数组.只能特殊处理 $data[263] = array('gid'=>263,'num'=>1); $data[266 ...
- Rails--content_for和yield
--使用: (1)layout中使用<%=yield(:js)%> (2)html中使用<% content_for :js do %>[XXX]<% end %>
- a compromise between lock overhead and data safety
High Performance My SQL THIRD EDITION A locking strategy is a compromise between lock overhead and ...
- Andrew Ng机器学习公开课笔记–Principal Components Analysis (PCA)
网易公开课,第14, 15课 notes,10 之前谈到的factor analysis,用EM算法找到潜在的因子变量,以达到降维的目的 这里介绍的是另外一种降维的方法,Principal Compo ...
- [Virtualization][SDN] VXLAN到底是什么 [转]
写在转发之前: 几个月以前,在北大机房和燕园大厦直接拉了一根光钎.两端彼此为校园内公网IP.为了方便连接彼此机房,我做个一个VPN server在燕园的边界,北大机房使用client拨回.两个物理机房 ...
- Bluetooth RFCOMM介绍
目录 1. 介绍 2. 服务概述 2.1 RS-232控制信号 2.2 Null Modem Emulation 2.3 多串口仿真 3. 服务接口描述 4. RFCOMM帧类型 5. RFCOMM帧 ...
- 【Java 进阶篇】【第一课】String类
引用 String类包含在java.lang包中.这个包会在Java启动的时候自动import,所以可以当做一个内置类(built-in class).我们不需要显式的使用import引入String ...
- Bulk Insert & BCP执行效率对比(续)
上回由于磁盘空间(约70G)不足,导致Bulk Insert和BCP导入中途失败:今次统一一些操作,以得到Bulk insert与BCP分别执行效率: 1. 15435390笔数据,21.7G csv ...
- 通过自定义属性存储数据实现输入框获得焦点与失去焦点改变value值
http://gopro.ee.cagoe.com/index.html html: <div class="name"><input value=&qu ...
- Ubuntu deb包使用
deb是debian linus的安装格式,跟red hat的rpm非常相似,最基本的安装命令是:dpkg -i file.deb dpkg 是Debian Package的简写,是为Debian 专 ...