Java I/O基础

  字节流和字符流的区别,字节流一次读取一个字节,字符流一次读取的是一个Unicode码,读取了2个字节。

可以以文本编辑器打开的可以使用字符流读取,否则用字符流读取可能就会出错。图像文件就需要用字节流读取,不能用字符流操作。

字节流的基类是InputStream和OutputStream,字符流的基类是Reader和Writer。

读取并复制图片的代码,读取到缓存,一边读一边写。 ps:文件的相对路径要写什么看看System.getProperty("user.dir")就知道了,这个是基础路径。

public class FileIOTest {
	public static void main(String [] args) throws IOException{

		int BYTE_SIZE = 1;
		int SAVE_SIZE = 1024;
		int i = 0;
		byte[] buff = new byte[BYTE_SIZE]; // 每次读的缓存
		byte[] save = new byte[SAVE_SIZE]; // 保存前缓存

		BufferedInputStream bf = new BufferedInputStream(new FileInputStream(new File("src/pattern/decorator/a.jpeg")));
		FileOutputStream  fs = new FileOutputStream(new File("src/pattern/decorator/b.jpeg"));
		while (bf.read(buff) != -1) { // 一个字节一个字节读
		    save[i] = buff[0];
		    if (i == SAVE_SIZE - 1) { // 达到保存长度时开始保存
		     fs.write(save, 0, SAVE_SIZE);
		     save = new byte[SAVE_SIZE];
		     i = 0;
		    } else {
		     i++;
		   }
		}
		// 最后这段如果没达到保存长度,需要把前面的保存下来
		if (i > 0) {
		   fs.write(save, 0, i);
		}
		fs.close();
		bf.close();
	}
}

   流的关闭应该放到finally中,上面的是不太合适的。

下面是关于大文件读取的一些数据,原文为:

http://aronlulu.iteye.com/blog/1018370

读取文件大小:1.45G 
第一种,OldIO:

Java代码  

  1. public static void oldIOReadFile() throws IOException{
  2. BufferedReader br = new BufferedReader(new FileReader("G://lily_947.txt"));
  3. PrintWriter pw = new PrintWriter("G://oldIO.tmp");
  4. char[] c = new char[100*1024*1024];
  5. for(;;){
  6. if(br.read(c)!=-1){
  7. pw.print(c);
  8. }else{
  9. break;
  10. }
  11. }
  12. pw.close();
  13. br.close();
  14. }

耗时70.79s

第二种,newIO:

Java代码  

  1. public static void newIOReadFile() throws IOException{
  2. FileChannel read = new RandomAccessFile("G://lily_947.txt","r").getChannel();
  3. FileChannel writer = new RandomAccessFile("G://newIO.tmp","rw").getChannel();
  4. ByteBuffer bb = ByteBuffer.allocate(200*1024*1024);
  5. while(read.read(bb)!=-1){
  6. bb.flip();
  7. writer.write(bb);
  8. bb.clear();
  9. }
  10. read.close();
  11. writer.close();
  12. }

耗时47.24s

第三种,RandomAccessFile:

Java代码  

  1. public static void randomReadFile() throws IOException{
  2. RandomAccessFile read = new RandomAccessFile("G://lily_947.txt","r");
  3. RandomAccessFile writer = new RandomAccessFile("G://random.tmp","rw");
  4. byte[] b = new byte[200*1024*1024];
  5. while(read.read(b)!=-1){
  6. writer.write(b);
  7. }
  8. writer.close();
  9. read.close();
  10. }

耗时46.65

第四种,MappedByteBuffer:

Java代码  

  1. public static void mappedBuffer() throws IOException{
  2. FileChannel read = new FileInputStream("G://lily_947.txt").getChannel();
  3. FileChannel writer = new RandomAccessFile("G://buffer.tmp","rw").getChannel();
  4. long i = 0;
  5. long size = read.size()/30;
  6. ByteBuffer bb,cc = null;
  7. while(i<read.size()&&(read.size()-i)>size){
  8. bb = read.map(FileChannel.MapMode.READ_ONLY, i, size);
  9. cc = writer.map(FileChannel.MapMode.READ_WRITE, i, size);
  10. cc.put(bb);
  11. i+=size;
  12. bb.clear();
  13. cc.clear();
  14. }
  15. bb = read.map(FileChannel.MapMode.READ_ONLY, i, read.size()-i);
  16. cc.put(bb);
  17. bb.clear();
  18. cc.clear();
  19. read.close();
  20. writer.close();
  21. }

