首页 > 其他分享 > RandomAccessFile 讲解与使用

RandomAccessFile 讲解与使用

时间:2023-04-24 16:47:05浏览次数:47  
标签:raf 文件 String 讲解 RandomAccessFile File 使用 new

RandomAccessFile的简介

RandomAccessFile可以实现对文件数据的随机读取。

RandomAccessFile类包含了一个记录指针,用以标识当前读写处的位置,当程序新创建一个RandomAccessFile对象时,该对象的文件记录指针位于文件头(也就是0处),当读/写了n个字节后,文件记录指针将会向后移动n个字节。除此之外,RandomAccessFile可以自由的移动记录指针,即可以向前移动,也可以向后移动。RandomAccessFile包含了以下两个方法来操作文件的记录指针.

  1. long getFilePointer(); 返回文件记录指针的当前位置
  2. void seek(long pos); 将文件记录指针定位到pos位置

RandomAccessFile即可以读文件,也可以写,所以它即包含了完全类似于InputStream的3个read()方法,其用法和InputStream的3个read()方法完全一样;也包含了完全类似于OutputStream的3个write()方法,其用法和OutputStream的3个Writer()方法完全一样。除此之外,RandomAccessFile还包含了一系类的readXXX()和writeXXX()方法来完成输入和输出。

RandomAccessFile有两个构造器,其实这两个构造器基本相同,只是指定文件的形式不同而已,一个使用String参数来指定文件名,一个使用File参数来指定文件本身。除此之外,创建RandomAccessFile对象还需要指定一个mode参数。该参数指定RandomAccessFile的访问模式,有以下4个值:

  1. “r” 以只读方式来打开指定文件夹。如果试图对该RandomAccessFile执行写入方法,都将抛出IOException异常。
  2. “rw” 以读,写方式打开指定文件。如果该文件尚不存在,则试图创建该文件。
  3. “rws” 以读,写方式打开指定文件。相对于”rw” 模式,还要求对文件内容或元数据的每个更新都同步写入到底层设备。
  4. “rwd” 以读,写方式打开指定文件。相对于”rw” 模式,还要求对文件内容每个更新都同步写入到底层设备。
     

我们平常创建流对象关联文件,开始读文件或者写文件都是从头开始的,不能从中间开始,如果是开多线程下载一个文件我们之前学过的FileWriter或者FileReader等等都无法完成,而当前介绍的RandomAccessFile他就可以解决这个问题,因为它可以指定位置读,指定位置写的一个类,通常开发过程中,多用于多线程下载一个大文件.


RandomAccessFile特点

RandomAccessFile是java Io体系中功能最丰富的文件内容访问类。即可以读取文件内容,也可以向文件中写入内容。但是和其他输入/输入流不同的是,程序可以直接跳到文件的任意位置来读写数据。
  因为RandomAccessFile可以自由访问文件的任意位置,所以如果我们希望只访问文件的部分内容,那就可以使用RandomAccessFile类。
  与OutputStearm,Writer等输出流不同的是,RandomAccessFile类允许自由定位文件记录指针,所以RandomAccessFile可以不从文件开始的地方进行输出,所以RandomAccessFile可以向已存在的文件后追加内容。则应该使用RandomAccessFile。

常用方法

使用RandomAccessFile写入数据

  1.   public static void main(String[] args) throws IOException {
  2.   File file = new File("text1.txt");
  3.   RandomAccessFile raf = new RandomAccessFile(file, "rw");//读写模式
  4.   //保证长度一致,采用空格填充
  5.   String names[] = new String[] {"zhangsan","lisi ","wangwu "};
  6.   int ages[] = new int[] {30,20,16};
  7.   for(int x = 0 ;x<names.length;x++) {
  8.   raf.write(names[x].getBytes());
  9.   raf.writeInt(ages[x]);
  10.   }
  11.   raf.close();
  12.   }

