Java IO

 

 

Java IO - File的要点,应该是

1、跨平台问题的解决

2、文件的安全

3、文件的检索方法

一、代码小引入

代请看一个简单的小demo:(ps:开源项目java-core-learning地址https://github.com/JeffLi1993

import java.io.File;
import java.util.Arrays;
/*
 * Copyright [2015] [Jeff Lee]
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

/**
 * @author Jeff Lee
 * @since 2015-7-13 07:58:56
 * 列出目录并排序
 */
public class DirListT {
    public static void main(String[] args) {
        // 获取当前目录
        File path =  new File( "." ); // .表示当前目录
        // 文件路径名数组
        String list[] = path.list();
       
        // 对String文件名进行排序
        Arrays.sort(list,String.CASE_INSENSITIVE_ORDER);
       
        // 打印
        for (String dirItem : list)
            System.out.println(dirItem);
    }
}

在eclipse中,右键run一下,可以得到如下的结果:

image

如图,很容易注意到了,其目录下的名字排序按字母并打印了。

先回顾下API知识吧,

首先构造函数 public File(String pathname)

通过将给定路径名字符串转换为抽象路径名来创建一个新 File 实例。如果给定字符串是空字符串,那么结果是空抽象路径名。

参数:
pathname - 路径名字符串
抛出:
NullPointerException - 如果 pathname 参数为 null
二者,File实现了Comparator接口,以便对FileName进行排序

static Comparator<String>
     CASE_INSENSITIVE_ORDER
          一个对 String 对象进行排序的 Comparator,作用与 compareToIgnoreCase 相同。

三者, path.list()为什么会返回String[] filenams的数组呢?怎么不是List呢?

自问自答:这时候,我们应该去看看ArrayList的实现,ArrayList其实是动态的数组实现。动态,动态的弊端就是效率低。此时,返回一个固定的数组,而不是一个灵活的类容器,因为其目录元素是固定的。下面是ArrayList和数组Array的比较

绘图1

 

二、深入理解源码

File,File究竟是怎么构成的。顺着源码,知道了File有几个重要属性

clipboard

1、static private FileSystem fs

     FileSystem : 对本地文件系统的抽象

2、String path 文件路径名

3、内联枚举类

     PathStatus 地址是否合法 ENUM类 private static enum PathStatus { INVALID, CHECKED };

4、prefixLength 前缀长度

如下,给出File相关核心的UML图
file

 

其实操作的是 FileSystem : 对本地文件系统的抽象,真正操作的是 FileSytem派生类。通过源码Ctrl+T发现如下:Win下操作的是 Win32FileSystem 和 WinNTFileSystem类。看来真正通过jvm,native调用系统的File是他们。

clipboard[1]

Linux呢?因此,下了个Linux版本的JDK,解压,找到rt.jar。然后java/io目录中,找到了UnixFileSystem类。真相大白了!

clipboard[2]

 

所以可以小结File操作源码这样调用的:中间不同JDK,其实是不同的类调用本机native方法

绘图1

三、小demo再来一发

File 其实和我们在系统中看的的文件一样。就像我们右键,属性。可以看到很多File的信息。Java File也有。下面是一个文件的相关方法详情:

import java.io.File;
/*
 * Copyright [2015] [Jeff Lee]
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

/**
 * @author Jeff Lee
 * @since 2015-7-13 10:06:28
 * File方法详细使用
 */
public class FileMethodsT {
   
    private static void fileData(File f) {
        System.out.println(
            " 绝对路径:" + f.getAbsolutePath() +
            "\n 可读:" + f.canRead() +
            "\n 可写:" + f.canWrite() +
            "\n 文件名:" + f.getName() +
            "\n 上级目录:" + f.getParent() +
            "\n 相对地址:" + f.getPath() +
            "\n 长度:" + f.length() +
            "\n 最近修改时间:" + f.lastModified()
            );
        if (f.isFile())
            System.out.println( " 是一个文件" );
        else if (f.isDirectory())
            System.out.println( " 是一个目录" );
    }
   
    public static void main(String[] args) {
        // 获取src目录
        File file =  new File( "src" );
        // file详细操作
        fileData(file);
    }
}

在eclipse中,右键run一下,可以得到如下的结果:大家应该都明白了吧。

image

 

从上一篇 图解 Java IO : 一、File源码 并没有把所有File的东西讲完。这次讲讲FilenameFilter,关于过滤器文件《Think In Java》中写道:

更具体地说,这是一个策略模式的例子,因为list()实现了基本功能,而按着形式提供了这个策略,完善list()提供服务所需的算法。

java.io.FilenameFilter是文件名过滤器接口,即过滤出符合规则的文件名组。

一、FilenameFilter源码

image

从IO的UML可以看出,FilenameFilter接口独立,而且没有它的实现类。下面就看看它的源码:

public interface FilenameFilter {
    /**
     * 测试指定文件是否应该包含在某一文件列表中。
     *
     * @param   被找到的文件所在的目录。
     * @param   文件的名称
     */
    boolean accept(File dir, String name);
}

从JDK1.0就存在了,功能也很简单:就是为了过滤文件名。只要在accept()方法中传入相应的目录和文件名即可。

深度分析:接口要有真正的实现才能算行为模式中真正实现。所以这里使用的是策略模式,涉及到三个角色:

环境(Context)角色

抽象策略(Strategy)角色

具体策略(Context Strategy)角色

结构图如下:

filenameFilter

其中,FilenameFiler Interface 就是这里的抽象策略角色。其实也可以用抽象类实现。

但是,装饰器模式也有缺点:在编写程序的时候,它提供了相当多的灵活性(容易混合和匹配属性),同时也增加了代码的复杂性

 

 

 

二、使用方法

image

如图 FilenameFiler使用如图所示。上代码吧:(small 广告是要的,代码都在 开源项目java-core-learning。地址https://github.com/JeffLi1993

package org.javacore.io;

import java.io.File;
import java.io.FilenameFilter;
/*
 * Copyright [2015] [Jeff Lee]
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

/**
 * @author Jeff Lee
 * @since 2015-7-20 13:31:41
 * 类名过滤器的使用
 */
public class FilenameFilterT {

    public static void main(String[] args) {
        // IO包路径
        String dir =  "src" + File.separator +
                "org" + File.separator +
                "javacore" + File.separator +
                "io" ;
        File file =  new File(dir);
        // 创建过滤器文件
        MyFilter filter =  new MyFilter( "y.java" );
        // 过滤
        String files[] = file.list(filter);
       
        // 打印
        for (String name : files) {
            System.err.println(name);
        }
    }
   
    /**
     *  内部类实现过滤器文件接口
     */
    static class MyFilter  implements FilenameFilter {
       
        private String type;
       
        public MyFilter (String type) {
            this .type = type;
        }

        @Override
        public boolean accept(File dir, String name) {
            return name.endsWith(type); // 以Type结尾
        }
       
    }
}

其中我们用内部类的实现,实现了FilenameFilter Interface。所以当我们File list调用接口方法时,传入MyFilter可以让文件名规则按我们想要的获得。

右键 Run 下,可以看到如图所示的输出:

image

 

补充:

String[] fs = f.list()

File[] fs = f.listFiles()

String []fs = f.list(FilenameFilter filter);;

File[]fs = f.listFiles(FilenameFilter filter);

image

 

 

一、InputStream

InputStream是一个抽象类,即表示所有字节输入流实现类的基类。它的作用就是抽象地表示所有从不同数据源产生输入的类,例如常见的FileInputStream、FilterInputStream等。那些数据源呢?比如:

1) 字节数组(不代表String类,但可以转换)

2) String对象

