谈谈Java中的新的IO特性
1.昨天和大家讲了有关Java的IO特性,现在应该可以读写文本文件和二进制文件了吧,但是在一些特殊情况下我们需要更快的IO操作,这就谈到我们今天要讲的java.nio.*包了;
2.实现更快的IO操作,我们通常使用的结构类似与操作系统执行IO的方式:通道和缓冲器;
我们不直接和通道打交道,我们是通过缓冲器和通道打交道的。所以通道要么从缓冲器获得数据,要么向缓冲器发送数据
3.唯一和通道直接打交道的缓冲器是ByteBuffer(可以查看Java的doc);
下面看示例程序,通过文件通道和ByteBuffer实现文件的读写和随机读写
package MyJava.Base;
import java.io.*;
import java.nio.*;
import java.nio.channels.*;
import java.nio.*;
import java.nio.channels.*;
public class ChannelDemo
{
public static void main(String[] args) throws IOException
{
//获得文件的通道,
FileChannel fc=new FileOutputStream("E://Java//JCreator2.5//加密解密.txt").getChannel();
//写入数据
fc.write(ByteBuffer.wrap("some text".getBytes()));
fc.close();
//随机读写
fc=new RandomAccessFile("E://Java//JCreator2.5//加密解密.txt","rw").getChannel();
fc.position(fc.size());
fc.write(ByteBuffer.wrap("some more text".getBytes()));
fc.close();
fc=new FileInputStream("E://Java//JCreator2.5//加密解密.txt").getChannel();
ByteBuffer buff=ByteBuffer.allocate(1024);
fc.read(buff);
buff.flip(); //告诉通道已经准备好了,可以读写了
while(buff.hasRemaining())
{
System.out.println((char)buff.get());
}
}
{
public static void main(String[] args) throws IOException
{
//获得文件的通道,
FileChannel fc=new FileOutputStream("E://Java//JCreator2.5//加密解密.txt").getChannel();
//写入数据
fc.write(ByteBuffer.wrap("some text".getBytes()));
fc.close();
//随机读写
fc=new RandomAccessFile("E://Java//JCreator2.5//加密解密.txt","rw").getChannel();
fc.position(fc.size());
fc.write(ByteBuffer.wrap("some more text".getBytes()));
fc.close();
fc=new FileInputStream("E://Java//JCreator2.5//加密解密.txt").getChannel();
ByteBuffer buff=ByteBuffer.allocate(1024);
fc.read(buff);
buff.flip(); //告诉通道已经准备好了,可以读写了
while(buff.hasRemaining())
{
System.out.println((char)buff.get());
}
}
}
4.通过连接两个通道实现文件的拷贝
package MyJava.Base;
import java.nio.*;
import java.io.*;
import java.nio.channels.*;
import java.io.*;
import java.nio.channels.*;
public class CopyByConnectChannelDemo
{
public static void main(String[] args) throws IOException
{
FileChannel srcfile=new FileInputStream("E://Java//JCreator2.5//加密解密.txt").getChannel();
FileChannel destfile=new FileOutputStream("E://Java//JCreator2.5//加密解密(Copy).txt").getChannel();
srcfile.transferTo(0,srcfile.size(),destfile);
}
}
5.ByteBuffer操作基本数据类型
ByteBuffer操作的是字节,那么如果我们想向ByteBuffer中插入或者提取基本类型数据时该怎么办呢,解决的方法是:
提取:使用ByteBuffer的getInt()或者getFloat()等方法;
插入:使用ByteBuffer的asCharBuffer().put()等方法;
6.视图缓冲器的作用
通过同一个ByteBuffer上建立不同的视图缓冲器
CharBuffer cb=((ByteBuffer)bb.rewind()).asCharBuffer();
7.存储器映射文件
(1).为什么需要存储器映射文件??
存储器映射文件允许我们创建和修改那些因为太大而不能放在内存的文件,有了存储器映射文件,我们就可以假定整个文件都是放在内存中的,而且可以完全把它当作非常大的数组来访问.
(2).使用存储器映射文件可以更加显著地加快速度
(3)存储器映射文件使用MappedByteBuffer;
(4).示例:创建一个大文件通道,并对文件进行读写
package MyJava.Base;
import java.io.*;
import java.nio.*;
import java.nio.channels.*;
import java.nio.*;
import java.nio.channels.*;
public class MappedFileDemo
{
static int length=0xffffff;
public static void main(String[] args) throws IOException
{
MappedByteBuffer out=new RandomAccessFile("E://Java//JCreator2.5//sweetcoffee.exe","rw").getChannel().map(FileChannel.MapMode.READ_WRITE ,0,length);
for(int i=0;i<length;i++)
out.put((byte)'x');
}
{
static int length=0xffffff;
public static void main(String[] args) throws IOException
{
MappedByteBuffer out=new RandomAccessFile("E://Java//JCreator2.5//sweetcoffee.exe","rw").getChannel().map(FileChannel.MapMode.READ_WRITE ,0,length);
for(int i=0;i<length;i++)
out.put((byte)'x');
}
}