文件操作——I/O

什么是I/O?

I/O计算机硬件视角下,非内存数据的读写动作,当下以文件(硬盘)IO为主

什么是文件?

1.侠义上文件(File):一种抽象的概念,表示硬盘中的数据块。变量,对象是堆内存中数据的抽象,文件则是对硬盘中数据的抽象

2.文件系统:对硬盘上的数据——文件进行直接组织和管理的一个模块。

3.逻辑上,一个文件是由两块数据组成的,包括该文件内容本身和该文件的元数据

4.文件系统是按照树形结构,进行组织的

5.目录/文件夹也是一类文件,只是一种特殊的文件 

文件 = 文本文件 or 二进制文件

文件 = 普通文件(文件)or 文件夹文件

文件的位置/路径

用于定位到是哪个文件的(用树的视角看,就是树上的一个结点)

1.绝对路径:从一棵树的根结点出发,到达对应节点的完整路径

    路径存在,并不代表文件一定存在,路径是描述树中一个可能的位置,但并不代表这个位置一定有结点

2.相对路径

    相对"我们"当前所在位置/当前工作目录 而言的路径

硬盘存储 VS 内存存储

硬盘存储:非易失性存储,可以在断电的情况进行数据的保存,但是读写速度低于内存

硬盘中的数据更适合:

1.块状读取(block) VS 字节读取(char)

块状读取:可以一次性读写很多的数据,并且是连续的读写

字节读取:频繁的进行读写,可以是随机的

对数据的认识

我们一般都把数据抽象成数字(什么进制,什么宽度);

如何解释这些数据 -> 文本,像素,波……

文本数据:可以按照一定的字符集编码解释成文本的数据

二进制数据:不可以直接解释成文本

代码实现

java提供了一个java.io.File类

File 对象就是对 文件的管理数据的抽象。描述文件

 java程序运行时的我们身处何处?

File类的常见方法

   文件中是被OS进行了权限管理的,以用户单位进行管理,  普通用户/管理员用户

新建普通文件

前提:路径对应的文件原来是不存在的

 

 通过运行程序,我们可以在指定路径下创建文件,但是不可以再次创建,因为文件已经存在了。

 新建目录文件

public class Main4 {
    public static void main(String[] args) {
        File f1 = new File("../a");
        f1.mkdir();
        File f2 = new File("../a/b");
        f2.mkdir();
        File f3 = new File("../a/b/c");
        f3.mkdir();
        File f4 = new File("../a/b/c/d");
        System.out.println(f4.mkdir());

    }
}

 

 在创建目录是我们可以用mkdir()一级一级创建,不可以一次性创建好。

public class Main4 {
    public static void main(String[] args) {
        File f4 = new File("../a/b/c/d");
        System.out.println(f4.mkdirs());

    }
}

  当然我们可以用mkdirs()一次性创建好,但是前提是在创建前文件目录都不存在

文件的删除   

​编辑  deleteOnExit()只是标记要删除,真正删除的时候是JVM进程退出的时候,不是放入回收站而是彻底删除。文件正常删除,目录删除要求目录必须已经是空目录了。

小Tips:windows上的回收站只是一个特殊的目录,回收站的删除其实不是删除,只是把文件移动到对应的目录。

public class Main5 {
    public static void main(String[] args) {
        File file = new File("../a");
        System.out.println(file.delete());
    }
}

 我们可以看到在执行完上述代码后a文件夹被删除了。

文件的重命名

 通过重命名的形式把一个文件移动成另一个结点

public class Main6 {
    public static void main(String[] args) {
        File src = new File("../a/a.txt");
        File dest = new File("../b/a.txt");
        System.out.println(src.renameTo(dest));
    }
}

 在代码执行后我们可以看到a文件夹下的a.txt成功移到了b文件夹下。

列出目录下的所有文件

 给定起始目录,遍历这个目录下的所有文件,本质上就是进行一棵树的遍历,对所有结点做条件判断

public class Main7 {
    static List<String> target = new ArrayList<>();
    private static void traversal(File dir) throws IOException {
        File[] files = dir.listFiles(); //该目录下的所有孩子
        if (files == null){
            return;
        }
        for (File file  :  files){
            if (file.isDirectory()){
                traversal(file);
            }else if (file.isFile()){
                if (file.getName().endsWith(".exe"));
                target.add(file.getCanonicalPath());
            }
        }
    }
    public static void main(String[] args) throws IOException {
        File startDir = new File("D:/");
        traversal(startDir);
        for (String path : target){
            System.out.println(path);
        }
    }
}

我们可以通过以上代码检索到所需要的文件

普通文件的内容读取

InputStream 概述 

                     方法

修饰符及返回值类型方法签名说明
int
read()
读取一个字节的数据,返回 -1 代表已经完全读完了
int
read(byte[] b)
最多读取 b.length 字节的数据到 b 中,返回实际读到的数 量;-1 代表以及读完了
int
read(byte[] b,
int off, int len)
最多读取 len - off 字节的数据到 b 中,放在从 off 开始,返
回实际读到的数量; -1 代表以及读完了
void
close()
关闭字节流

说明:

InputStream 只是一个抽象类,要使用还需要具体的实现类。关于 InputStream 的实现类有很多,基本可以认为不同的输入设备都可以对应一个 InputStream 类,我们现在只关心从文件中读取,所以使用FileInputStream

