FileChannel不能设置为非阻塞模式。 它总是以阻止模式运行。
在您可以使用FileChannel之前,您必须打开它。 您无法直接打开FileChannel。 您需要通过InputStream,OutputStream或RandomAccessFile获取FileChannel。 以下是通过RandomAccessFile打开FileChannel的方法:
要从FileChannel读取数据,您可以调用read()方法之一。 这是一个例子:
首先分配一个缓冲区。 从FileChannel读取的数据被读入缓冲区。
二来调用FileChannel.read()方法。 此方法将数据从FileChannel读入缓冲区。 read()方法返回的int指示缓冲区中有多少个字节。 如果返回-1,则到达文件结尾。
将文件写入FileChannel是使用FileChannel.write()方法完成的,该方法采用Buffer作为参数。 这是一个例子:
注意在一个while循环中如何调用FileChannel.write()方法。 不能保证write()方法写入FileChannel的字节数。 因此,我们重复write()调用,直到Buffer中已经没有尚未写入通道的字节。
用完FileChannel后必须将其关闭。如:
有时可能需要在FileChannel的某个特定位置进行数据的读/写 *** 作。可以通过调用position()方法获取FileChannel的当前位置。
也可以通过调用position(long pos)方法设置FileChannel的当前位置。
如果将位置设置在文件结束符之后,然后试图从文件通道中读取数据,读方法将返回-1 —— 文件结束标志。
如果将位置设置在文件结束符之后,然后向通道中写数据,文件将撑大到当前位置并写入数据。这可能导致“文件空洞”,磁盘上物理文件中写入的数据间有空隙。
FileChannel实例的size()方法将返回该实例所关联文件的大小
可以使用FileChannel.truncate()方法截取一个文件。截取文件时,文件将中指定长度后面的部分将被删除。如:
FileChannel.force()方法将通道里尚未写入磁盘的数据强制写到磁盘上。出于性能方面的考虑, *** 作系统会将数据缓存在内存中,所以无法保证写入到FileChannel里的数据一定会即时写到磁盘上。要保证这一点,需要调用force()方法。
force()方法有一个boolean类型的参数,指明是否同时将文件元数据(权限信息等)写到磁盘上。
下面的例子同时将文件数据和元数据强制写到磁盘上:
给你一个简单的文件通道示例public class ChannelTest{
public static void main(String[] args){
File f=new File("XXXX.txt")//后面要创建一个读通道,所以文件最好事先已存在
FileInputStream fis=new FileInputStream(f)
FileChannel fc=fis.getChannel()//获得文件读入流通道
//现在可以用通道读取文件了
ByteBuffer buf=ByteBuffer.allocate(1024)//定义一个缓冲区,用来存放通道读取的数据
byte[] b=new byte[buf.capacity()]//定义一个数组用来存放缓冲里面的数据,好像不能直接访问缓冲区里面的数据
long n=0//判断标志
while(true){
n=fc.read(b)
if(n==-1)//-1说明已经到达文件流末尾
break
//输出读取的数据
buf.flip()
buf.get(b,0,buf.limit())//将缓冲区里的实际数据长度赋给数组
System.out.println(new String(b,0,b.length))
//必须清空缓冲区,否则读不进来
buf.clear()
}
//关闭流
fis.close()
fc.close()
}
}
在Java编程中,复制文件的方法有很多,而且经常要用到。我以前一直是缓冲输入输出流来实现的(绝大多数人都是如此),近来在研究JDK文档时发现,用文件通道(FileChannel)来实现文件复制竟然比用老方法快了近三分之一。下面我就来介绍一下如何用文件通道来实现文件复制,以及在效率上的对比用文件通道的方式来进行文件复制
/**
* 使用文件通道的方式复制文件
*
* @param s
*源文件
* @param t
*复制到的新文件
*/
public void fileChannelCopy(File s, File t) {
FileInputStream fi = null
FileOutputStream fo = null
FileChannel in = null
FileChannel out = null
try {
fi = new FileInputStream(s)
fo = new FileOutputStream(t)
in = fi.getChannel()//得到对应的文件通道
out = fo.getChannel()//得到对应的文件通道
in.transferTo(0, in.size(), out)//连接两个通道,并且从in通道读取,然后写入out通道
} catch (IOException e) {
e.printStackTrace()
} finally {
try {
fi.close()
in.close()
fo.close()
out.close()
} catch (IOException e) {
e.printStackTrace()
}
}
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)