使用RandomAccessFile读取数据

  1.   public static void main(String[] args) throws IOException {
  2.   File file = new File("text1.txt");
  3.   RandomAccessFile raf = new RandomAccessFile(file, "rw");//读写模式
  4.   {//读取王五的数据,字符串8位,数字4位z
  5.   raf.skipBytes(24);
  6.   byte[] data = new byte[8];
  7.   int len = raf.read(data);
  8.   System.out.println("姓名:"+new String(data,0,len).trim() +
  9.   ",年龄:"+raf.readInt());
  10.   }
  11.   {//读取李四的数据,字符串8位,数字4位z
  12.   raf.seek(12);
  13.   byte[] data = new byte[8];
  14.   int len = raf.read(data);
  15.   System.out.println("姓名:"+new String(data,0,len).trim() +
  16.   ",年龄:"+raf.readInt());
  17.   }
  18.   {//读取张三的数据,字符串8位,数字4位z
  19.   raf.seek(0);
  20.   byte[] data = new byte[8];
  21.   int len = raf.read(data);
  22.   System.out.println("姓名:"+new String(data,0,len).trim() +
  23.   ",年龄:"+raf.readInt());
  24.   }
  25.   raf.close();
  26.   }

使用RandomAccessFile实现从指定位置读取文件的功能

  1.   public static void main(String[] args)throws IOException {
  2.   String filePath="test.txt";
  3.   RandomAccessFile raf=null;
  4.   File file=null;
  5.   try {
  6.   file=new File(filePath);
  7.   raf=new RandomAccessFile(file,"r");
  8.   // 获取 RandomAccessFile对象文件指针的位置,初始位置为0
  9.   System.out.print("输入内容:"+raf.getFilePointer());
  10.   //移动文件记录指针的位置
  11.   raf.seek(1000);
  12.    
  13.   byte[] b=new byte[1024];
  14.   int hasRead=0;
  15.   //循环读取文件
  16.   while((hasRead=raf.read(b))>0){
  17.   //输出文件读取的内容
  18.   System.out.print(new String(b,0,hasRead));
  19.   }
  20.   }catch (IOException e){
  21.   e.printStackTrace();
  22.   }finally {
  23.   raf.close();
  24.   }
  25.   }

