Netty基础-NIO(二)

前置文章:
Netty基础-NIO(一),该文主要引入NIO三大组件,介绍了Buffer结构,及Buffer具体使用。

零、本文纲要

一、Channel

  1. FileChannel
  2. FileChannel传输

二、补充:Path & Paths & Files 类

  1. Path & Paths
  2. Files
  3. Files 类的 walkFileTree方法 & walk方法

一、Channel

1. FileChannel

注意:FileChannel 只能工作在阻塞模式下

① 获取方式

FileInputStream → 只读;
FileOutputStream → 只写;
RandomAccessFile → 读写;

FileChannel readChannel = new FileInputStream("src/main/resources/data.txt").getChannel();
FileChannel writeChannel = new FileOutputStream("src/main/resources/data_back.txt").getChannel();
final RandomAccessFile rwChannel = new RandomAccessFile("src/main/resources/data.txt", "rw");

② read方法

读数据到buffer,返回值表示读到了多少字节,0 / -1 表示到达了文件的末尾。

ByteBuffer buffer = ByteBuffer.allocate(16);
FileChannel readChannel = new FileInputStream("src/main/resources/data.txt").getChannel();
// TODO 只读Channel中读取数据,填充至Buffer
int read = readChannel.read(buffer);
log.debug("当前读取了:{}字节内容。", read);
debugAll(buffer);
readChannel.close();

注意:此处官方一般用 -1 来做读完数据的标志。

③ channel#write方法 & buffer#hasRemaining方法

FileChannel writeChannel = new FileOutputStream("src/main/resources/data_back.txt").getChannel();
while (buffer.hasRemaining()) {
    writeChannel.write(buffer);
}
writeChannel.close();

④ close方法

因为涉及到资源流处理,使用完关闭资源是良好的习惯。

⑤ position方法

该方法用于获取position指针的当前位置

log.debug("buffer position is: {}", buffer.position());
log.debug("channel position is: {}", writeChannel.position());
buffer position is: 11
channel position is: 11

⑥ size方法

该方法获取能文件的大小

⑦ force方法

操作系统出于性能的考虑,会将数据缓存,不是立刻写入磁盘。
可以调用 force(true) 方法将文件内容和元数据(文件的权限等信息)立刻写入磁盘。

2. FileChannel传输

① 传统的方式

readChannel 从文件读取数据至 buffer ,而后从 buffer 取数据通过 writeChannel 写数据至文件

ByteBuffer buffer = ByteBuffer.allocate(16);
FileChannel readChannel = new FileInputStream("src/main/resources/data.txt").getChannel();
int read = readChannel.read(buffer);
log.debug("当前读取了:{}字节内容。", read);
debugAll(buffer);
readChannel.close();

buffer.flip(); //注意:要手动将position重置为0

FileChannel writeChannel = new FileOutputStream("src/main/resources/data_back.txt").getChannel();
while (buffer.hasRemaining()) {
    writeChannel.force(true);
    writeChannel.write(buffer);
    log.debug("buffer position is: {}", buffer.position());
    log.debug("channel position is: {}", writeChannel.position());
}
writeChannel.close();

② transferTo方法

final String IN_FILE = "src/main/resources/data.txt";
final String OUT_FILE = "src/main/resources/data_back.txt";
try (
    FileChannel readChannel = new FileInputStream(IN_FILE).getChannel();
    FileChannel writeChannel = new FileOutputStream(OUT_FILE).getChannel();
) {
    readChannel.transferTo(0, readChannel.size(), writeChannel);
} catch (IOException e) {
    log.debug(e.getMessage());
}

二、补充:Path & Paths & Files 类

1. Path & Paths

Path:用来表示文件路径
Paths:用于获取Path的工具类

String URI = "src/main/resources/data.txt";
String URL = "D:\\JavaStudy\\Level1\\netty-nio\\src\\main\\resources\\data.txt";
Path sourceOne = Paths.get(URI);
Path sourceTwo = Paths.get(URL);

2. Files

① exists方法

用于判断文件是否存在。

log.debug("Is file exists? {}", Files.exists(sourceOne));

② createDirectory方法 & createDirectories方法

用于创建目录/多级目录的方法。

createDirectory方法
a、如果目录已存在,会抛异常 FileAlreadyExistsException;
b、不能一次创建多级目录,否则会抛异常 NoSuchFileException。
c、createDirectories方法可以创建多级目录。

③ copy方法

用于拷贝文件的方法。

Path source = Paths.get("stone/source.txt");
Path backup = Paths.get("stone/backup.txt");

Files.copy(source, backup);

注意:如果文件已存在,会抛异常 FileAlreadyExistsException。

StandardCopyOption可以用于控制copy文件的形式,REPLACE_EXISTING用于覆盖已存在的backup文件,如下:

Files.copy(source, backup, StandardCopyOption.REPLACE_EXISTING);

④ move方法

用于移动文件,StandardCopyOption.ATOMIC_MOVE 保证文件移动的原子性,如下:

