今天小伙突然我FileDescriptor这个类有什么用?然后我就一脸懵逼的回答,我翻认识这个单词:“文件描述!”,这个类在Java中怎么用还真是不知道,于是我就开始看源码,百度,决定把它记录一下!

首先什么是FileDescriptor?

文件描述符。

用来表示开放文件、开放套接字等。

当FileDescriptor表示某文件时,我们可以通俗的将FileDescriptor看成是该文件。但是,我们不能直接通过FileDescriptor对该文件进行操作;若需要通过FileDescriptor对该文件进行操作,则需要新创建FileDescriptor对应的FileOutputStream,再对文件进行操作。

在FileDescriptor源码中,我们可以看到文件描述符有三种描述方式:

public static final FileDescriptor in = standardStream(0);
public static final FileDescriptor out = standardStream(1);
public static final FileDescriptor err = standardStream(2);

也就是说FileDescriptor类中有三种文件描述方式:

(01) in  -- 标准输入(键盘)的描述符

(02) out -- 标准输出(屏幕)的描述符

(03) err -- 标准错误输出(屏幕)的描述符

而且它们的实现方式都是基本一致的,所以我们重点根据out这个输出描述符来研究下这个类具体的作用。

out标准输出(屏幕)的描述符

out是标准输出(屏幕)的描述符。但是它有什么作用呢?

我们可以通俗理解,out就代表了标准输出(屏幕)。若我们要输出信息到屏幕上,即可通过out来进行操作;但是,out又没有提供输出信息到屏幕的接口(因为out本质是FileDescriptor对象,而FileDescriptor没有输出接口)。怎么办呢?

很简单,我们创建out对应的“输出流对象”,然后通过“输出流”的write()等输出接口就可以将信息输出到屏幕上。

如下代码:

private static final void testOut(){
        try {
            FileOutputStream out = new FileOutputStream(java.io.FileDescriptor.out);
            out.write('A');
            out.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

输出:

A

java早已为我们封装好了“能方便的在屏幕上输出信息的接口”:通过System.out,我们能方便的输出信息到屏幕上。

因此,我们可以等价的将上面的程序转换为如下代码:

System.out.print('A');

那么Out在FileDescriptor中是怎实现的呢?请看代码:

public final class FileDescriptor {

    private int fd;
    private long handle;
    private AtomicInteger useCount;
    public /**/ FileDescriptor() {
    fd = -1;
        handle = -1;
        useCount = new AtomicInteger();
    }

    private /* */ FileDescriptor(int fd) {
    this.fd = fd;
        handle = -1;
        useCount = new AtomicInteger();
    }

    static {
        initIDs();
    }
    public static final FileDescriptor out = standardStream(1);
    public boolean valid() {
    return ((handle != -1) || (fd != -1));
    }
    private static native void initIDs();

    private static native long set(int d);

    private static FileDescriptor standardStream(int fd) {
        FileDescriptor desc = new FileDescriptor();
        desc.handle = set(fd);
        return desc;
    }
    int incrementAndGetUseCount() {
        return useCount.incrementAndGet();
    }
    int decrementAndGetUseCount() {
        return useCount.decrementAndGet();
    }

}

从中,可以看出:

(01) out就是一个FileDescriptor对象。它是通过构造函数FileDescriptor(int fd)创建的。

(02) FileDescriptor(int fd)的操作:就是给fd对象(int类型)赋值,并新建一个使用计数变量useCount。

fd对象是非常重要的一个变量,“fd=1”就代表了“标准输出”,“fd=0”就代表了“标准输入”,“fd=2”就代表了“标准错误输出”。

FileOutputStream out = new FileOutputStream(FileDescriptor.out); 就是利用构造函数FileOutputStream(FileDescriptor fdObj)来创建“Filed.out对应的FileOutputStream对象”。