3) 文件

4) 一个其他种类的流组成的序列化 (在分布式系统中常见)

5) 管道(多线程环境中的数据源)

等等

二者,注意它是属于字节流部分,而不是字符流(java.io中Reader\Writer,下面会讲到)。

FilterInputStream是为各种InputStream实现类提供的“装饰器模式”的基类。因此,可以分为原始的字节流和“装饰”过的功能封装字节流。

 

二、细解InputStream源码的核心

源码如下:

/**
 * 所有字节输入流实现类的基类
 */
public abstract class SInputStream {

    // 缓存区字节数组最大值
    private static final int MAX_SKIP_BUFFER_SIZE =  2048 ;

    // 从输入流中读取数据的下一个字节,以int返回
    public abstract int read()  throws IOException;

    // 从输入流中读取数据的一定数量字节,并存储在缓存数组b
    public int read( byte b[])  throws IOException {
        return read(b,  0 , b.length);
    }

    // 从输入流中读取数据最多len个字节,并存储在缓存数组b
    public int read( byte b[],  int off,  int len)  throws IOException {
        if (b ==  null ) {
            throw new NullPointerException();
        }  else if (off <  0 || len <  0 || len > b.length - off) {
            throw new IndexOutOfBoundsException();
        }  else if (len ==  0 ) {
            return 0 ;
        }

        int c = read();
        if (c == - 1 ) {
            return - 1 ;
        }
        b[off] = ( byte )c;

        int i =  1 ;
        try {
            for (; i < len ; i++) {
                c = read();
                if (c == - 1 ) {
                    break ;
                }
                b[off + i] = ( byte )c;
            }
        }  catch (IOException ee) {
        }
        return i;
    }