Files.move(source, backup, StandardCopyOption.ATOMIC_MOVE);

⑤ delete方法

用于删除目录的方法。

Path target = Paths.get("stone/directory");

Files.delete(target);

注意:如果目录还有内容,会抛异常 DirectoryNotEmptyException。

3. Files 类的 walkFileTree方法 & walk方法

① 遍历目录文件

此处我们重写 SimpleFileVisitor 内部类的 preVisitDirectory、visitFile 方法

final String JAVA_PATH = "C:\\Program Files\\Java\\jdk1.8.0_311";
Path path = Paths.get(JAVA_PATH);
AtomicInteger dirCount = new AtomicInteger();
AtomicInteger fileCount = new AtomicInteger();
Files.walkFileTree(path, new SimpleFileVisitor<Path>(){
    @Override
    public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs)
            throws IOException {
        log.debug(dir.getFileName().toString());
        dirCount.incrementAndGet();
        return super.preVisitDirectory(dir, attrs);
    }

    @Override
    public FileVisitResult visitFile(Path file, BasicFileAttributes attrs)
            throws IOException {
        log.debug(file.getFileName().toString());
        fileCount.incrementAndGet();
        return super.visitFile(file, attrs);
    }
});
log.debug("directory count is: {}.", dirCount);
log.debug("file count is: {}.", fileCount);

② 统计指定文件类型的数目

比如统计 .jar包 的数量

final String JAVA_PATH = "C:\\Program Files\\Java\\jdk1.8.0_311";
Path path = Paths.get(JAVA_PATH);
AtomicInteger fileCount = new AtomicInteger();
Files.walkFileTree(path, new SimpleFileVisitor<Path>(){
    @Override
    public FileVisitResult visitFile(Path file, BasicFileAttributes attrs)
            throws IOException {
        String fileName = file.getFileName().toString();
        if (fileName.endsWith(".jar")) {
            log.debug("{}", fileName);
            fileCount.incrementAndGet();
        }
        return super.visitFile(file, attrs);
    }
});
log.debug("file count is: {}.", fileCount);

③ 删除多级目录

注意:谨慎操作

final String DELETE_PATH = "D:\\use-to-delete";
Path path = Paths.get(DELETE_PATH);
AtomicInteger dirCount = new AtomicInteger();
AtomicInteger fileCount = new AtomicInteger();
Files.walkFileTree(path, new SimpleFileVisitor<Path>(){
    @Override
    public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
        fileCount.incrementAndGet();
        Files.delete(file);
        return super.visitFile(file, attrs);
    }

    @Override
    public FileVisitResult postVisitDirectory(Path dir, IOException exc) throws IOException {
        dirCount.incrementAndGet();
        Files.delete(dir);
        return super.postVisitDirectory(dir, exc);
    }
});
log.debug("The deleted file count is: {}.", fileCount);
log.debug("The deleted directories count is: {}.", dirCount);

④ 拷贝多级目录

final String SOURCE = "D:\\logs";
final String BACKUP = "D:\\logs-backup";
Files.walk(Paths.get(SOURCE)).forEach( path -> {
    try {
        String backupName = path.toString().replace(SOURCE, BACKUP);
        if (Files.isDirectory(path)) {
            Files.createDirectory(Paths.get(backupName));
        }
        else if (Files.isRegularFile(path)) {
            Files.copy(path, Paths.get(backupName));
        }
    } catch (IOException e) {
        log.debug(e.getMessage());
    }
});

⑤ 补充:SimpleFileVisitor类

属性Spring框架的话这部分是不是非常熟悉类似,此处不展开。

a、preVisitDirectory:访问目录前;
b、visitFile:访问文件;
c、visitFileFailed:访问不可访问的文件(无权限);
d、postVisitDirectory:访问目录后。

public class SimpleFileVisitor<T> implements FileVisitor<T> {

    protected SimpleFileVisitor() {
    }

    @Override
    public FileVisitResult preVisitDirectory(T dir, BasicFileAttributes attrs)
        throws IOException
    {
        Objects.requireNonNull(dir);
        Objects.requireNonNull(attrs);
        return FileVisitResult.CONTINUE;
    }

    @Override
    public FileVisitResult visitFile(T file, BasicFileAttributes attrs)
        throws IOException
    {
        Objects.requireNonNull(file);
        Objects.requireNonNull(attrs);
        return FileVisitResult.CONTINUE;
    }

    @Override
    public FileVisitResult visitFileFailed(T file, IOException exc)
        throws IOException
    {
        Objects.requireNonNull(file);
        throw exc;
    }

    @Override
    public FileVisitResult postVisitDirectory(T dir, IOException exc)
        throws IOException
    {
        Objects.requireNonNull(dir);
        if (exc != null)
            throw exc;
        return FileVisitResult.CONTINUE;
    }
}

三、结尾

以上即为Netty基础-NIO(二)的全部内容,感谢阅读。

注意:该文集系列铺垫的内容会稍微有些多。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容