Java之IO(十四)IO包中其它类
转载请注明出处: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包中其它类的更多相关文章
- 27 Java动态加载第三方jar包中的类
我加载的方法是://参数fileName是jar包的路径,processorName 是业务类的包名+类名public static A load(String fileName, String pr ...
- Java并发(十四):并发工具类——CountDownLatch
先做总结: 1.CountDownLatch 是什么? CountDownLatch 允许一个或多个线程等待其他线程(不一定是线程,某个操作)完成之后再执行. CountDownLatch的构造函数接 ...
- Java从零开始学十四(包和访问控制)
一.java中的包 Java文件的组织形式Windows中的文件功能类似 在开发比较大的项目时,不可能只涉及到一个java文件,可能要创建几十,甚至几百个java文件,这个时候,我们就可以使用包,把相 ...
- “全栈2019”Java异常第十四章:将异常输出到文本文件中
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java异 ...
- “全栈2019”Java第八十四章:接口中嵌套接口详解
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- Java进阶(三十四)Integer与int的种种比较你知道多少?
Java进阶(三十四)Integer与int的种种比较你知道多少? 前言 如果面试官问Integer与int的区别:估计大多数人只会说到两点:Ingeter是int的包装类,注意是一个类:int的初值 ...
- Java 线程总结(十四)
1.在异步任务进程中,一种常见的场景是,主线程提交多个异步任务,然后希望有任务完成就处理结果,并且按任务完成顺序逐个处理,对于这种场景,Java 并发包提供了一个方便的方法,使用 Completion ...
- Java开发学习(十四)----Spring整合Mybatis及Junit
一.Spring整合Mybatis思路分析 1.1 环境准备 步骤1:准备数据库表 Mybatis是来操作数据库表,所以先创建一个数据库及表 create database spring_db cha ...
- 二十四、Struts2中的UI标签
二十四.Struts2中的UI标签 Struts2中UI标签的优势: 数据回显 页面布局和排版(Freemark),struts2提供了一些常用的排版(主题:xhtml默认 simple ajax) ...
随机推荐
- 19. Fight over Fox-hunting 猎狐引发的冲突
. Fight over Fox-hunting 猎狐引发的冲突 ① Foxes and farmers have never got on well.These small dog-like ani ...
- C++之类和对象的使用(二)
析构函数 析构函数的作用并不是删除对象,而是在撤销对象占用的内存之前完成一系列清理工作,使这部分内存可以被程序分配给新对象使用.对象生命周期结束,程序就自动执行析构函数来完成这些工作. 析构函数是一种 ...
- (水题) Div 3 -- SGU -- 105
链接: http://vj.acmclub.cn/contest/view.action?cid=168#problem/E 时限:250MS 内存:4096KB 64位IO格式:%I ...
- gridview的编辑,更新,取消,自动分页等
gridview编辑列,把左下角的"自动生成字段"的复选框的勾去掉 添加boundfield(绑定列)将其datafield设置为productname,headertext设置为 ...
- 译:Microsoft/ReactXP 简介
在Github的Microsoft项目中发现一个名为ReactXP的项目,这是一个由Skype团队开发的,用于进行Web及跨平台APP开发的库(建立在React Js 和 ReactNative之上) ...
- SQL Server 统计信息(Statistics)-概念,原理,应用,维护
前言:统计信息作为sql server优化器生成执行计划的重要参考,需要数据库开发人员,数据库管理员对其有一定的理解,从而合理高效的应用,管理. 第一部分 概念 统计信息(statistics):描述 ...
- 设计模式之代理模式(Proxy Pattern)_补充篇
写在前面: 代理模式的内部原理,作用及远程代理的实现在上一篇博文中都做了详细解释,本文只是对其内容的补充,介绍其它代理 一.虚拟代理 首先,明确虚拟代理的作用:在巨大对象被真正创建出来之前,用虚拟代理 ...
- NLayerAppV3--DDD之领域层
回顾:NLayerAppV3是一个使用.net 2.1实现的经典DDD的分层架构的项目. NLayerAppV3是在NLayerAppV2的基础上,使用.net core2.1进行重新构建的:它包含了 ...
- 使用ABP框架踩过的坑系列1
企业级(例如ERP)应用, 一遍一遍的在重复:认证.验证.异常处理.日志.国际化和本地化.数据库连接管理.配置管理. 审计记录等,同时.NET有很多最佳实践:分层.模块化.DDD领域驱动.DI ...
- 构建NetCore应用框架之实战篇(四):BitAdminCore框架1.0登录功能细化及技术选型
本篇承接上篇内容,如果你不小心点击进来,建议从第一篇开始完整阅读,文章内容继承性连贯性. 构建NetCore应用框架之实战篇系列 一.BitAdminCore框架1.0版本 1.1.0版本是指最小版本 ...