    // 跳过输入流中数据的n个字节
    public long skip( long n)  throws IOException {

        long remaining = n;
        int nr;

        if (n <=  0 ) {
            return 0 ;
        }

        int size = ( int )Math.min(MAX_SKIP_BUFFER_SIZE, remaining);
        byte [] skipBuffer =  new byte [size];
        while (remaining >  0 ) {
            nr = read(skipBuffer,  0 , ( int )Math.min(size, remaining));
            if (nr <  0 ) {
                break ;
            }
            remaining -= nr;
        }

        return n - remaining;
    }

    // 返回下一个方法调用能不受阻塞地从此读取(或者跳过)的估计字节数
    public int available()  throws IOException {
        return 0 ;
    }

    // 关闭此输入流,并释放与其关联的所有资源

    public void close()  throws IOException {}

    // 在此输出流中标记当前位置
    public synchronized void mark( int readlimit) {}

    // 将此流重新定位到最后一次对此输入流调用 mark 方法时的位置。
    public synchronized void reset()  throws IOException {
        throw new IOException( "mark/reset not supported" );
    }

    // 测试此输入流是否支持 mark 和 reset 方法
    public boolean markSupported() {
        return false ;
    }

}

其中,InputStream下面三个read方法才是核心方法:

public abstract int read()

抽象方法,没有具体实现。因为子类必须实现此方法的一个实现。这就是输入流的关键方法。

二者,可见下面两个read()方法都调用了这个方法子类的实现来完成功能的。

 

public int read( byte b[])

该方法是表示从输入流中读取数据的一定数量字节,并存储在缓存字节数组b。其效果等同于调用了下面方法的实现:

read(b,  0 , b.length)

如果b的长度为 0,则不读取任何字节并返回 0;否则,尝试读取至少 1 字节。如果因为流位于文件末尾而没有可用的字节,则返回值 -1;否则,至少读取一个字节并将其存储在 b 中。

思考:这时候,怪不得很多时候, b != –1 或者 b != EOF

 

public int read( byte b[],  int off,  int len)


在输入数据可用、检测到流末尾或者抛出异常前,此方法一直阻塞。

该方法先进行校验,然后校验下个字节是否为空。如果校验通过后,
如下代码:

int i =  1 ;
try {
    for (; i < len ; i++) {
        c = read();
        if (c == - 1 ) {
            break ;
        }
        b[off + i] = ( byte )c;
    }
}  catch (IOException ee) {
}

将读取的第一个字节存储在元素 b[off] 中,下一个存储在 b[off+1] 中,依次类推。读取的字节数最多等于 len。设 k 为实际读取的字节数;这些字节将存储在 b[off] 到 b[off+k-1] 的元素中,不影响 b[off+k] 到 b[off+len-1] 的元素。

因为有上面两个read的实现,所以这里InputStream设计为抽象类。

 

 

一、前言

上一篇《Java IO 之 InputStream源码》,说了InputStream。JDK1.0中就有了这传统的IO字节流,也就是 InputStream 和 OutputStream。梳理下两者的核心:

