转载请注明出处:http://www.cnblogs.com/lighten/p/7267553.html

1.前言

  此章介绍IO包中剩余未介绍的几个流和工具类,包括LineNumberReader、PushbackReader、RandomAccessFile、StreamTokenizer这四个类。至此IO模块就基本讲述完毕,能力有限,有些叙述的不是很好,对象流也大体跳过了主要内容,之后会有一个整个IO中比较重要的知识的总结,IO就告一段落了。

2.LineNumberReader

  在之前的章节介绍其它字节流的时候已经介绍了LineNumberInputStream,这里对此类也不做过多介绍,大致与其一样,只是换成了对字符的操作罢了。

  这里的逻辑有了些小修改。都是默认\r后面是\n符号,读取到了\r就改变skipLF的值,行号+1返回\n。下一个读到\n就把状态改回来。

@Test
public void test() throws IOException {
String line = "123\r\t134\n43\n\rdad\n";
StringReader reader = new StringReader(line);
LineNumberReader numberReader = new LineNumberReader(reader);
StringWriter writer = new StringWriter();
int length;
char[] buffer = new char[1024];
while((length = numberReader.read(buffer))!=-1) {
writer.write(buffer, 0, length);
}
System.out.println(writer);
}

  根据那段代码的逻辑就会发生这种情况,\r单独被看做是一个换行符。一般跟上\n就判断是同一个换行符,不是就重置重新判断。

3.PushbackReader

  这个和之前所说的PushbackInputStream也相似。其就是将流中读出来的数据放入一个内存数据中,再读取的时候先读取内存数组中的内容,再从流中获取新的数据,一个"反悔"的机制。但是使用的时候要注意。如果只是一个字符,可以通过unread(int)方法放回去,但是没有再次读取出来的时候,不能再放入一个数据,否则会打乱整个数据流的顺序。如果要回滚的不止一个字符,那就使用unread(char[],int,int)方法,同样要先读取这些回滚的数据,才能再次放入。总而言之,同时只能回滚一次,必须等回滚的数据再次被读取,才能再次回滚。

@Test
public void test2() throws IOException {
String line = "123";
StringReader reader = new StringReader(line);
PushbackReader pushReader = new PushbackReader(reader, 2);
int c;
c = reader.read();
pushReader.unread(c);
c = reader.read();
pushReader.unread(c);
while((c=pushReader.read()) != -1) {
System.out.println((char)c);
}
}

  当然,这里只是这样一提,因为没有人会这样写代码,字符流被pushreader包装后只会操作pushReader的read方法,而原流的方法是不会使用的。这样你要回滚后再回滚,这中间肯定要读取pushReader(不然数据从哪来,数据从其它源来也就违背了这个类的本意),也就完成了上面所说的回滚后的读取了。

4.RandomAccessFile

  这个类也是一个读取文件流的类,但是与FileInputStream不同的地方在于,之前的文件流只能从头读取到尾,写入也一样(除了append追加到文件末尾)。这个类的主要作用在于能够随机的读取指定位置的文件内容。

  其有四种模式:r只读(1);

         rw读写,不存在创建(2);

         rws除了rw的内容,对文件的内容和元数据的更新同步到底层存储设备(4)

         rwd除了rw的内容,对文件的内容的更新同步到底层存储设备(8)

  RandomAccessFile随机读取文件指定位置的内容是通过native方法seek实现的。其它的也只是一般的方法而已,没有什么可说的。下面看一个例子来熟悉一下这个类的使用。

    @Test
public void test3() throws IOException {
String path = getClass().getClassLoader().getResource("").getPath();
String filePath = path+"test.txt";
File file = new File(filePath);
if(file.exists()) {
file.delete();
file.createNewFile();
} else {
file.createNewFile();
}
FileOutputStream fos = new FileOutputStream(file);
fos.write("0123456789".getBytes());
fos.flush();
fos.close();
RandomAccessFile accessFile = new RandomAccessFile(filePath, "rw");
accessFile.seek(2);
System.out.println((char)accessFile.read());
accessFile.seek(4);
System.out.println((char)accessFile.read());
accessFile.write("abcd".getBytes());
accessFile.close();
FileReader reader = new FileReader(filePath);
int length;
char[] buffer = new char[1024];
StringWriter writer = new StringWriter();
while((length = reader.read(buffer))!=-1) {
writer.write(buffer, 0, length);
}
System.out.println(writer.toString());
reader.close();
writer.close();
}

  可以看出上面的seek方法,和seek后写入的最终情况。seek可以直接定位到制定的位置(从0开始),write是从当前位置内容开始写入,至于上面是5被替换了,而不是4是因为为了输出效果read了一次,位置后移了一位造成的。其它的也没有什么好说明的了。

5.StreamTokenizer

  这个类是一个工具类,主要用于处理字符流。它可以将一个字符流切出tokens,一次读取一个,识别的有标识符,数字,引用,字符串和不同风格的注释。控制方法就是通过一个table,在程序中就是ctype数组,还有一些数字标记。使用起来十分简单,看一个例子就可以了:

    @Test
