相关java IO专题
前言
关于这个问题,网上说法很多,有的说四次有的说五次,因此写这篇文章仔细梳理一下
示例代码
File file = new File("/path");
FileInputStream in = new FileInputStream(file);
//假设一次可以读取完
byte[] buf = new byte[1024];
//文件数据读取到buf数组
in.read(buf);
//开启socket
Socket socket = new Socket("localhost", 9099);
OutputStream outputStream = socket.getOutputStream();
//数据写出去
outputStream.write(buf);
复制代码
字面意思,就是将file读取到buf,再把buf通过socket发送出去。我们一步一步来分析其中的原理
首先是 fileInputStream.read(buf)
public int read(byte b[]) throws IOException{
return readBytes(b, 0, b.length);
}
private native int readBytes(byte b[], int off, int len) throws IOException;
复制代码
fileInputStream.read方法实际调用了native的readBytes(byte b[], int off, int len),那么再来看看native的具体实现:
// 下列内容文件路径: openjdk-jdk8u-jdk8u/jdk/src/share/native/java/io/io_util.c
jint readBytes(JNIEnv *env, jobject this, jbyteArray bytes, jint off, jint len, jfieldID fid){
jint nread;
//事先定义的堆栈内存
char stackBuf[BUF_SIZE];
char *buf = NULL;
FD fd;
... 忽略不重要代码
if (len == 0) {
return 0;
} else if (len > BUF_SIZE) {
//如果希望读取的长度大于BUF_SIZE,则开辟一个新的内存,这个内存是native堆内存,属于用户空间的堆外内存
buf = malloc(len);
if (buf == NULL) {
JNU_ThrowOutOfMemoryError(env, NULL);
return 0;</