InputStream中有几个 read() 方法和 OutputStream中有几个 write() 方法。它们是一一对应的,而核心的是read()和write()方法。它们都没实现,所有本质调用是各自实现类实现的该两个方法。

read() 和 write() ,对应着系统的Input和Output,即系统的输出输入。

 

二、OutputStream

也是一个抽象类,即表示所有字节输入流实现类的基类。它的作用就是抽象地表示所有要输出到的目标,例如常见的FileOutStream、FilterOutputStream等。它实现了java.io.Closeable和java.io.Flushable两个接口。其中空实现了flush方法,即拥有刷新缓存区字节数组作用。

那些输出目标呢?比如:

1) 字节数组(不代表String类,但可以转换)

2) 文件

3) 管道(多线程环境中的数据源)

等等

FilterOutputStream是为各种OutputStream实现类提供的“装饰器模式”的基类。将属性或者有用的接口与输出流连接起来。

 

三、细解OutputStream源码的核心

一样的,先看源码:

/**
 * 所有字节输出流实现类的基类
 */
public abstract class SOutputStream  implements Closeable, Flushable {

    // 将指定的字节写入输出流
    public abstract void write( int b)  throws IOException;

    // 将指定的byte数组的字节全部写入输出流
    public void write( byte b[])  throws IOException {
        write(b,  0 , b.length);
    }

    // 将指定的byte数组中从偏移量off开始的len个字节写入输出流
    public void write( byte b[],  int off,  int len)  throws IOException {
        if (b ==  null ) {
            throw new NullPointerException();
        }  else if ((off <  0 ) || (off > b.length) || (len <  0 ) ||
                   ((off + len) > b.length) || ((off + len) <  0 )) {
            throw new IndexOutOfBoundsException();
        }  else if (len ==  0 ) {
            return ;
        }
        for ( int i =  0 ; i < len ; i++) {
            write(b[off + i]);
        }
    }

    // 刷新输出流,并强制写出所有缓冲的输出字节
    public void flush()  throws IOException {
    }

    // 关闭输出流,并释放与该流有关的所有资源
    public void close()  throws IOException {
    }

}

其中三个核心的write()方法,对应着三个InputStream的read()方法:

1. abstract void write(int b) 抽象方法

public abstract void write( int b)  throws IOException;

对应着,InputStream的read()方法,此方法依旧是抽象方法。因为子类必须实现此方法的一个实现。这就是输入流的关键方法。

二者,下面两个write方法中调用了此核心方法。

可见,核心的是read()和write()方法在传统的IO是多么重要。

 

2.  void write(byte b[]) 方法

public void write( byte b[])  throws IOException

将指定的byte数组的字节全部写入输出流。该效果实际上是由下一个write方法实现的,只是调用的额时候指定了长度:

 

3.  void write(byte b[], int off, int len) 方法

public void write( byte b[],  int off,  int len)  throws IOException

将指定的byte数组中从偏移量off开始的len个字节写入输出流。代码详细流程解读如下:

    a.  如果 b 为 null,则抛出 NullPointerException

b.  如果 off 为负,或 len 为负,或者 off+len 大于数组 b 的长度,则抛出 IndexOutOfBoundsException

c.  将数组 b 中的某些字节按顺序写入输出流;元素 b[off] 是此操作写入的第一个字节,b[off+len-1] 是此操作写入的最后一个字节。

 

四、小结

重要的事情说三遍:

OutputStream 解读对照着 InputStream来看!注意 一个write对应一个read

OutputStream 解读对照着 InputStream来看!注意 一个write对应一个read

OutputStream 解读对照着 InputStream来看!注意 一个write对应一个read

 

 

一、引子

文件,作为常见的数据源。关于操作文件的字节流就是 --- FileInputStream & FileOutputStream。它们是Basic IO字节流中重要的实现类。

 

二、FileInputStream源码分析

FileInputStream源码如下:

/**
 * FileInputStream 从文件系统的文件中获取输入字节流。文件取决于主机系统。
 *  比如读取图片等的原始字节流。如果读取字符流,考虑使用 FiLeReader。
 */
public class SFileInputStream  extends InputStream
{
    /* 文件描述符类---此处用于打开文件的句柄 */
    private final FileDescriptor fd;