耗时:36

前三种读法对应的资源占用图如下: 
相对于最后一种内存直接映射方式前面的测试其实无意义,基本秒杀。。。。。 
对于很大的文件直接分块映射时内存会不够,这是因为MappedByteBuffer未被释放造成的,sun未提供直接回收MappedByteBuffer区域的方法,这个时候有两种方法解决,第一种比较愚笨的:

Java代码  

  1. System.gc();
  2. System.runFinalization();
  3. try {
  4. Thread.sleep(3000);
  5. } catch (InterruptedException e) {
  6. e.printStackTrace();
  7. }

第二种网上找来的,利用反射调用clean方法:

Java代码  

  1. public static void unmap(final MappedByteBuffer buffer) {
  2. if (buffer == null) {
  3. return;
  4. }
  5. AccessController.doPrivileged(new PrivilegedAction<Object>() {
  6. public Object run() {
  7. try {
  8. Method getCleanerMethod = buffer.getClass().getMethod("cleaner", new Class[0]);
  9. if (getCleanerMethod != null) {
  10. getCleanerMethod.setAccessible(true);
  11. Object cleaner = getCleanerMethod.invoke(buffer, new Object[0]);
  12. Method cleanMethod = cleaner.getClass().getMethod("clean", new Class[0]);
  13. if (cleanMethod != null) {
  14. cleanMethod.invoke(cleaner, new Object[0]);
  15. }
  16. }
  17. } catch (Exception e) {
  18. e.printStackTrace();
  19. }
  20. return null;
  21. }
  22. });
  23. }

以上两种方法感觉都别扭,还有就是可以自己分割成物理文件再循环调用,这个也不太美观。 
速度也会减慢好多。

时间: 2024-11-03 14:29:08

Java I/O基础的相关文章

java String 类 基础笔记

字符串是一个特殊的对象. 字符串一旦初始化就不可以被改变. String s = "abc";//存放于字符串常量池,产生1个对象 String s1=new String("abc");//堆内存中new创建了一个String对象,产生2个对象 String类中的equals比较字符串中的内容. 常用方法: 一:获取 1.获取字符串中字符的个数(长度):length();方法. 2.根据位置获取字符:charAt(int index); 3.根据字符获取在字符串中

2.2JAVA基础复习——JAVA语言的基础组成运算符和语句

JAVA语言的基础组成有: 1.关键字:被赋予特殊含义的单词. 2.标识符:用来标识的符号. 3.注释:用来注释说明程序的文字. 4.常量和变量:内存存储区域的表示. 5.运算符:程序中用来运算的符号. 6.语句:程序中常用的一些语句. 7.函数:也叫做方法,用来做一些特定的动作. 8.数组:用来存储多个数据的集合. JAVA中的运算符 1.算术运算符:用来进行一些数据算法的符号 算术运算符分为单目运算符.双目运算符.三目运算符. 单目运算符有:+(取正)-(取负)++(自增)--(自减)代码如

Java网络编程基础(六)— 基于TCP的NIO简单聊天系统

在Java网络编程基础(四)中提到了基于Socket的TCP/IP简单聊天系统实现了一个多客户端之间护法消息的简单聊天系统.其服务端采用了多线程来处理多个客户端的消息发送,并转发给目的用户.但是由于它是基于Socket的,因此是阻塞的. 本节我们将通过SocketChannel和ServerSocketChannel来实现同样的功能. 1.客户端输入消息的格式 username:msg    username表示要发送的的用户名,msg为发送内容,以冒号分割 2.实现思路 实现思路与Java网络

Java语言的基础知识

