Java 流的概述及操作(转)
一、什么是流?
流就是字节序列的抽象概念,能被连续读取数据的数据源和能被连续写入数据的接收端就是流,流机制是Java及C++中的一个重要机制,通过流我们可以自由地控制文件、内存、IO设备等数据的流向。而IO流就是用于处理设备上的数据,如:硬盘、内存、键盘录入等。IO流根据处理类型的不同可分为字节流和字符流,根据流向的不同可分为输入流和输出流。
二、字节流和字符流的区别:
字符流,因为文件编码的不同,就有了对字符进行高效操作的字符流对象,它的原理就是基于字节流读取字节时去查了指定的码表。它和字节流的区别有两点:1.在读取数据的时候,字节流读到一个字节就返回一个字节,字符流使用了字节流读到一个或多个字节(一个中文对应的字节数是两个,在UTF-8码表中是3个字节)时,先去查指定的编码表,再将查到的字符返回;2.字节流可以处理所有类型的数据,如jpg、avi、mp3、wav等等,而字符流只能处理字符数据。所以可以根据处理的文件不同考虑使用字节流还是字符流,如果是纯文本数据可以优先考虑字符流,否则使用字节流。
三、IO体系,所具备的基本功能就是读和写:
1.字符流 |-- Reader(读) |-- Writer(写) Reader |--InputStreamReader |--FileReader:用于处理文件的字符读取流对象 Writer |--OutputStreamWriter |--FileWriter:用于处理文件的字符写入流对象
其实很容易就可以看出来,IO体系中的子类名后缀绝大部分是父类名称,而前缀则是体现子类特有功能的名称。
Reader中常见的方法: |--int read() 读取一个字符,并返回读到的这个字符,读到流的末尾则返回-1。 |--int read(char[]) 将读到的字符存入指定的数组中,返回的是读到的字符个数, 读到流的末尾则返回-1。 |--close() 读取字符其实用的是window系统的功能,就希望使用完毕后, 进行资源的释放。 FileReader除了自己的构造函数外没有特有的方法: |--用于读取文本文件的流对象。 |--用于关联文本文件。 |--构造函数FileReader(String fileName)
在读取流对象初始化时,必须要指定一个被读取的文件,
如果该文件不存在则会发生FileNotFoundException异常。
Writer中常见的方法:
|--write() 将一个字符写入到流中。 |--write(char[]) 将一个字符数组写入到流中。 |--writer(String) 将一个字符写入到流中。 |--flush() 刷新流,将流中的数据刷新到目的地中,流还存在。 |--close() 关闭资源,在关闭钱会先调用flush(), 刷新流中的数据到目的地。
FileWriter,除了自己的构造函数外没有特有的方法:
|--该类的特点
|--用于处理文本文件
|--没有默认的编码表
|--有临时缓冲
|--构造函数,在写入流对象初始化时,必须要有一个存储数据的目的地。
|--FileWriter(String fileName),该构造器是干什么用的呢?
|--调用系统资源
|--在指定位置创建一个文件,如果该文件已经存在则被覆盖。
|--FileWriter(String filename,Boolean append),这构造器的作用是?
当传入的boolean类型的值为true时,会在指定文件末尾处进行数据的续写。
清单1,将文本数据保存到文件中代码
private static void test1(){ FileWriter fw=null; try { //初始化FileWriter对象,指定文件名已经存储路径 fw=new FileWriter("D:/test.txt"); fw.write("将字符串写入流"); //将流中的数据刷新到目的地,流还在 fw.flush(); fw.write("将字符串写入流"); } catch (IOException e) { e.printStackTrace(); }finally{ if(fw!=null){ try { fw.close(); } catch (IOException e1) { e1.printStackTrace(); } } } }
清单2,读取一个已有文本文件,并将文本内容打印出来代码
private static void test2(){ FileReader fr=null; try { //初始化FileReader对象,指定文件路径 fr=new FileReader("D:/test.txt"); int ch=0; while((ch=fr.read())!=-1){ //每次读取一个字符,直到读到末尾-1为止 System.out.println((char)ch); } } catch (IOException e) { e.printStackTrace(); }finally{ if(fr!=null){ try { fr.close(); } catch (IOException e1) { e1.printStackTrace(); } } } } 这样每读到一个字符就打印出来,效率很不高,能不能按指定大小读取完后再打印出来呢?答案是当然可以的。
清单3,读取一个已有文本文件,读完1kb再将其读到的内容打印出来代码
private static void test3(){ FileReader fr=null; try { //初始化FileReader对象,指定文件路径 fr=new FileReader("D:/test.txt"); char[] buf=new char[1024]; int len=0; while((len=fr.read(buf))!=-1){ //每次读取1kb大小的字符,直到读到末尾-1为止 System.out.println(new String(buf,0,len)); } } catch (IOException e) { e.printStackTrace(); }finally{ if(fr!=null){ try { fr.close(); } catch (IOException e1) { e1.printStackTrace(); } } } }
字符流的缓冲区:
|--缓冲区的出现提高了对流的操作效率。
原理:其实就是将数组进行封装。
|--对应的对象
|--BufferedWriter
特有方法newLine(),跨平台的换行符。
|--BufferedReader
特有方法readLine(),一次读一行,到行标记时,将行标记
之前的字符数据作为字符串返回,读到末尾返回null。
|--说明
在使用缓冲区对象时,要明确,缓冲的存在是为了增强流
的功能而存在,所以在建立缓冲区对象时,要先有流对象
存在。其实缓冲区内部就是在使用流对象的方法,只不过
加入了数组对数据进行了临时存储,为了提高操作数据的
效率。
|--代码上的体现
|--写入缓冲区对象
根据前面所说的建立缓冲区时要先有流对象,并将其作为参数传递给缓冲区的构造函数
BufferedWriter bufw=new BufferedWriter(new FileWriter(“test.txt”));
bufw.write(“将数据写入缓冲区”);
bufw.flush();//将缓冲区的数据刷新到目的地
bufw.close();//其实关闭的是被包装在内部的流对象
|--读取缓冲区对象
BufferedReader bufr=new BufferedReader(new FileReader(“test.txt”));
String line=null;
while((line=bufr.readLine())!=null){
//每次读取一行,取出的数据不包含回车符
system.out.println(line);
}
bufr.close();
清单4,使用缓冲区对文本文件进行拷贝代码
private static void test4(){ BufferedReader bufr=null; BufferedWriter bufw=null; try { bufr=new BufferedReader(new FileReader("D:/a.txt")); bufw=new BufferedWriter(new FileWriter("D:/b.txt")); String line=null; while((line=bufr.readLine())!=null){ bufw.write(line);//每次将一行写入缓冲区 bufw.flush();//刷新到目的地 } } catch (IOException e) { e.printStackTrace(); }finally{ try { if(bufw!=null){ bufw.close(); } if(bufr!=null){ bufr.close(); } } catch (IOException e1) { e1.printStackTrace(); } } }
仔细看可以发现,程序里面的FileReader对象和FileWriter对象直接new出来且没有调用close(),因为缓冲对象调用了这两个方法,前面说了,缓冲对象调用的flush()和close()其实就是关闭被包装在其内部的流对象。关闭流的先后顺序也要注意,如果流之间有依赖关系,则被依赖的流要后关闭。readLine()方法原理:其实缓冲区中的该方法,用的还是与缓冲区关联的流对象的read方法,只不过,每一次读到一个字符先不进行具体操作,先进行临时存储,当读到回车标记时,将临时容器中存储的数据一次性返回。我们可以根据这个原理来自己编写一个缓冲区对象。
清单5,编写一个自己的bufferedreader代码
public class MyBufferedReader { private Reader reader; public MyBufferedReader(Reader reader){ this.reader=reader; } public String readLine() throws IOException{ StringBuilder sb=new StringBuilder(); int ch=0; while((ch=reader.read())!=-1){ if(ch=='\r'){//空格则继续 continue; }else if(ch=='\n'){//每次返回一行 return sb.toString(); }else{ sb.append((char)ch); } } return sb.toString(); } public void close() throws IOException{ //缓冲对象的关闭方法其实就是调用流本身的close() reader.close(); } }
测试时把清单4的BufferedReader对象替换成MyBufferedReader对象即可。
清单6,测试mybufferedreader代码
private static void test4(){ MyBufferedReader bufr=null; BufferedWriter bufw=null; try { bufr=new MyBufferedReader(new FileReader("D:/a.txt")); bufw=new BufferedWriter(new FileWriter("D:/b.txt")); String line=null; while((line=bufr.readLine())!=null){ bufw.write(line);//每次将一行写入缓冲区 bufw.flush();//刷新到目的地 } } catch (IOException e) { e.printStackTrace(); }finally{ try { if(bufw!=null){ bufw.close(); } if(bufr!=null){ bufr.close(); } } catch (IOException e1) { e1.printStackTrace(); } } }
其实我们自己写的这个缓存对象就是对Reader对象进行了功能的增强,Reader对象每次只能返回一个字符,而增强了功能之后该类就可以每次返回一行字符,也就是设计模式中所说的装饰模式。
Java 流的概述及操作(转)的更多相关文章
- java流与文件的操作 文件加密
课后作业 1,源代码 import java.io.*; import java.nio.file.*; import java.nio.file.attribute.BasicFileAttribu ...
- -1-4 java io java流 常用流 分类 File类 文件 字节流 字符流 缓冲流 内存操作流 合并序列流
File类 •文件和目录路径名的抽象表示形式 构造方法 •public File(String pathname) •public File(String parent,Stringchild) ...
- Java知多少(66)输入输出(IO)和流的概述
输入输出(I/O)是指程序与外部设备或其他计算机进行交互的操作.几乎所有的程序都具有输入与输出操作,如从键盘上读取数据,从本地或网络上的文件读取数据或写入数据等.通过输入和输出操作可以从外界接收信息, ...
- Java 字符流实现文件读写操作(FileReader-FileWriter)
Java 字符流实现文件读写操作(FileReader-FileWriter) 备注:字符流效率高,但是没有字节流底层 字节流地址:http://pengyan5945.iteye.com/blog/ ...
- java 学习笔记之 流、文件的操作
ava 学习笔记之 流.文件的操作 对于一些基础的知识,这里不再过多的解释, 简单的文件查询过滤操作 package com.wfu.ch08; import java.io.File; import ...
- Java基础-IO流对象之内存操作流(ByteArrayOutputStream与ByteArrayInputStream)
Java基础-IO流对象之内存操作流(ByteArrayOutputStream与ByteArrayInputStream) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.内存 ...
- java I/O流类概述
java I/O流类概述
- 使用Socket&反射&Java流操作进行方法的远程调用(模拟RPC远程调用)
写在前面 阅读本文首先得具备基本的Socket.反射.Java流操作的基本API使用知识:否则本文你可能看不懂... 服务端的端口监听 进行远程调用,那就必须得有客户端和服务端.服务端负责提供服务,客 ...
- java 流操作对文件的分割和合并的实例详解_java - JAVA
文章来源:嗨学网 敏而好学论坛www.piaodoo.com 欢迎大家相互学习 java 流操作对文件的分割和合并的实例详解 学习文件的输入输出流,自己做一个小的示例,对文件进行分割和合并. 下面是代 ...
随机推荐
- Lua-C交互函数
lua_gettable(lua_State * , tableIndex) //获取表的在key位置的值 过程:tableIndex为表在栈的位置,例:-2为第二个位置 , 此时会弹(出)栈作为参数 ...
- 对 const char* const &a 的理解
定义中用到&是独立引用. 比如: char i; char &a=i; 表示a是i的一个单独引用. 当有i='a'时,也有a='a'; 当有a='c'时,也有i='c'; 两个变量的标 ...
- Android自定义View基础
自定义控件, 视频教程 http://www.jikexueyuan.com/course/1748.html 1. 编写自定义view 2. 加入逻辑线程 3. 提取和封装自定义view 4. 利用 ...
- TatukGIS - GisDefs - CheckDir 函数
函数名称 CheckDir 所在单元 GisDefs 函数原型 function CheckDir(const _path: String): Boolean; 函数说明 如果 _path ...
- SVN允许修改日志
1.强制写日志在每次提交的时候写明提交的目的是一个很好的习惯,Subversion默认没有提供,但是可以通过钩子实现:将下面的代码存为放到版本库的hooks目录下即可,当你不写日志提交的话就会报告错误 ...
- openerp 产品图片的批量写入
Write a short python script which loops over the image files, encode with base64 and write to OpenER ...
- Python Tutorial 学习(七)--Input and Output
7. Input and Output Python里面有多种方式展示程序的输出.或是用便于人阅读的方式打印出来,或是存储到文件中以便将来使用.... 本章将对这些方法予以讨论. 两种将其他类型的值转 ...
- lsmod
http://blog.csdn.net/yuan892173701/article/details/8960607 抽空写下
- PHP与MYSQL配合完成IP的存取
如何存储IP 程序设计要在功能实现的基础上最大限度的优化性能.而数据库设计是程序设计中不可忽略的重要部分,巧存IP地址可以一定程度提升性能. 利用函数算法处理 MySQL没有直接提供IP类型字段,但有 ...
- 转: Linux C 动态内存分配 malloc及相关内容 .
一.malloc()和free()的基本概念以及基本用法: 1.函数原型及说明: void *malloc(long NumBytes):该函数分配了NumBytes个字节,并返回了指向这块内存的指针 ...