Socket、read函数、http中headers说明
Socket
1. Socket中几个函数
- setSoTimeout(int time)。设置read函数的阻塞时长,超过时长抛出
SocketTimeoutException
。 - shutdownOutput() / shutdownInput()。
shutdownOutput()
函数是关闭输出流,例如在客户端调用,此时代表客户端输出流关闭,不能再向服务端发送数据。但是客户端依然可以接受服务端的数据,即read
函数依然可以正常接受数据。同时,Socket连接没有关闭。 - close()。会直接关闭Socket连接。此时输入流、输出流都关闭。
- flush()函数。flush函数可以将数据发送到另一端,但是不代表文件的结尾,也不会使read函数返回-1,不会换行。
2. Socket服务端监听客户端断开
服务端再监听到客户端关闭输出流、或者Socket连接后,应该同样停止阻塞等待,关闭Socket连接。
- 客户端调用
shutdownOutput()
和close()
函数后,服务端的read(byte[])
函数返回-1 . 因为客户端再也不会向服务端发送数据了,因此可以通过read(byte[])
函数返回-1监听客户端的断开。此时应该关闭Socket连接,关闭输入输出流。
try {
server = new ServerSocket(8080);
socket = server.accept();
System.out.println("接收到客户端连接");
input = socket.getInputStream();
output = socket.getOutputStream();
byte[] bytes = new byte[1024];
int len = 0;
System.out.println("等待客户端发送来的数据");
//只有客户端通过调用shutdownOutput关闭输出流或者调用close关闭Socket连接,
//read才可能返回-1,否则一直堵塞
while ((len = input.read(bytes)) != -1) {
String str = new String(bytes, 0, len);
System.out.println(str);
}
System.out.println("接收read程序执行完成,跳出while循环");
output.write("服务端数据".getBytes());
} catch (IOException e) {
e.printStackTrace();
}
read函数说明
这里的read函数是指inputStream和outputStream中的读取函数read(byte[])
;
1. 阻塞等待
This method blocks until input data is available, end of file is detected, or an exception is thrown.
- read函数是一个会发生阻塞等待的方法,默认情况下read函数会一直阻塞。
- Socket中的
setSoTimeout(int time)
设置阻塞等待的时长,在超过该时长还未接收到数据流,则抛出SocketTimeoutException
异常,执行catch中的语句,此时read函数没有返回值。
2. 返回值
- 正常情况返回数字大于0。在正常读取输入流后,会返回读取到的字节数,这是一个大于0的数字。
- 返回数字为-1的情况。如果读取到输入流的结尾,会返回-1。具体来讲那些情况会返回-1呢?
- 发送端调用
shutdownOutput()
函数,接收端read函数会检测到输入流到文件结尾,返回-1 - 发送端的outputStream或者inputStream调用
close()
函数,接收端read函数返回 -1. 注意 : 如果发送端程序执行完毕,会默认调用close()
函数关闭Socket连接。
- 发送端调用
- 返回0.这种情况在read源码中有注释,不过暂时没遇到,似乎用的不多。
3. while循环接收
- 对于程序中情况只有客户端调用
shutdownOutput()
和close()
函数才有可能跳出while循环。 - 用户也可以自己设定标志跳出循环。例如传输字符可以设定特殊字符跳出循环,例如通过函数
indexof(string)
进行检查。参考链接
boolean flag = true;
while (flag) {
len = input.read(bytes);
//只有客户端通过调用shutdownOutput关闭输出流或者调用close关闭Socket连接,
// read才可能返回-1,代表可以关闭流了
if (len == -1) {
flag = false;
continue;
}
if (len != -1) {
String string = new String(bytes);
System.out.println(string);
}
}
- 如果在调用read(byte)函数的之前,byte不为空,比如已经有一些数据了,读入数据也是是从第0位开始
- 如果数据接收线程在sleep,比如sleep了8秒,在这8秒之内,发送端发送了两次数据,两次数据内容为A和B,这时候A和B会同时在网络中等待接收,因此在接收线程停止sleep,就会一次性接收到A和B,假设A和B是字符串,那么A和B就会拼接在一起,是的后面对字符串的处理出现错误.