    /* 引用文件的路径 */
    private final String path;

    /* 文件通道,NIO部分 */
    private FileChannel channel =  null ;

    private final Object closeLock =  new Object();
    private volatile boolean closed =  false ;

    private static final ThreadLocal runningFinalize =
        new ThreadLocal<>();

    private static boolean isRunningFinalize() {
        Boolean val;
        if ((val = runningFinalize.get()) !=  null )
            return val.booleanValue();
        return false ;
    }

    /* 通过文件路径名来创建FileInputStream */
    public FileInputStream(String name)  throws FileNotFoundException {
        this (name !=  null ?  new File(name) :  null );
    }

    /* 通过文件来创建FileInputStream */
    public FileInputStream(File file)  throws FileNotFoundException {
        String name = (file !=  null ? file.getPath() :  null );
        SecurityManager security = System.getSecurityManager();
        if (security !=  null ) {
            security.checkRead(name);
        }
        if (name ==  null ) {
            throw new NullPointerException();
        }
        if (file.isInvalid()) {
            throw new FileNotFoundException( "Invalid file path" );
        }
        fd =  new FileDescriptor();
        fd.incrementAndGetUseCount();
        this .path = name;
        open(name);
    }

    /* 通过文件描述符类来创建FileInputStream */
    public FileInputStream(FileDescriptor fdObj) {
        SecurityManager security = System.getSecurityManager();
        if (fdObj ==  null ) {
            throw new NullPointerException();
        }
        if (security !=  null ) {
            security.checkRead(fdObj);
        }
        fd = fdObj;
        path =  null ;
        fd.incrementAndGetUseCount();
    }

    /* 打开文件,为了下一步读取文件内容。native方法 */
    private native void open(String name)  throws FileNotFoundException;

    /* 从此输入流中读取一个数据字节 */
    public int read()  throws IOException {
        Object traceContext = IoTrace.fileReadBegin(path);
        int b =  0 ;
        try {
            b = read0();
        }  finally {
            IoTrace.fileReadEnd(traceContext, b == - 1 ?  0 :  1 );
        }
        return b;
    }

    /* 从此输入流中读取一个数据字节。native方法 */
    private native int read0()  throws IOException;

    /* 从此输入流中读取多个字节到byte数组中。native方法 */
    private native int readBytes( byte b[],  int off,  int len)  throws IOException;

    /* 从此输入流中读取多个字节到byte数组中。 */
    public int read( byte b[])  throws IOException {
        Object traceContext = IoTrace.fileReadBegin(path);
        int bytesRead =  0 ;
        try {
            bytesRead = readBytes(b,  0 , b.length);
        }  finally {
            IoTrace.fileReadEnd(traceContext, bytesRead == - 1 ?  0 : bytesRead);
        }
        return bytesRead;
    }

    /* 从此输入流中读取最多len个字节到byte数组中。 */
    public int read( byte b[],  int off,  int len)  throws IOException {
        Object traceContext = IoTrace.fileReadBegin(path);
        int bytesRead =  0 ;
        try {
            bytesRead = readBytes(b, off, len);
        }  finally {
            IoTrace.fileReadEnd(traceContext, bytesRead == - 1 ?  0 : bytesRead);
        }
        return bytesRead;
    }

   
    public native long skip( long n)  throws IOException;

    /* 返回下一次对此输入流调用的方法可以不受阻塞地从此输入流读取(或跳过)的估计剩余字节数。 */
    public native int available()  throws IOException;

    /* 关闭此文件输入流并释放与此流有关的所有系统资源。 */
    public void close()  throws IOException {
        synchronized (closeLock) {
            if (closed) {
                return ;
            }
            closed =  true ;
        }
        if (channel !=  null ) {
           fd.decrementAndGetUseCount();
           channel.close();
        }

        int useCount = fd.decrementAndGetUseCount();

        if ((useCount <=  0 ) || !isRunningFinalize()) {
            close0();
        }
    }

    public final FileDescriptor getFD()  throws IOException {
        if (fd !=  null )  return fd;
        throw new IOException();
    }