第三章 1.在java源文件编辑器中,选择某个成员变量,然后按住shift+alt+j,Eclipse会自动添加JavaDoc文档注释结构,如果选择的是方法,还会自动添加参数名称. 2.Java语言规定标示符是由任意的字母.下划线.美元符号和数字组成,并且第一个字符不能使数字,标示符不能使java中的保留关键字. 3.在Java语言中允许使用汉字或其他语言文字作为变量名,如int 年龄 =21;在程序运行时不会报错,但建议尽量不要使用这些语言作为变量. 4.java用关键字final来声明常量,

[Java 05 OO] (基础篇) 《Java开发实战经典》

p5OO 第五章 面向对象 (基础篇) Notes (1), Constructor / this / String   String str1 = "hello"; 解释 : 是把一个在堆内存空间的使用权给了 str1 对象.   String str2 = "hello"; str1 == str2 是 true   String 字符串的内容不可改变 (2), Java 常用的内存区域    1), 栈内存空间    2), 堆内存空间    3), 全局数据

JAVA学习(五):Java面向对象编程基础

Java面向对象编程基础 面向对象(Object oriented programming,OOP)技术是一种强有力的软件开发方法,它採用数据抽象与信息隐藏技术,来使软件开发简单化,以达到代码重用的目的. 1.OOP的3个特性(封装.继承和多态性) 封装是类的基础.指把类的相关实现细节隐藏起来,在类中将数据和实现操作的代码集中起来放在对象的内部.调用这些类时仅仅需直接使用类预留的接口就能够了. 继承提供了子类自己主动拥有父类数据结构和方法的机制.它表示类之间的一种关系. 多态指使一个对象被看成还

Java并发(基础知识)—— Executor框架及线程池

在Java并发(基础知识)—— 创建.运行以及停止一个线程中讲解了两种创建线程的方式:直接继承Thread类以及实现Runnable接口并赋给Thread,这两种创建线程的方式在线程比较少的时候是没有问题的,但是当需要创建大量线程时就会出现问题,因为这种使用方法把线程创建语句随意地散落在代码中,无法统一管理线程,我们将无法管理创建线程的数量,而过量的线程创建将直接使系统崩溃. 从高内聚角度讲,我们应该创建一个统一的创建以及运行接口,为我们管理这些线程,这个统一的创建与运行接口就是JDK 5的Ex

Java语言的基础知识4

第五章(数组) 1.在Java中可以将数组看做是一个对象虽然基本数据类型不是对象但有基本数据类型组成的数组是对象. 2.对于二维数组求第二维就用array[0].length, array.length就是默认的是第一维的长度. 3.foreach并不是一个新的语法它是for的循环的格式化主要执行遍历功能的循环,example: int arry ={1,2,3,4,5}; for(int i :array){ system.out.println(): } 4.数组元素定义完以后可通过Arra

黑马程序员——Java I/O基础知识之I/O流

I/O流基础知识--字节流和字符流 文件存储在硬盘中,是以二进制表示的,只有内存中才能形成字符.数据的来源可以有硬盘,内存,控制台,网络,Java把数据从一个地方转到另一个地方的现象称为流,用InputStream和OutputStream接口来表示,这两个流里面的都是以字节为单位的,后来加入了Reader和Writer,里面操作的是字符,是两个字节为单位的. 字节流 字节流将数据写入文件 try { File file =new File("d:" +File .separator+

2.1JAVA基础复习——JAVA语言的基础组成注释和常量变量

/** 这是 JAVA中独有的多行注释 */ JAVA语言的基础组成有: 1.关键字:被赋予特殊含义的单词. 2.标识符:用来标识的符号. 3.注释:用来注释说明程序的文字. 4.常量和变量:内存存储区域的表示. 5.运算符:程序中用来运算的符号. 6.语句:程序中常用的一些语句. 7.函数:也叫做方法,用来做一些特定的动作. 8.数组:用来存储多个数据的集合. JAVA中的注释: 注释还可以用来缩小程序错误的范围,方便查找错误. // :表示单行注释. //这是一个单行注释 /**/:表示多行