草庐IT

Netty基础-NIO(二)

石头耳东 2023-04-14 原文

前置文章:
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(二)的全部内容,感谢阅读。

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

有关Netty基础-NIO(二)的更多相关文章

  1. postman接口测试工具-基础使用教程 - 2

    1.postman介绍Postman一款非常流行的API调试工具。其实,开发人员用的更多。因为测试人员做接口测试会有更多选择,例如Jmeter、soapUI等。不过,对于开发过程中去调试接口,Postman确实足够的简单方便,而且功能强大。2.下载安装官网地址:https://www.postman.com/下载完成后双击安装吧,安装过程极其简单,无需任何操作3.使用教程这里以百度为例,工具使用简单,填写URL地址即可发送请求,在下方查看响应结果和响应状态码常用方法都有支持请求方法:getpostputdeleteGet、Post、Put与Delete的作用get:请求方法一般是用于数据查询,

  2. 软件测试基础 - 2

    Ⅰ软件测试基础一、软件测试基础理论1、软件测试的必要性所有的产品或者服务上线都需要测试2、测试的发展过程3、什么是软件测试找bug,发现缺陷4、测试的定义使用人工或自动的手段来运行或者测试某个系统的过程。目的在于检测它是否满足规定的需求。弄清预期结果和实际结果的差别。5、测试的目的以最小的人力、物力和时间找出软件中潜在的错误和缺陷6、测试的原则28原则:20%的主要功能要重点测(eg:支付宝的支付功能,其他功能都是次要的)80%的错误存在于20%的代码中7、测试标准8、测试的基本要求功能测试性能测试安全性测试兼容性测试易用性测试外观界面测试可靠性测试二、质量模型衡量一个优秀软件的维度①功能性功

  3. ES基础入门 - 2

    ES一、简介1、ElasticStackES技术栈:ElasticSearch:存数据+搜索;QL;Kibana:Web可视化平台,分析。LogStash:日志收集,Log4j:产生日志;log.info(xxx)。。。。使用场景:metrics:指标监控…2、基本概念Index(索引)动词:保存(插入)名词:类似MySQL数据库,给数据Type(类型)已废弃,以前类似MySQL的表现在用索引对数据分类Document(文档)真正要保存的一个JSON数据{name:"tcx"}二、入门实战{"name":"DESKTOP-1TSVGKG","cluster_name":"elasticsear

  4. 【网络】-- 网络基础 - 2

    (本文是网络的宏观的概念铺垫)目录计算机网络背景网络发展认识"协议"网络协议初识协议分层OSI七层模型TCP/IP五层(或四层)模型报头以太网碰撞路由器IP地址和MAC地址IP地址与MAC地址总结IP地址MAC地址计算机网络背景网络发展        是最开始先有的计算机,计算机后来因为多项技术的水平升高,逐渐的计算机变的小型化、高效化。后来因为计算机其本身的计算能力比较的快速:独立模式:计算机之间相互独立。    如:有三个人,每个人做的不同的事物,但是是需要协作的完成。    而这三个人所做的事是需要进行协作的,然而刚开始因为每一台计算机之间都是互相独立的。所以前面的人处理完了就需要将数据

  5. 【Elasticsearch基础】Elasticsearch索引、文档以及映射操作详解 - 2

    文章目录概念索引相关操作创建索引更新副本查看索引删除索引索引的打开与关闭收缩索引索引别名查询索引别名文档相关操作新建文档查询文档更新文档删除文档映射相关操作查询文档映射创建静态映射创建索引并添加映射概念es中有三个概念要清楚,分别为索引、映射和文档(不用死记硬背,大概有个印象就可以)索引可理解为MySQL数据库;映射可理解为MySQL的表结构;文档可理解为MySQL表中的每行数据静态映射和动态映射上面已经介绍了,映射可理解为MySQL的表结构,在MySQL中,向表中插入数据是需要先创建表结构的;但在es中不必这样,可以直接插入文档,es可以根据插入的文档(数据),动态的创建映射(表结构),这就

  6. c++基础-运算符 - 2

    目录1关系运算符2运算符优先级3关系表达式的书写代码实例:下面是面试中可能遇到的问题:1关系运算符C++中有6个关系运算符,用于比较两个值的大小关系,它们分别是:运算符描述==等于!=不等于小于>大于小于等于>=大于等于这些运算符返回一个布尔值,即true或false。例如,当x等于y时,x==y的结果为true,否则结果为false。2运算符优先级在C++中,关系运算符的优先级高于赋值运算符,但低于算术运算符。以下是关系运算符的优先级,从高到低排列:运算符描述>,,>=,关系运算符==,!=相等性运算符&&逻辑与`如果在表达式中有多个运算符,则按照优先级顺序依次进行运算。3关系表达式的书写在

  7. 计算机必读基础书籍 - 2

    一.计算机组成原理    这本书利用组合逻辑、同步时序逻辑电路设计的相关知识,从逻辑门开始逐步构建运算器、存储器、数据通路和控制器,最终集成为完整的CU原型系统,使读者从设计者的角度理解计算机部件构成及运行的基本原理,掌握软硬件协同的概念。    全书共9章,主要内容包括计算机系统概述、数据信息的表示、运算方法与运算器、存储系统、指令系统、中央处理器、指令流水线、总线系统、输入输出系统。1.计算机系统概述1.1计算机发展历程    计算机是一种能够按照事先存储的程序,自动、高速、准确地对相关信息进行处理的电子设备。1946年2月,世界上第一台电子数字计算机ENIAC(ElectronicNum

  8. 0基础学习软件测试有哪些建议 - 2

    其实现在基础的资料和视频到处都是,就是看你有没有认真的去找学习资源了,去哪里学习都是要看你个人靠谱不靠谱,再好的教程和老师,你自己学习不进去也是白搭在正式选择之前,大可以在各种学习网站里面找找学习资源先自己学习一下为什么选择学软件测试?同学们理由众多!大概分这几类:①不受开发语言、行业产品变化限制;②入门更简单,对零基础、女生都友好;③软件项目都需要测试人员,职业生涯稳;④学习周期短,但薪资并不低。要想“肩扛”一条线?需掌握三大技能:技能1:掌握测试流程,熟悉系统框架能提前与开发人员一起制定测试计划,通过测试左移,推动代码评审,代码审计,单元测试,自动化冒烟测试,来保证研发阶段的质量。技能2:

  9. ruby - Ruby基础知识 - 2

    Asitcurrentlystands,thisquestionisnotagoodfitforourQ&Aformat.Weexpectanswerstobesupportedbyfacts,references,orexpertise,butthisquestionwilllikelysolicitdebate,arguments,polling,orextendeddiscussion.Ifyoufeelthatthisquestioncanbeimprovedandpossiblyreopened,visitthehelpcenter提供指导。已关闭8年。什么是学习ruby语言

  10. ruby-on-rails - 安装 nio4r 时出错 - 2

    请帮帮我。我尝试安装gemnio4r,但此日志有错误:ERROR:Errorinstallingnio4r:ERROR:Failedtobuildgemnativeextension.currentdirectory:/var/lib/gems/2.3.0/gems/nio4r-1.2.1/ext/nio4r/usr/bin/ruby2.3-r./siteconf20161020-13985-1c6zxok.rbextconf.rbmkmf.rbcan'tfindheaderfilesforrubyat/usr/lib/ruby/include/ruby.hextconffailed,

随机推荐