    /* 获取此文件输入流的唯一FileChannel对象 */
    public FileChannel getChannel() {
        synchronized ( this ) {
            if (channel ==  null ) {
                channel = FileChannelImpl.open(fd, path,  true ,  false ,  this );
                fd.incrementAndGetUseCount();
            }
            return channel;
        }
    }

    private static native void initIDs();

    private native void close0()  throws IOException;

    static {
        initIDs();
    }

    protected void finalize()  throws IOException {
        if ((fd !=  null ) &&  (fd != FileDescriptor.in)) {
            runningFinalize.set(Boolean.TRUE);
            try {
                close();
            }  finally {
                runningFinalize.set(Boolean.FALSE);
            }
        }
    }
}

1. 三个核心方法

三个核心方法,也就是Override(重写)了抽象类InputStreamread方法。

int read() 方法,即

public int read()  throws IOException

代码实现中很简单,一个try中调用本地nativeread0()方法,直接从文件输入流中读取一个字节。IoTrace.fileReadEnd(),字面意思是防止文件没有关闭读的通道,导致读文件失败,一直开着读的通道,会造成内存泄露。

 

int read(byte b[]) 方法,即

public int read( byte b[])  throws IOException

代码实现也是比较简单的,也是一个try中调用本地nativereadBytes()方法,直接从文件输入流中读取最多b.length个字节到byte数组b中。

 

int read(byte b[], int off, int len) 方法,即

public int read( byte b[],  int off,  int len)  throws IOException

代码实现和 int read(byte b[])方法 一样,直接从文件输入流中读取最多len个字节到byte数组b中。

可是这里有个问答:

Q: 为什么 int read(byte b[]) 方法需要自己独立实现呢? 直接调用 int read(byte b[], int off, int len) 方法,即read(b , 0 , b.length),等价于read(b)?

A:待完善,希望路过大神回答。。。。向下兼容?? Finally??

 

2. 值得一提的native方法

上面核心方法中为什么实现简单,因为工作量都在native方法里面,即JVM里面实现了。native倒是不少一一列举吧:

native void open(String name) // 打开文件,为了下一步读取文件内容

native int read0() // 从文件输入流中读取一个字节

native int readBytes(byte b[], int off, int len) // 从文件输入流中读取,从off句柄开始的len个字节,并存储至b字节数组内。

native void close0() // 关闭该文件输入流及涉及的资源,比如说如果该文件输入流的FileChannel对被获取后,需要对FileChannel进行close。

 

其他还有值得一提的就是,在jdk1.4中,新增了NIO包,优化了一些IO处理的速度,所以在FileInputStream和FileOutputStream中新增了FileChannel getChannel()的方法。即获取与该文件输入流相关的 java.nio.channels.FileChannel对象。

 

三、FileOutputStream 源码分析

FileOutputStream 源码如下:

/**
 * 文件输入流是用于将数据写入文件或者文件描述符类
 *  比如写入图片等的原始字节流。如果写入字符流,考虑使用 FiLeWriter。
 */
public class SFileOutputStream  extends OutputStream
{
    /* 文件描述符类---此处用于打开文件的句柄 */
    private final FileDescriptor fd;

    /* 引用文件的路径 */
    private final String path;

    /* 如果为 true,则将字节写入文件末尾处,而不是写入文件开始处 */
    private final boolean append;

    /* 关联的FileChannel类,懒加载 */
    private FileChannel channel;

    private final Object closeLock =  new Object();
    private volatile boolean closed =  false ;
    private static final ThreadLocal runningFinalize =
        new ThreadLocal<>();

    private static boolean isRunningFinalize() {
        Boolean val;
        if ((val = runningFinalize.get()) !=  null )
            return val.booleanValue();
        return false ;
    }

    /* 通过文件名创建文件输入流 */
    public FileOutputStream(String name)  throws FileNotFoundException {
        this (name !=  null ?  new File(name) :  null ,  false );
    }

    /* 通过文件名创建文件输入流,并确定文件写入起始处模式 */
    public FileOutputStream(String name,  boolean append)
        throws FileNotFoundException
    {
        this (name !=  null ?  new File(name) :  null , append);
    }