使用RandomAccessFile实现向文件中追加内容的功能

  1.   public class RandomAccessFileTest2 {
  2.   public static void main(String[] args)throws IOException {
  3.   String filePath="test.txt";
  4.   RandomAccessFile raf=null;
  5.   File file=null;
  6.   try {
  7.   file=new File(filePath);
  8.   // 以读写的方式打开一个RandomAccessFile对象
  9.   raf=new RandomAccessFile(file,"rw");
  10.   //将记录指针移动到该文件的最后
  11.   raf.seek(raf.length());
  12.   //向文件末尾追加内容
  13.   raf.writeChars("这是追加内容。。");
  14.   }catch (IOException e){
  15.   e.printStackTrace();
  16.   }finally {
  17.   raf.close();
  18.   }
  19.   }

使用RandomAccessFile实现向文件指定位置插入内容的功能

注:RandomAccessFile不能向文件的指定位置插入内容,如果直接将文件记录指针移动到中间某位置后开始输出,则新输出的内容会覆盖文件原有的内容,如果需要向指定位置插入内容,程序需要先把插入点后面的内容写入缓存区,等把需要插入的数据写入到文件后,再将缓存区的内容追加到文件后面。

  1.   /**
  2.   * 插入文件指定位置的指定内容
  3.   * @param filePath 文件路径
  4.   * @param pos 插入文件的指定位置
  5.   * @param insertContent 插入文件中的内容
  6.   * @throws IOException
  7.   */
  8.   public static void insert(String filePath,long pos,String insertContent)throws IOException{
  9.   RandomAccessFile raf=null;
  10.   File tmp=File.createTempFile("tmp",null);
  11.   tmp.deleteOnExit();
  12.   try {
  13.   // 以读写的方式打开一个RandomAccessFile对象
  14.   raf = new RandomAccessFile(new File(filePath), "rw");
  15.   //创建一个临时文件来保存插入点后的数据
  16.   FileOutputStream fileOutputStream = new FileOutputStream(tmp);
  17.   FileInputStream fileInputStream = new FileInputStream(tmp);
  18.   //把文件记录指针定位到pos位置
  19.   raf.seek(pos);
  20.   raf.seek(pos);
  21.   //------下面代码将插入点后的内容读入临时文件中保存-----
  22.   byte[] bbuf = new byte[64];
  23.   //用于保存实际读取的字节数据
  24.   int hasRead = 0;
  25.   //使用循环读取插入点后的数据
  26.   while ((hasRead = raf.read(bbuf)) != -1) {
  27.   //将读取的内容写入临时文件
  28.   fileOutputStream.write(bbuf, 0, hasRead);
  29.   }
  30.   //-----下面代码用于插入内容 -----
  31.   //把文件记录指针重新定位到pos位置
  32.   raf.seek(pos);
  33.   //追加需要插入的内容
  34.   raf.write(insertContent.getBytes());
  35.   //追加临时文件中的内容
  36.   while ((hasRead = fileInputStream.read(bbuf)) != -1) {
  37.   //将读取的内容写入临时文件
  38.   raf.write(bbuf, 0, hasRead);
  39.   }
  40.   }catch (Exception e){
  41.   throw e;
  42.   }
  43.   }
  44.   public static void main(String[] args)throws IOException {
  45.   String filePath="test.txt";
  46.   insert(filePath,1000,"插入指定位置指定内容");
  47.   }

RandomAccessFile 文件下载

首先创建一个DownLoadThread的类继承Thread,

  1.   public class DownLoadThread extends Thread {
  2.   ​
  3.   private long start;
  4.   private File src;
  5.   private long total;
  6.   private File desc;
  7.   ​
  8.   /**
  9.   *
  10.   * @param start
  11.   * 开始下载的位置
  12.   * @param src
  13.   * 要下载的文件
  14.   * @param desc
  15.   * 要下载的目的地
  16.   * @param total
  17.   * 要下载的总量
  18.   */
  19.   public DownLoadThread(long start, File src, File desc, long total) {
  20.   this.start = start;
  21.   this.src = src;
  22.   this.desc = desc;
  23.   this.total = total;
  24.   }
  25.   ​
  26.   @Override
  27.   public void run() {
  28.   try {
  29.   // 创建输入流关联源,因为要指定位置读和写,所以我们需要用随机访问流
  30.   RandomAccessFile src = new RandomAccessFile(this.src, "rw");
  31.   RandomAccessFile desc = new RandomAccessFile(this.desc, "rw");
  32.   ​
  33.   // 源和目的都要从start开始
  34.   src.seek(start);
  35.   desc.seek(start);
  36.   // 开始读写
  37.   byte[] arr = new byte[1024];
  38.   int len;
  39.   long count = 0;
  40.   while ((len = src.read(arr)) != -1) {
  41.   //分三种情况
  42.   if (len + count > total) {
  43.   //1.当读取的时候操作自己该线程的下载总量的时候,需要改变len
  44.   len = (int) (total - count);
  45.   desc.write(arr, 0, len);
  46.   //证明该线程下载任务已经完毕,结束读写操作
  47.   break;
  48.   } else if (len + count < total) {
  49.   //2.证明还没有到下载总量,直接将内容写入
  50.   desc.write(arr, 0, len);
  51.   //并且使计数器任务累加
  52.   count += arr.length;
  53.   } else {
  54.   //3.证明改好到下载总量
  55.   desc.write(arr, 0, len);
  56.   //结束读写
  57.   break;
  58.   }
  59.   }
  60.   src.close();
  61.   desc.close();
  62.   ​
  63.   } catch (Exception e) {
  64.   e.printStackTrace();
  65.   }
  66.   }
  67.   }

文件的测试

  1.   public static void main(String[] args) {
  2.   //关联源
  3.   File src = new File("a.txt");
  4.   //关联目的
  5.   File desc = new File("b.txt");
  6.   ​
  7.   //获取源的总大小
  8.   long length = src.length();
  9.   // 开两条线程,并分配下载任务
  10.   new DownLoadThread(0, src, desc, length / 2).start();
  11.   new DownLoadThread(length / 2 , src, desc, length - (length / 2)).start();
  12.   }

总结

从以上分析可以看出RandomAccessFile最大两个特点:

1.可以指定位置开始操作;

2.既可以读,也可以写;

所以,我们但凡遇到不能从中间开始读取的时候,可以使用RandomAccessFile这个类,比如:多线程下载是最常用的应该场景

 

标签:raf,文件,String,讲解,RandomAccessFile,File,使用,new
From: https://www.cnblogs.com/tiancai/p/17350024.html

相关文章

  • WebSphere Message Broker -JavaCompute组件的使用
      IBMWebSphereMessageBrokerJavaCompute节点的使用. importjava.util.List;importcom.ibm.broker.javacompute.MbJavaComputeNode;importcom.ibm.broker.plugin.*;publicclassSub_FFN_JavaComputeextendsMbJavaComputeNode{ privatefinalArticleCreator......
  • GitLab-CI/CD使用
    一、 二、   GitLabCI/CD是GitLabContinuousIntegration(Gitlab持续集成)的简称。只要在项目仓库的根目录添加.gitlab-ci.yml文件,并且配置了gitlabRunner(运行器),那么每一次push或者合并请求(MergeRequest)都会触发CIPipeline。  1)GitLabRunner可以运行在GNU/Lin......
  • substring截取使用
    privatefinalstaticintentNameLength=32;publicstaticvoidmain(String[]args){Stringentname="王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王王......
  • 一个简单的 rust 项目 使用 bevy 引擎 复刻 Flappy Bird 小游戏
    Rust+Bevy实现的FlappyBird游戏简介一个使用bevy引擎复刻的FlappyBird经典小游戏。通过该项目我们可以学到:bevy的自定义组件,自定义插件,自定义资源,sprite的旋转,sprite的移动,spritesheet动画的定义使用,状态管理,等内容…简单介绍一下包含的内容:游戏状态管理Me......
  • spring boot 过滤器、拦截器的区别和使用
    区别:一、过滤器与拦截器的对比1.使用范围不同:过滤器是基于Servlet,而拦截器是基于Spring的,Spring框架底层又离不开Servlet,所以过滤器也能在Spring体系中使用。2.使用资源不同:拦截器有Spring的支持,能够方便的向容器中注册对象和使用对象,但是过滤器就不能。3.使用场景不同:灵活性上......
  • Go语言介绍、Go开发环境搭建、第一个helloworld、变量命名规范、变量的定义和使用
    目录1Go语言介绍2Go开发环境搭建3第一个helloworld4变量命名规范5变量的定义和使用1Go语言介绍#Go语言介绍Go即Golang,是Google公司2009年11月正式对外公开的一门编程语言Go是【静态强类型】语言,是区别于解析型语言的编译型语言(静态:类型固定强类型:不同类型不允许直接......
  • 第六讲 weBASE IDE 的使用和智能合约的开发
    01智能合约概述智能合约产生价值的最基本前提是有一个强有力的底层介质用于储存,让其不可被物理破坏。然而,智能合约的本体是一份代码,非常容易被慧改,如何为其提供强力的存储介质就成了问题。这正好是区块链擅长解决的——通过比特币的实践,证明了区块链可以在分布式环境下让电......
  • vue前端使用nexus配置npm私有仓库
    当我们运行前端项目的时候,常常在解决依赖的时候会加上一个参数npminstall--registry=https://registry.npm.taobao.org将源指定为淘宝的源,以期让速度加快起来,事实上这种的确能够让速度变快,但是长久来看,如果想真正的快速敏捷开发部署,搭建企业内部的私服,则会让速度更上一个台阶。......
  • 越来越多的半导体企业放弃使用FTP,这对行业来说意味着什么?
    FTP作为世界第一款文件传输协议,曾在世界范围内被各行业领域广泛应用。作为近些年关注度最高、发展最快速、最为知识密集型行业之一的半导体行业,其中的集成电路设计制造企业和代工生产企业,在日常运营中,都不可避免涉及到文件和数据的交换,这些文件数据通常体量较大、数量较多,交换频次......
  • 使用curl请求接口
    1、基本用法curl-X[request][options][URL] -X选项指定了在与远程服务器通信时将使用哪种HTTP请求方法。   常用的请求方法[request]:GER,POST,DELETE,PUT,PATCH 2、常用参数-v参数输出通信的整个过程,用于调试。-b参数用来向服务器发送Cookie。 ......