FileInputStream 概述
签名说明
FileInputStream(File file)
利用 File 构造文件输入流
FileInputStream(String name)
利用文件路径构造文件输入流

                                               

// 需要先在项目目录下准备好一个 hello.txt 的文件,里面填充 "Hello" 的内容
public class Main {
public static void main(String[] args) throws IOException {
try (InputStream is = new FileInputStream("hello.txt")) {
byte[] buf = new byte[1024];
int len;
while (true) {
len = is.read(buf);
if (len == -1) {
// 代表文件已经全部读完
break;
}
for (int i = 0; i < len; i++) {
System.out.printf("%c", buf[i]);
}
}
}
}
}

  中文内容的读取           

​
// 需要先在项目目录下准备好一个 hello.txt 的文件,里面填充 "你好中国" 的内容
public class Main {
public static void main(String[] args) throws IOException {
try (InputStream is = new FileInputStream("hello.txt")) {
byte[] buf = new byte[1024];
int len;
while (true) {
len = is.read(buf);
if (len == -1) {
// 代表文件已经全部读完
break;
}
// 每次使用 3 字节进行 utf-8 解码,得到中文字符
// 利用 String 中的构造方法完成
// 这个方法了解下即可,不是通用的解决办法
for (int i = 0; i < len; i += 3) {
String s = new String(buf, i, 3, "UTF-8");
System.out.printf("%s", s);
}
}
}
}
}

​

 OutputStream 概述

修饰符及返回值类型方法签名说明
void
write(int b)
写入要给字节的数据
void
write(byte[] b)
b 这个字符数组中的数据全部写入 os
int
write(byte[] b, int off, int len)
b 这个字符数组中从 off 开始的数据写入 os 中,一共写 len
void
close()
关闭字节流
void
flush()
重要:我们知道 I/O 的速度是很慢的,所以,大多的 OutputStream 为了减少设备操作的次数,在写数据的时候都会将数据先暂时写入内存的
一个指定区域里,直到该区域满了或者其他指定条件时才真正将数据写 入设备中,这个区域一般称为缓冲区。但造成一个结果,就是我们写的
数据,很可能会遗留一部分在缓冲区中。需要在最后或者合适的位置,调用 flush (刷新)操作,将数据刷到设备中。
利用 OutputStreamWriter 进行字符写入
import java.io.*;
public class Main {
public static void main(String[] args) throws IOException {
try (OutputStream os = new FileOutputStream("output.txt")) {
os.write('H');
os.write('e');
os.write('l');
os.write('l');
os.write('o');
// 不要忘记 flush
os.flush();
}
}
}
import java.io.*;
public class Main {
public static void main(String[] args) throws IOException {
try (OutputStream os = new FileOutputStream("output.txt")) {
byte[] b = new byte[] {
(byte)'G', (byte)'o', (byte)'o', (byte)'d'
};
os.write(b);
// 不要忘记 flush
os.flush();
}
}
}
利用 PrintWriter 找到我们熟悉的方法
上述,我们其实已经完成输出工作,但总是有所不方便,我们接来下将 OutputStream 处理下,使用 PrintWriter 类来完成输出,因为 PrintWriter 类中提供了我们熟悉的 print/println/printf 方法
import java.io.*;
public class Main {
public static void main(String[] args) throws IOException {
   try (OutputStream os = new FileOutputStream("output.txt")) {
     try (OutputStreamWriter osWriter = new OutputStreamWriter(os, "UTF-
         8")) {
         try (PrintWriter writer = new PrintWriter(osWriter)) {
          writer.println("我是第一行");
          writer.print("我的第二行\r\n");
          writer.printf("%d: 我的第三行\r\n", 1 + 1);
          writer.flush();
         }
    }
  }
 }
}

​

让我们一起写一个小程序

1.文本替换工具:“你好” -> “我好” (读 替换 写)

public class ReplaceTool {
    public static void main(String[] args) throws IOException {
        //输入的对象
        try (InputStream is = new FileInputStream("src.txt")){
            try (Scanner scanner = new Scanner(is,"UTF-8")){
                //输出的对象
                try(OutputStream os = new FileOutputStream("dest.text")) {
                    try (Writer writer = new OutputStreamWriter(os,"UTF-8")) {
                        try (PrintWriter printWriter = new PrintWriter(writer)) {
                            while (scanner.hasNextLine()) {
                                String line = scanner.nextLine();
                                String replace = line.replace("你好", "我好");
                                printWriter.println(replace);
                            }
                            printWriter.flush();
                        }
                    }

                }
            }
        }
    }
}

我们可以看到替换成功 !

2.文件的复制(任意类型的文件都可以)InputStream + OutputStream

public class CopyFile {
    public static void main(String[] args) throws IOException {
        //源文件:src.png
        //目标文件:dest.png
        try (InputStream is = new FileInputStream("src.png")){
            try (OutputStream os = new FileOutputStream("dest.png" )){
                byte[] buf = new byte[1024];
                while (true){
                    int n = is.read(buf);
                    if (n == -1){
                        break;
                    }
                    os.write(buf,0,n);
                }
                os.flush();
            }
        }
    }
}

 我们可以看到文件被复制成功!

3.文件夹的复制 (树的复制)

递归的方式,对源目录进行遍历(树的遍历)

对于每个结点

if是目录,则手动创建新的目录(名字一样)

if是文件,按照刚才的方式进行文件的复制(名字一样)

                                                                                                                                                                                                                              

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值