    /* 通过文件创建文件输入流,默认写入文件的开始处 */
    public FileOutputStream(File file)  throws FileNotFoundException {
        this (file,  false );
    }

    /* 通过文件创建文件输入流,并确定文件写入起始处  */
    public FileOutputStream(File file,  boolean append)
        throws FileNotFoundException
    {
        String name = (file !=  null ? file.getPath() :  null );
        SecurityManager security = System.getSecurityManager();
        if (security !=  null ) {
            security.checkWrite(name);
        }
        if (name ==  null ) {
            throw new NullPointerException();
        }
        if (file.isInvalid()) {
            throw new FileNotFoundException( "Invalid file path" );
        }
        this .fd =  new FileDescriptor();
        this .append = append;
        this .path = name;
        fd.incrementAndGetUseCount();
        open(name, append);
    }

    /* 通过文件描述符类创建文件输入流 */
    public FileOutputStream(FileDescriptor fdObj) {
        SecurityManager security = System.getSecurityManager();
        if (fdObj ==  null ) {
            throw new NullPointerException();
        }
        if (security !=  null ) {
            security.checkWrite(fdObj);
        }
        this .fd = fdObj;
        this .path =  null ;
        this .append =  false ;

        fd.incrementAndGetUseCount();
    }

    /* 打开文件,并确定文件写入起始处模式 */
    private native void open(String name,  boolean append)
        throws FileNotFoundException;

    /* 将指定的字节b写入到该文件输入流,并指定文件写入起始处模式 */
    private native void write( int b,  boolean append)  throws IOException;

    /* 将指定的字节b写入到该文件输入流 */
    public void write( int b)  throws IOException {
        Object traceContext = IoTrace.fileWriteBegin(path);
        int bytesWritten =  0 ;
        try {
            write(b, append);
            bytesWritten =  1 ;
        }  finally {
            IoTrace.fileWriteEnd(traceContext, bytesWritten);
        }
    }

    /* 将指定的字节数组写入该文件输入流,并指定文件写入起始处模式 */
    private native void writeBytes( byte b[],  int off,  int len,  boolean append)
        throws IOException;

    /* 将指定的字节数组b写入该文件输入流 */
    public void write( byte b[])  throws IOException {
        Object traceContext = IoTrace.fileWriteBegin(path);
        int bytesWritten =  0 ;
        try {
            writeBytes(b,  0 , b.length, append);
            bytesWritten = b.length;
        }  finally {
            IoTrace.fileWriteEnd(traceContext, bytesWritten);
        }
    }

    /* 将指定len长度的字节数组b写入该文件输入流 */
    public void write( byte b[],  int off,  int len)  throws IOException {
        Object traceContext = IoTrace.fileWriteBegin(path);
        int bytesWritten =  0 ;
        try {
            writeBytes(b, off, len, append);
            bytesWritten = len;
        }  finally {
            IoTrace.fileWriteEnd(traceContext, bytesWritten);
        }
    }

    /* 关闭此文件输出流并释放与此流有关的所有系统资源 */
    public void close()  throws IOException {
        synchronized (closeLock) {
            if (closed) {
                return ;
            }
            closed =  true ;
        }

        if (channel !=  null ) {
            fd.decrementAndGetUseCount();
            channel.close();
        }

        int useCount = fd.decrementAndGetUseCount();

        if ((useCount <=  0 ) || !isRunningFinalize()) {
            close0();
        }
    }

     public final FileDescriptor getFD()   throws IOException {
        if (fd !=  null )  return fd;
        throw new IOException();
     }

    public FileChannel getChannel() {
        synchronized ( this ) {
            if (channel ==  null ) {
                channel = FileChannelImpl.open(fd, path,  false ,  true , append,  this );

                fd.incrementAndGetUseCount();
            }
            return channel;
        }
    }

    protected void finalize()  throws IOException {
        if (fd !=  null ) {
            if (fd == FileDescriptor.out || fd == FileDescriptor.err) {
                flush();
            }  else {

                runningFinalize.set(Boolean.TRUE);
                try {
                    close();
                }  finally {
                    runningFinalize.set(Boolean.FALSE);
                }
            }
        }
    }

    private native void close0()  throws IOException;

    private static native void initIDs();