public void test4() throws IOException {
String text = "她说:\"这个 5 元!\"";
StringReader reader = new StringReader(text);
StreamTokenizer tokenizer = new StreamTokenizer(reader);
// tokenizer.ordinaryChar('\"');
while(tokenizer.nextToken()!=StreamTokenizer.TT_EOF) {
if(tokenizer.ttype == StreamTokenizer.TT_NUMBER) {
System.out.println(tokenizer.nval);
} else if(tokenizer.ttype == StreamTokenizer.TT_WORD){
System.out.println(tokenizer.sval);
} else if(tokenizer.ttype == StreamTokenizer.TT_EOL){
System.out.println("换行符");
} else {
System.out.println(tokenizer.sval);
}
}
}

  这个类的作用也比较明显了,就是提取出合乎规则的内容了。具体解析过程不再叙述。util包中还有一个StringTokenizer类,正则同样可以完成这样的任务,不过都是必须先取出来,不像这个类流式处理罢了。

Java之IO(十四)IO包中其它类的更多相关文章

  1. 27 Java动态加载第三方jar包中的类

    我加载的方法是://参数fileName是jar包的路径,processorName 是业务类的包名+类名public static A load(String fileName, String pr ...

  2. Java并发(十四):并发工具类——CountDownLatch

    先做总结: 1.CountDownLatch 是什么? CountDownLatch 允许一个或多个线程等待其他线程(不一定是线程,某个操作)完成之后再执行. CountDownLatch的构造函数接 ...

  3. Java从零开始学十四(包和访问控制)

    一.java中的包 Java文件的组织形式Windows中的文件功能类似 在开发比较大的项目时,不可能只涉及到一个java文件,可能要创建几十,甚至几百个java文件,这个时候,我们就可以使用包,把相 ...

  4. “全栈2019”Java异常第十四章:将异常输出到文本文件中

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java异 ...

  5. “全栈2019”Java第八十四章:接口中嵌套接口详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  6. Java进阶(三十四)Integer与int的种种比较你知道多少?

    Java进阶(三十四)Integer与int的种种比较你知道多少? 前言 如果面试官问Integer与int的区别:估计大多数人只会说到两点:Ingeter是int的包装类,注意是一个类:int的初值 ...

  7. Java 线程总结(十四)

    1.在异步任务进程中,一种常见的场景是,主线程提交多个异步任务,然后希望有任务完成就处理结果,并且按任务完成顺序逐个处理,对于这种场景,Java 并发包提供了一个方便的方法,使用 Completion ...

  8. Java开发学习(十四)----Spring整合Mybatis及Junit

    一.Spring整合Mybatis思路分析 1.1 环境准备 步骤1:准备数据库表 Mybatis是来操作数据库表,所以先创建一个数据库及表 create database spring_db cha ...

  9. 二十四、Struts2中的UI标签

    二十四.Struts2中的UI标签 Struts2中UI标签的优势: 数据回显 页面布局和排版(Freemark),struts2提供了一些常用的排版(主题:xhtml默认 simple ajax) ...

随机推荐

  1. 19. Fight over Fox-hunting 猎狐引发的冲突

    . Fight over Fox-hunting 猎狐引发的冲突 ① Foxes and farmers have never got on well.These small dog-like ani ...

  2. C++之类和对象的使用(二)

    析构函数 析构函数的作用并不是删除对象,而是在撤销对象占用的内存之前完成一系列清理工作,使这部分内存可以被程序分配给新对象使用.对象生命周期结束,程序就自动执行析构函数来完成这些工作. 析构函数是一种 ...

  3. (水题) Div 3 -- SGU -- 105

    链接: http://vj.acmclub.cn/contest/view.action?cid=168#problem/E 时限:250MS     内存:4096KB     64位IO格式:%I ...

  4. gridview的编辑,更新,取消,自动分页等

    gridview编辑列,把左下角的"自动生成字段"的复选框的勾去掉 添加boundfield(绑定列)将其datafield设置为productname,headertext设置为 ...

  5. 译:Microsoft/ReactXP 简介

    在Github的Microsoft项目中发现一个名为ReactXP的项目,这是一个由Skype团队开发的,用于进行Web及跨平台APP开发的库(建立在React Js 和 ReactNative之上) ...

  6. SQL Server 统计信息(Statistics)-概念,原理,应用,维护

    前言:统计信息作为sql server优化器生成执行计划的重要参考,需要数据库开发人员,数据库管理员对其有一定的理解,从而合理高效的应用,管理. 第一部分 概念 统计信息(statistics):描述 ...

  7. 设计模式之代理模式(Proxy Pattern)_补充篇

    写在前面: 代理模式的内部原理,作用及远程代理的实现在上一篇博文中都做了详细解释,本文只是对其内容的补充,介绍其它代理 一.虚拟代理 首先,明确虚拟代理的作用:在巨大对象被真正创建出来之前,用虚拟代理 ...

  8. NLayerAppV3--DDD之领域层

    回顾:NLayerAppV3是一个使用.net 2.1实现的经典DDD的分层架构的项目. NLayerAppV3是在NLayerAppV2的基础上,使用.net core2.1进行重新构建的:它包含了 ...

  9. 使用ABP框架踩过的坑系列1

        企业级(例如ERP)应用, 一遍一遍的在重复:认证.验证.异常处理.日志.国际化和本地化.数据库连接管理.配置管理. 审计记录等,同时.NET有很多最佳实践:分层.模块化.DDD领域驱动.DI ...

  10. 构建NetCore应用框架之实战篇(四):BitAdminCore框架1.0登录功能细化及技术选型

    本篇承接上篇内容,如果你不小心点击进来,建议从第一篇开始完整阅读,文章内容继承性连贯性. 构建NetCore应用框架之实战篇系列 一.BitAdminCore框架1.0版本 1.1.0版本是指最小版本 ...