    static {
        initIDs();
    }

}

1. 三个核心方法

三个核心方法,也就是Override(重写)了抽象类OutputStreamwrite方法。

void write(int b) 方法,即

public void write( int b)  throws IOException

代码实现中很简单,一个try中调用本地nativewrite()方法,直接将指定的字节b写入文件输出流。IoTrace.fileReadEnd()的意思和上面FileInputStream意思一致。

 

void write(byte b[]) 方法,即

public void write( byte b[])  throws IOException

代码实现也是比较简单的,也是一个try中调用本地nativewriteBytes()方法,直接将指定的字节数组写入该文件输入流。

 

void write(byte b[], int off, int len) 方法,即

public void write( byte b[],  int off,  int len)  throws IOException

代码实现和 void write(byte b[]) 方法 一样,直接将指定的字节数组写入该文件输入流。

 

2. 值得一提的native方法

上面核心方法中为什么实现简单,因为工作量都在native方法里面,即JVM里面实现了。native倒是不少一一列举吧:

native void open(String name) // 打开文件,为了下一步读取文件内容

native void write(int b, boolean append) // 直接将指定的字节b写入文件输出流

native native void writeBytes(byte b[], int off, int len, boolean append) // 直接将指定的字节数组写入该文件输入流。

native void close0() // 关闭该文件输入流及涉及的资源,比如说如果该文件输入流的FileChannel对被获取后,需要对FileChannel进行close。

 

相似之处

其实到这里,该想一想。两个源码实现很相似,而且native方法也很相似。其实不能说“相似”,应该以“对应”来概括它们。

它们是一组,是一根吸管的两个孔的关系:“一个Input一个Output”。

 

休息一下吧~ 看看小广告:

开源代码都在我的gitHub上哦 — https://github.com/JeffLi1993 作者留言“请手贱,点项目star,支持支持拜托拜托”

 

四、使用案例

下面先看代码:

package org.javacore.io;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;

/*
 * Copyright [2015] [Jeff Lee]
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

/**
 * @author Jeff Lee
 * @since 2015-10-8 20:06:03
 * FileInputStream&FileOutputStream使用案例
 */
public class FileIOStreamT {
    private static final String thisFilePath = 
            "src" + File.separator +
            "org" + File.separator +
            "javacore" + File.separator +
            "io" + File.separator +
            "FileIOStreamT.java" ;
    public static void main(String[] args)  throws IOException {
        // 创建文件输入流
        FileInputStream fileInputStream =  new FileInputStream(thisFilePath);
        // 创建文件输出流
        FileOutputStream fileOutputStream =   new FileOutputStream( "data.txt" );
       
        // 创建流的最大字节数组
        byte [] inOutBytes =  new byte [fileInputStream.available()];
        // 将文件输入流读取,保存至inOutBytes数组
        fileInputStream.read(inOutBytes);
        // 将inOutBytes数组,写出到data.txt文件中
        fileOutputStream.write(inOutBytes);
       
        fileOutputStream.close();
        fileInputStream.close();
    }
}

运行后,会发现根目录中出现了一个“data.txt”文件,内容为上面的代码。

1. 简单地分析下源码:

1、创建了FileInputStream,读取该代码文件为文件输入流。

2、创建了FileOutputStream,作为文件输出流,输出至data.txt文件。

3、针对流的字节数组,一个 read ,一个write,完成读取和写入。

4、关闭流

2. 代码调用的流程如图所示:

iostream

3. 代码虽简单,但是有点小问题

FileInputStream.available() 是返回流中的估计剩余字节数。所以一般不会用此方法。

一般做法,比如创建一个 byte数组,大小1K。然后read至其返回值不为-1,一直读取即可。边读边写。

 

五、思考与小结

FileInputStream & FileOutputStream 是一对来自 InputStream和OutputStream的实现类。用于本地文件读写(二进制格式按顺序读写)

本文小结:

1、FileInputStream 源码分析

2、FileOutputStream 资源分析

3、FileInputStream & FileOutputStream 使用案例

4、其源码调用过程

 

posted @ 2022-08-20 19:42  云散轻尘  阅读(19)  评论(0)    收藏  举报