File类和IO流

File类和IO流

文章目录

    • File类和IO流
    • @[TOC](文章目录)
    • 前言
    • 一、java.io.File类&IO流原理及流的分类
      • 1.1 File类及其API
      • 1.2 IO流原理及分类
    • 二、节点流的介绍(字符/字节)
      • 2.1 Reader\Writer--字符IO`抽象基类`
      • 2.2 FileReader\FileWriter--字符IO`节点流`
      • 2.3 InputStream\OutputStream--字节IO`抽象基类`
      • 2.4 FileInputStream/FileOutputStream--字节IO`节点流`
    • 三、处理流的介绍
      • 3.1 缓冲流Buffered--字符流/字节流
      • 3.2 转换流InputStreamReader/OutputStreamWriter(字符-字节)
      • 3.3 数据流&对象流--字节
      • 3.4 补充:序列化&反序列化机制
    • 四、其他流的使用
      • 4.1 标准输入、输出流
      • 4.2 打印流PrintStream/PrintWriter
      • 4.3 Scanner类
      • 4.4 apache-common包的使用
    • 五、企业真题

前言

File及各种流都在java.io包下

知识补充:绝对路径&相对路径

  • 绝对路径:从盘符开始的路径,这是一个完整的路径。
  • 相对路径:相对于项目目录的路径,这是一个便捷的路径,开发中经常使用。
    • IDEA中,main中的文件的相对路径,是相对于"当前工程"
    • IDEA中,单元测试方法中的文件的相对路径,是相对于"当前module"

一、java.io.File类&IO流原理及流的分类

1.1 File类及其API

1、File类的理解:——流的端点

  • File类位于java.io包下,本章中涉及到的相关流也都声明在java.io包下。
  • File类的一个对象,对应与操作系统下的一个文件或一个文件目录(或文件夹)
  • File类中声明了新建、删除、获取名称、重命名、删除等操作并没有涉及到文件内容的读写操作——需要使用IO流
  • File类的对象,通常是作为io流操作的文件的端点出现

2、File的内部构造器&API
(1)构造器——创建File对象

  • public File(String pathname) :以pathname为路径创建File对象,可以是绝对路径或者相对路径,如果pathname是相对路径,则默认的当前路径在系统属性user.dir中存储。
  • public File(String parent, String child) :以parent为父路径,child为子路径创建File对象。
  • public File(File parent, String child) :根据一个父File对象和子文件路径创建File对象

(2)常用方法

  • 获取文件和目录的基本信息:
    • public String getName() :获取名称
    • public String getPath() :获取路径,创建File对象提供的路径
    • public String getAbsolutePath():获取绝对路径(完整)
    • public File getAbsoluteFile():获取绝对路径表示的文件
    • public String getParent():获取上层文件目录路径。若无,返回null
    • public long length() :获取文件长度(即:字节数)。不能获取目录的长度–递归可以。
    • public long lastModified() :获取最后一次的修改时间,毫秒值
File file1 = new File("abc.txt");
String str1 = file.getParent();//相对路径的上层文件目录为空
String str1 = file.getAbsolutePath().getParent();//先得到绝对路径,再根据绝对路径得到当前文件的上层文件路径
File file2 = new File("src/abc.txt");
String str2 = file.getParent();//上层目录为src
  • 列出文件的下一级:
    • public String[] list() :表示该File目录中的所有子文件或目录。——可以添加过滤器FilenameFilter(),需重写方法public boolean accept(File dir, String name)
    • public File[] listFiles() :表示该File目录中的所有的子文件或目录。
  • File类的重命名功能:
    • public boolean renameTo(File dest):把文件重命名为指定的文件路径。
  • 判断功能的方法:
    • public boolean exists()、isDirectory()、isFile() 、canRead()、canWrite()、isHidden():此File表示的文件或目录是否实际存在、是否为目录、是否为文件、可读、可写、隐藏。
  • 创建或者删除:
    • public boolean createNewFile()、mkdir()、mkdirs()、delete():创建文件,如果已存在则不创建返回false;创建文件目录,如果已存在/上层目录不存在则不创建;创建文件,如果上层不存在则一起创建;删除文件或文件夹(不走回收站,如果是文件目录则里面不能包含文件/文件目录)

判断指定目录下是否有后缀名为.jpg的文件。如果有,就输出该文件名称

public class FindJPGFileTest {
	//方法1:
	@Test
	public void test1(){
		File srcFile = new File("d:\\code");
		
		String[] fileNames = srcFile.list();
		for(String fileName : fileNames){
			if(fileName.endsWith(".jpg")){
				System.out.println(fileName);
			}
		}
	}
    //方法2:
	@Test
	public void test2(){
		File srcFile = new File("d:\\code");
		
		File[] listFiles = srcFile.listFiles();
		for(File file : listFiles){
			if(file.getName().endsWith(".jpg")){
				System.out.println(file.getAbsolutePath());
			}
		}
	}
    //方法3:
	/*
	 * File类提供了两个文件过滤器方法
	 * public String[] list(FilenameFilter filter)
	 * public File[] listFiles(FileFilter filter)

	 */
	@Test
	public void test3(){
		File srcFile = new File("d:\\code");
		
		File[] subFiles = srcFile.listFiles(new FilenameFilter() {
			
			@Override
			public boolean accept(File dir, String name) {
				return name.endsWith(".jpg");
			}
		});
		
		for(File file : subFiles){
			System.out.println(file.getAbsolutePath());
		}
	}
	
}

遍历指定目录所有文件名称,包括子文件目录中的文件。
拓展1:并计算指定目录占用空间的大小
拓展2:删除指定文件目录及其下的所有文件

public class ListFilesTest {
	//练习3:(方式1)
    public static void printSubFile(File dir) {
        // 打印目录的子文件
        File[] subfiles = dir.listFiles();
        for (File f : subfiles) {
            if (f.isDirectory()) {// 文件目录
                printSubFile(f);
            } else {// 文件
                System.out.println(f.getAbsolutePath());//输出绝对路径
            }
        }
    }
    // //练习3:(方式2)
    public void listAllSubFiles(File file) {
        if (file.isFile()) {
            System.out.println(file);
        } else {
            File[] all = file.listFiles();
            // 如果all[i]是文件,直接打印
            // 如果all[i]是目录,接着再获取它的下一级
            for (File f : all) {
                listAllSubFiles(f);// 递归调用:自己调用自己就叫递归
            }
        }
    }
    @Test
    public void testListAllFiles(){
        // 1.创建目录对象
        File dir = new File("E:\\teach\\01_javaSE\\_尚硅谷Java编程语言\\3_软件");

        // 2.打印目录的子文件
        printSubFile(dir);
    }

    // 拓展1:求指定目录所在空间的大小
    public long getDirectorySize(File file) {
        // file是文件,那么直接返回file.length()
        // file是目录,把它的下一级的所有file大小加起来就是它的总大小
        long size = 0;
        if (file.isFile()) {
            size = file.length();
        } else {
            File[] all = file.listFiles();// 获取file的下一级
            // 累加all[i]的大小
            for (File f : all) {
                size += getDirectorySize(f);// f的大小;
            }
        }
        return size;
    }

    // 拓展2:删除指定的目录
    public void deleteDirectory(File file) {
        // 如果file是文件,直接delete
        // 如果file是目录,先把它的下一级干掉,然后删除自己
        if (file.isDirectory()) {
            File[] all = file.listFiles();
            // 循环删除的是file的下一级
            for (File f : all) {// f代表file的每一个下级
                deleteDirectory(f);
            }
        }
        // 删除自己
        file.delete();
    }
}

1.2 IO流原理及分类

1、Java IO流原理:
Java程序中,对于数据的输入/输出操作以“流(stream)” 的方式进行,可以看做是一种数据的流动。
I/O流中的I/O是Input/Output的缩写, I/O技术是非常实用的技术,用于处理设备之间的数据传输。如读/写文件,网络通讯等。

  • 输入input:读取外部数据(磁盘、光盘等存储设备的数据)到程序(内存)中。
  • 输出output:将程序(内存)数据输出到磁盘、光盘等存储设备中。

IO流的场景
2、流的分类

  • 按数据的流向不同分为:输入流输出流
    • 输入流 :把数据从其他设备上读取到内存中的流。
    • 输出流 :把数据从内存 中写出到其他设备上的流。
  • 按操作数据单位的不同分为:字节流(8bit)字符流(16bit)
    • 字节流 :以字节为单位,读写数据的流。
    • 字符流 :以字符为单位,读写数据的流。
  • 根据IO流的角色不同分为:节点流处理流
    • 节点流:直接从数据源或目的地读写数据
    • 处理流:不直接连接到数据源或目的地,而是“连接”在已存在的流(节点流或处理流)之上,通过对数据的处理为程序提供更为强大的读写功能。
      IO流的分类

3、流的API
流的API理解:字符流可以是输入/输出流,也可以是节点流/处理流,字节流同样也是。
节点流可以是字符流/字节流,处理流也是

  • 4个抽象基类:以其父类名作为子类名后缀
抽象基类抽象基类节点流缓冲流(处理流的一种)转换流对象流
字节输入InputStreamFileInputStreamBufferedInputStreamObjectInputStream
字节输出OutputStreamFileOutputStreamBufferedOutputStreamObjecOutputStream
字符输入ReaderFileReaderBufferedReaderInputStreamReader
字符输出WriterFileWriterBufferedWriterOutputStreamWriter

上述处理流连接每一行对应的节点流/处理流。

  • 常用的节点流:
    • 文件流: FileInputStream、FileOutputStrean、FileReader、FileWriter
    • 字节/字符数组流: ByteArrayInputStream、ByteArrayOutputStream、CharArrayReader、CharArrayWriter
    • 对数组进行处理的节点流(对应的不再是文件,而是内存中的一个数组)。
  • 常用处理流:
    • 缓冲流:BufferedInputStream、BufferedOutputStream、BufferedReader、BufferedWriter
      • 作用:增加缓冲功能,避免频繁读写硬盘,进而提升读写效率。
    • 转换流:InputStreamReader、OutputStreamReader
      • 作用:实现字节流和字符流之间的转换。
    • 对象流:ObjectInputStream、ObjectOutputStream
      • 作用:提供直接读写Java对象功能
        在Java IO流的分类

二、节点流的介绍(字符/字节)

1、字符流说明:Java提供一些字符流类,以字符为单位读写数据,专门用于处理文本文件。不能操作图片,视频等非文本文件。
常见的文本文件有如下的格式:.txt、.java、.c、.cpp、.py等。注意:.doc、.xls、.ppt这些都不是文本文件。
2、字节流说明:非文本文件,必须使用字节流。如图片

FileReader、FileWriter本质上调用的还是FileInputStream、FileOutputStream字节流

2.1 Reader\Writer–字符IO抽象基类

1、java.io.Reader抽象类:用于读取字符流的所有类的父类,可以读取字符信息到内存中。定义了字符输入流的基本共性功能方法。

  • public int read(): 从输入流读取一个字符。 虽然读取了一个字符,但是会自动提升为int类型。返回该字符的Unicode编码值。如果已经到达流末尾了,则返回-1。
  • public int read(char[] cbuf): 从输入流中读取一些字符,并将它们存储到字符数组 cbuf中 。每次最多读取cbuf.length个字符。返回实际读取的字符个数。如果已经到达流末尾,没有数据可读,则返回-1。
  • public int read(char[] cbuf,int off,int len):从输入流中读取一些字符,并将它们存储到字符数组 cbuf中,从cbuf[off]开始的位置存储。每次最多读取len个字符。返回实际读取的字符个数。如果已经到达流末尾,没有数据可读,则返回-1。
  • public void close() :关闭此流并释放与此流相关联的任何系统资源。 释放系统资源,否则会造成内存泄漏。

2、java.io.Writer 抽象类:用于写出字符流的所有类的超类,将指定的字符信息写出到目的地。定义了字节输出流的基本共性功能方法。

  • public void write(int c) :写出单个字符。
  • public void write(char[] cbuf) :写出字符数组。
  • public void write(char[] cbuf, int off, int len) :写出字符数组的某一部分。off:数组的开始索引;len:写出的字符个数。
  • public void write(String str) :写出字符串。
  • public void write(String str, int off, int len) :写出字符串的某一部分。off:字符串的开始索引;len:写出的字符个数。
  • public void flush() :刷新该流的缓冲。
  • public void close() :关闭此流。

2.2 FileReader\FileWriter–字符IO节点流

FileReader:文件不存在读取不了
FileWriter:文件不存在会自动创建一个写入,如果已经存在-追加?覆盖?——可以设置

1、执行步骤:

  • 第1步:创建读取或写出的File类的对象
  • 第2步:创建输入流或输出流
  • 第3步:具体的读入或写出的过程。
    • 读入:read(char[] cbuffer)//造小车 / read()//一个一个读
    • 写出:write(string str) / write(char[] cbuffer,0,len)
  • 第四步:关闭流资源,避免内存泄漏

2、注意点

  • 因为涉及到流资源的关闭操作,所以出现异常的话,需要使用try-catch-finally的方式来处理异常
  • 对于输入流来讲,要求File类的对象对应的物理磁盘上的文件必须存在。否则,会报FileNotFoundException
  • 对于输出流来讲,File类的对象对应的物理磁盘上的文件可以不存在。
    • 如果此文件不存在,则在输出的过程中,会自动创建此文件,并写出数据到此文件中。
    • 如果此文件存在,使用 FileWriter(File file)或 FileWriter(File file,false):输出数据过程中,会新建同名的文件对现有的文件进行覆盖。FileWriter(File file,true):输出数据过程中,会在现有的文件的末尾追加写出内容

3、Java.io.FileReader & java.io.FileWriter的API

  • java.io.FileReader 类用于读取字符文件,构造时使用系统默认的字符编码和默认字节缓冲区。
    • FileReader(File file): 创建一个新的 FileReader ,给定要读取的File对象。
    • FileReader(String fileName): 创建一个新的 FileReader ,给定要读取的文件的名称。
  • java.io.FileWriter:写出字符到文件,构造时使用系统默认的字符编码和默认字节缓冲区
    • FileWriter(File file): 创建一个新的 FileWriter,给定要读取的File对象。
    • FileWriter(String fileName): 创建一个新的 FileWriter,给定要读取的文件的名称。
    • FileWriter(File file,boolean append): 创建一个新的 FileWriter,指明是否在现有文件末尾追加内容。

4、flush(刷新)
内置缓冲区,FileWriter只有关闭输出流,才可将写字符写出到文件中。flush()方法可以实现既不关闭流,又可以写出数据。

  • flush() :刷新缓冲区,流对象可以继续使用。
  • close():先刷新缓冲区,然后通知系统释放资源。流对象不可以再被使用了。

读取hello.txt文件中的字符数据,并显示在控制台上——小车是每次读取/输出覆盖,最后每次读取/输出指定当前轮次的长度,避免最后覆盖不了
写入字符数据

public class FileReaderWriterTest {
//实现方式3:调用read(char[] cbuf),每次从文件中读取多个字符
    @Test
    public void test3() {
        FileReader fr = null;//避免文件不存在得到的是nul对象
        try {
            //1. 创建File类的对象,对应着物理磁盘上的某个文件
            File file = new File("hello.txt");
            //2. 创建FileReader流对象,将File类的对象作为参数传递到FileReader的构造器中
            fr = new FileReader(file);
            //3. 通过相关流的方法,读取文件中的数据
            char[] cbuf = new char[5];//造小车,每次读取5个字符
            /*
             * read(char[] cbuf) : 每次将文件中的数据读入到cbuf数组中,并返回读入到数组中的
             * 字符的个数。
             * */
            int len; //记录每次读入的字符的个数
            while ((len = fr.read(cbuf)) != -1) {
                //处理char[]数组即可
                //错误:最后不足5个字符的小车没有完全覆盖前5个
//                for(int i = 0;i < cbuf.length;i++){
//                    System.out.print(cbuf[i]);
//                }
                //错误:同样的道理
//                String str = new String(cbuf);
//                System.out.print(str);
                //正确:长度使用当前循环轮次read的长度
//                for(int i = 0;i < len;i++){
//                    System.out.print(cbuf[i]);
//                }
                //正确:和上面是一样的
                String str = new String(cbuf, 0, len);
                System.out.print(str);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //4. 关闭相关的流资源,避免出现内存泄漏
            try {
                if (fr != null)
                    fr.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    @Test
    public void test04(){
        FileWriter fw = null;
        try {
            //1. 创建File的对象
            File file = new File("personinfo.txt");
            //2. 创建FileWriter的对象,将File对象作为参数传递到FileWriter的构造器中
            //如果输出的文件已存在,则会对现有的文件进行覆盖
            fw = new FileWriter(file);
//            fw = new FileWriter(file,false);
            //如果输出的文件已存在,则会在现有的文件末尾写入数据
//            fw = new FileWriter(file,true);

            //3. 调用相关的方法,实现数据的写出操作
            //write(String str) / write(char[] cbuf)
            fw.write("I love you,");
            fw.write("you love him.");
            fw.write("so sad".toCharArray());
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //4. 关闭资源,避免内存泄漏
            try {
                if (fw != null)
                    fw.close();
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
        }
    }
}

2.3 InputStream\OutputStream–字节IO抽象基类

1、java.io.InputStream 抽象类:字节输入流的所有类的超类,读取字节信息到内存中。它定义了字节输入流的基本共性功能方法。

  • public int read(): 从输入流读取一个字节。返回读取的字节值。虽然读取了一个字节,但是会自动提升为int类型。如果已经到达流末尾,没有数据可读,则返回-1。
  • public int read(byte[] b): 从输入流中读取一些字节数,并将它们存储到字节数组 b中 。每次最多读取b.length个字节。返回实际读取的字节个数。如果已经到达流末尾,没有数据可读,则返回-1。
  • public int read(byte[] b,int off,int len):从输入流中读取一些字节数,并将它们存储到字节数组 b中,从b[off]开始存储,每次最多读取len个字节 。返回实际读取的字节个数。如果已经到达流末尾,没有数据可读,则返回-1。
  • public void close() :关闭此输入流并释放与此流相关联的任何系统资源。

2、java.io.OutputStream 抽象类:字节输出流的所有类的超类,将指定的字节信息写出到目的地。它定义了字节输出流的基本共性功能方法。

  • public void write(int b) :将指定的字节输出流。虽然参数为int类型四个字节,但是只会保留一个字节的信息写出。
  • public void write(byte[] b):将 b.length字节从指定的字节数组写入此输出流。
  • public void write(byte[] b, int off, int len) :从指定的字节数组写入 len字节,从偏移量 off开始输出到此输出流。
  • public void flush() :刷新此输出流并强制任何缓冲的输出字节被写出。
  • public void close():关闭此输出流并释放与此流相关联的任何系统资源。

2.4 FileInputStream/FileOutputStream–字节IO节点流

1、执行步骤

  • 第1步:创建读取或写出的File类的对象
  • 第2步:创建输入流或输出流
  • 第3步:具体的读入或写出的过程。读入:read(byte[] buffer)写出:write(byte[] buffer,0,len)
  • 第4步:关闭流资源,避免内存泄漏

2、注意点:和FileReader、FileWriter基础之上

  • 字符流只能处理文本文件,非文本文件不行
    字节流通常用来处理非文本文件,但是如果涉及到文本文件的复制也可以。

3、输入/输出节点流的API:

  • java.io.FileInputStream :文件输入流,从文件中读取字节。
    • FileInputStream(File file)`: 通过打开与实际文件的连接来创建一个 FileInputStream ,该文件由文件系统中的 File对象 file命名。
    • FileInputStream(String name): 通过打开与实际文件的连接来创建一个 FileInputStream ,该文件由文件系统中的路径名 name命名。
  • java.io.FileOutputStream :文件输出流,用于将数据写出到文件。
    • public FileOutputStream(File file):创建文件输出流,写出由指定的 File对象表示的文件。
    • public FileOutputStream(String name): 创建文件输出流,指定的名称为写出文件。
    • public FileOutputStream(File file, boolean append): 创建文件输出流,指明是否在现有文件末尾追加内容。

使用FileInputStream\FileOutputStream,实现对文件txt的复制
问题:如果读取/写入编码不一样,可能存在乱码。下面这个复制没有问题,但是如果输出到控制台上就可能会出现乱码

public class FOSWrite {
	@Test
    public void test05() {
        FileInputStream fis = null;
        FileOutputStream fos = null;
        try {
            //1. 造文件-造流
            //复制图片:成功
//            fis = new FileInputStream(new File("pony.jpg"));
//            fos = new FileOutputStream(new File("pony_copy1.jpg"));

            //复制文本文件:成功
            fis = new FileInputStream(new File("hello.txt"));
            fos = new FileOutputStream(new File("hello1.txt"));

            //2. 复制操作(读、写)
            byte[] buffer = new byte[1024];
            int len;//每次读入到buffer中字节的个数
            while ((len = fis.read(buffer)) != -1) {
                fos.write(buffer, 0, len);
//                String str = new String(buffer,0,len);
//                System.out.print(str);
            }
            System.out.println("复制成功");
        } catch (IOException e) {
            throw new RuntimeException(e);
        } finally {
            //3. 关闭资源
            try {
                if (fos != null)
                    fos.close();
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
            try {
                if (fis != null)
                    fis.close();
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
        }
    }
}

三、处理流的介绍

3.1 缓冲流Buffered–字符流/字节流

1、缓冲流介绍:不直接连接到数据源或目的地,而是“连接”在已存在的流(节点流或处理流)之上,通过对数据的处理为程序提供更为强大的读写功能。
针对字符/字节有不同的分类处理方式。
提升文件读写的效率,减少与磁盘交互的次数。

2、缓冲流分类

  • 处理非文本文件的字节流:如:FileInputStream、FileOutputStream
    • BufferedInputStream:read(bytel] buffer)
    • BufferedOutputStream:write(bytel] buffer,0,len)
  • 处理文本文件的字符流:如:FileReader、FileWriter
    • BufferedReader:read(char[] cBuffer)、readLine()//返回字符串–每次读取一行文本中数据,返回的字符串不包含换行符
    • BufferedWriter:write(char[] cBuffer,0,len)、public void newLine()// 写一行行分隔符,由系统属性定义符号。

3、构造器:处理现有流(如:节点流,不是File对象!!!)

  • public BufferedInputStream(InputStream in) :创建一个 新的字节型的缓冲输入流。
  • public BufferedOutputStream(OutputStream out): 创建一个新的字节型的缓冲输出流。
  • public BufferedReader(Reader in) :创建一个 新的字符型的缓冲输入流。
  • public BufferedWriter(Writer out): 创建一个新的字符型的缓冲输出流。

4、实现步骤

  • 第1步:创建File的对象、流的对象(包括文件流、缓冲流)
  • 第2步:使用缓冲流实现 读取数据 或 写出数据的过程(重点)
    • 读取:int read(char[] cbuf / byte[] buffer):每次将数据读入到cbuf / buffer数组中,并返回读入到数组中
    • 写出:void write(string str) / write(char[] cbuf):将str或cbuf写出到文件中
      void write(byte[] buffer)将byte[]写出到文件中
  • 第3步:关闭资源。
    说明:处理流(外层流)的关闭会自动对内层流的关闭操作,因此只需要关闭外层流即可。

测试缓冲流的效率提升:查询API,缓冲流读写方法与基本的流是一致的,我们通过复制大文件(375MB),测试它的效率。
FileInputStream\FileOutputStream及BufferedInputStream\BufferedOuputStream

//方法1:使用FileInputStream\FileOutputStream实现非文本文件的复制
public void copyFileWithFileStream(String srcPath,String destPath){
    FileInputStream fis = null;
    FileOutputStream fos = null;
    try {
        //1. 造文件-造流
        fis = new FileInputStream(new File(srcPath));
        fos = new FileOutputStream(new File(destPath));

        //2. 复制操作(读、写)
        byte[] buffer = new byte[100];
        int len;//每次读入到buffer中字节的个数
        while ((len = fis.read(buffer)) != -1) {
            fos.write(buffer, 0, len);
        }
        System.out.println("复制成功");
    } catch (IOException e) {
        throw new RuntimeException(e);
    } finally {
        //3. 关闭资源
        try {
            if (fos != null)
                fos.close();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
        try {
            if (fis != null)
                fis.close();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }
}

@Test
public void test1(){
    String srcPath = "C:\\Users\\shkstart\\Desktop\\01-复习.mp4";
    String destPath = "C:\\Users\\shkstart\\Desktop\\01-复习2.mp4";

    long start = System.currentTimeMillis();

    copyFileWithFileStream(srcPath,destPath);

    long end = System.currentTimeMillis();

    System.out.println("花费的时间为:" + (end - start));//7677毫秒

}

//方法2:使用BufferedInputStream\BufferedOuputStream实现非文本文件的复制
public void copyFileWithBufferedStream(String srcPath,String destPath){
    FileInputStream fis = null;
    FileOutputStream fos = null;
    BufferedInputStream bis = null;
    BufferedOutputStream bos = null;
    try {
        //1. 造文件
        File srcFile = new File(srcPath);
        File destFile = new File(destPath);
        //2. 造流
        fis = new FileInputStream(srcFile);
        fos = new FileOutputStream(destFile);

        bis = new BufferedInputStream(fis);
        bos = new BufferedOutputStream(fos);

        //3. 读写操作
        int len;
        byte[] buffer = new byte[100];
        while ((len = bis.read(buffer)) != -1) {
            bos.write(buffer, 0, len);//先暂存,一次性write到文件中
        }
        System.out.println("复制成功");
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        //4. 关闭资源(如果有多个流,我们需要先关闭外面的流,再关闭内部的流)
        try {
            if (bos != null)
                bos.close();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
        try {
            if (bis != null)
                bis.close();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }

    }
}
@Test
public void test2(){
    String srcPath = "C:\\Users\\shkstart\\Desktop\\01-复习.mp4";
    String destPath = "C:\\Users\\shkstart\\Desktop\\01-复习2.mp4";

    long start = System.currentTimeMillis();

    copyFileWithBufferedStream(srcPath,destPath);

    long end = System.currentTimeMillis();

    System.out.println("花费的时间为:" + (end - start));//415毫秒

}

3.2 转换流InputStreamReader/OutputStreamWriter(字符-字节)

回顾编码知识:
字符编码:字符、字符串、字符数组——>字节、字节数组(看得懂——>看不懂)
字符节码:字节、字节数组——>字符、字符串、字符数组(看不懂——>看得懂)
存储在文件中:UTF-8英文1字节,中文3字节;GBK中文2字节;ASCII全部1字节,其他字符集都兼容。

在内存中的字符:
char占用2个字节。在内存中使用的字符集是Unidoce字符集——统一码,万国码
注意:这里所说的占用几个字节?
编码方式指的是存储在文件中时占用的字节。而对于内存中的占用几个字节和这个又不一样 —— 一个字符(char)占用2个字节,举例:‘a’、‘中’:都表示一个字符在内存中都是占用2个字节,当存储到文件中时就根据编码方式占用n个字节(所有编码方式兼容ASCII字符集,因此对于所有的英文字母占1个字节,对于中文采用GBK占用2个字节,采用UTF-8占用3个字节)。

问题:
idea项目编码UTF-8,windows创建的文本文件编码GBK,导入内存就会乱码
使用字节流读入文本文件,数据显示在控制台上(也有自己的编码方式),此时会出现乱码
——解码使用的字符集必须与当初编码时使用的字符集的相同。

字符-字节的转换
1、转换流介绍及其构造器

  • InputStreamReader:Reader的子类,输入型的字节流—指定解码方式—>输入型的字符流。指定字符集方式:指定名称、接受平台的默认字符集。
    • InputStreamReader(InputStream in):创建一个使用默认字符集的字符流。
    • InputStreamReader(InputStream in, String charsetName):创建一个指定字符集的字符流。
  • OutputStreamWriter:Writer的子类,输出型的字符流—指定编码方式—>输出型的字节流。
    • OutputStreamWriter(OutputStream in):创建一个使用默认字符集的字符流。
    • OutputStreamWriter(OutputStream in,String charsetName):创建一个指定字符集的字符流。

转换流InputStreamReader/OutputStreamWriter的使用示例
解码方式必须和文件编码方式一致,即采用字节方式读取,然后解码为字符。

//使用默认字符集——idea指定的
InputStreamReader isr1 = new InputStreamReader(new FileInputStream("in.txt"));
//使用指定字符集:字节流方式读取文件,使用GBK解码字节,得到输入型的字符流
InputStreamReader isr2 = new InputStreamReader(new FileInputStream("in.txt") , "GBK");

//使用默认字符集
OutputStreamWriter isr = new OutputStreamWriter(new FileOutputStream("out.txt"));
//使用指定的字符集
OutputStreamWriter isr2 = new OutputStreamWriter(new FileOutputStream("out.txt") , "GBK");

把当前module下的《康师傅的话.txt》字符编码为GBK,复制到电脑桌面目录下的《寄语.txt》,字符编码为UTF-8。
思路:以字节流的方式读取gbk格式文件,使用InputStreamReader指定解码格式gbk得到输入型的字符流,再使用OutputStreamWriter将字符流以指定编码uft-8格式转换为字节流存储文件.

/**
 * @author 尚硅谷-宋红康
 * @create 9:06
 */
public class InputStreamReaderDemo {

    @Test
    public void test() {
        InputStreamReader isr = null;
        OutputStreamWriter osw = null;
        try {
            isr = new InputStreamReader(new FileInputStream("康师傅的话.txt"),"gbk");

            osw = new OutputStreamWriter(new FileOutputStream("C:\\Users\\shkstart\\Desktop\\寄语.txt"),"utf-8");

            char[] cbuf = new char[1024];
            int len;
            while ((len = isr.read(cbuf)) != -1) {
                osw.write(cbuf, 0, len);
                osw.flush();
            }
            System.out.println("文件复制完成");
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            try {
                if (isr != null)
                    isr.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
            try {
                if (osw != null)
                    osw.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }

    }

}

3.3 数据流&对象流–字节

为什么需要数据流、对象流?将内存中定义的变量(包括基本数据类型或引用数据类型)保存在文件中。

Student stu = new Student("张三",23,89);

1、数据流、对象流介绍

  • 数据流:缺点:支持基本数据类型、字符串,不支持其它Java对象的类型。
    • DataInputStream:文件中保存的数据还原为内存中的基本数据类型、String类型的变量。
    • DataOutputStream:内存中的变量——》输出流中
  • 对象流:可以把Java中的对象写入到数据源中,也能把对象从数据源中还原回来。
    • ObjectInputStream:还原之前写出的基本数据类型的数据和对象进行读入操作,保存在内存中。——反序列化
    • ObjectOutputStream:将 Java 基本数据类型和对象写入字节输出流中。——序列化

2、对象流构造器&API
public ObjectInputStream(InputStream in) : 创建一个指定的ObjectInputStream。
public ObjectOutputStream(OutputStream out) : 创建一个指定的ObjectOutputStream。

ObjectInputStream中的方法:

public boolean readBoolean():读取一个 booleanpublic byte readByte():读取一个 8 位的字节
public short readShort():读取一个 16 位的 shortpublic char readChar():读取一个 16 位的 charpublic int readInt():读取一个 32 位的 intpublic long readLong():读取一个 64 位的 longpublic float readFloat():读取一个 32 位的 floatpublic double readDouble():读取一个 64 位的 doublepublic String readUTF():读取 UTF-8 修改版格式的 String
public void readObject(Object obj):读入一个obj对象
public void close() :关闭此输入流并释放与此流相关联的任何系统资源

ObjectOutputStream中的方法:

public void writeBoolean(boolean val):写出一个 boolean 值。
public void writeByte(int val):写出一个8位字节
public void writeShort(int val):写出一个16位的 shortpublic void writeChar(int val):写出一个16位的 charpublic void writeInt(int val):写出一个32位的 intpublic void writeLong(long val):写出一个64位的 longpublic void writeFloat(float val):写出一个32位的 float 值。
public void writeDouble(double val):写出一个64位的 doublepublic void writeUTF(String str):将表示长度信息的两个字节写入输出流,后跟字符串 s 中每个字符的 UTF-8 修改版表示形式。根据字符的值,将字符串 s 中每个字符转换成一个字节、两个字节或三个字节的字节组。注意,将 String 作为基本数据写入流中与将它作为 Object 写入流中明显不同。 如果 s 为 null,则抛出 NullPointerExceptionpublic void writeObject(Object obj):写出一个obj对象
public void close() :关闭此输出流并释放与此流相关联的任何系统资源

3.4 补充:序列化&反序列化机制

1、对象序列化机制:使用ObjectOutputStream实现,将内存中的java对象保存在文件中或者通过网络传输出去。
把内存中的Java对象转换成平台无关的二进制流,从而允许把这种二进制流持久地保存在磁盘上,或通过网络将这种二进制流传输到另一个网络节点。//当其它程序获取了这种二进制流,就可以恢复成原来的Java对象。

  • 序列化过程:用一个字节序列可以表示一个对象,该字节序列包含该对象的类型和对象中存储的属性等信息。字节序列写出到文件之后,相当于文件中持久保存了一个对象的信息。
    实现方式:用ObjectOutputStream类保存基本类型数据或对象的机制。方法为:public final void writeObject (Object obj) : 将指定的对象写出。
  • 反序列化过程:该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化。对象的数据、对象的类型和对象中存储的数据信息,都可以用来在内存中创建对象。
    实现方式:反序列化:用ObjectInputStream类读取基本类型数据或对象的机制。方法为:public final Object readObject ():读取一个对象。

序列化-反序列化过程
序列化-反序列化实现原理

2、为什么需要序列化?
序列化是 RMI(Remote Method Invoke、远程方法调用)过程的参数和返回值都必须实现的机制,而 RMI 是 JavaEE 的基础。因此序列化机制是 JavaEE 平台的基础。
序列化的好处,在于可将任何实现了Serializable接口的对象转化为字节数据,使其在保存和传输时可被还原。

Serializable:是一个标记接口,不实现此接口的类将不会使任何状态序列化或反序列化,会抛出NotSerializableException。

3、自定义类实现序列化机制需要满足的条件?

  • 类必须实现java.io.Serializable 接口,使得对象所属的类及其属性是可序列化,即对象支持序列化机制;
  • 要求自定义类声明一个全局常量:static final long serialVersionUID = 42234234L;——唯一标识当前的类。
  • 要求自定义类的各个属性也必须是可序列化的:
    • 基本数据类型的属性默认是可序列化的,无需序列化就是瞬态的,使用transient关键字修饰。
    • 引用类型的属性:实现Serializable 接口。
    • 静态(static)变量的值不会序列化。因为静态变量的值不属于某个对象。

4、反序列化失败问题

  • 如果不声明全局常量serialVersionUID ,系统会自动生成一个针对于当前类的serialVersionUID ,如果修改此类会导致serialVersionUID发生改变,进而导致反序列化时,出现InvalidClassException异常。
    解决方法:显式声明serialVersionUID,如果声明了serialVersionUID,即使在序列化完成之后修改了类导致类重新编译,则原来的数据也能正常反序列化,只是新增的字段值是默认值而已。
  • 对于JVM可以反序列化对象,它必须是能够找到class文件的类。如果找不到该类的class文件,则抛出一个 ClassNotFoundException 异常。

5、使用场景:
网络中的数据传输:要求对象可序列化。
一般情况不会传输自定义的类,而是json字符串(特殊格式的字符串)
只要涉及到对象就需要序列化、反序列化。

四、其他流的使用

4.1 标准输入、输出流

1、系统标准的输入和输出设备:

  • System.in:键盘(默认)——》内存,类型InputStream
  • System.out:内存——》显示器(默认),类型PrintStream,OutputStream的子类FilterOutputStream 的子类。

2、修改默认设备:System类的setIn,setOut方法

  • public static void setIn(InputStream in)
  • public static void setOut(PrintStream out)——结合打印流可以改变

3、System类中有三个常量对象:System.out、System.in、System.err
提问:final声明的常量一旦赋值就不能修改,那么null会空指针异常吗?为什么要小写?set是如何修改这三个的值的?
答:final声明的常量在java的语法体系中不能修改,但是他们是由c/c++等系统函数进行初始化和修改的,所以没有大写,也有set方法。

public final static InputStream in = null;
public final static PrintStream out = null;
public final static PrintStream err = null;

源码解析:常量能修改的原因

public static void setOut(PrintStream out) {
    checkIO();
    setOut0(out);
}
public static void setErr(PrintStream err) {
    checkIO();
    setErr0(err);
}
public static void setIn(InputStream in) {
    checkIO();
    setIn0(in);
}
private static void checkIO() {
    SecurityManager sm = getSecurityManager();
    if (sm != null) {
        sm.checkPermission(new RuntimePermission("setIO"));
    }
}
private static native void setIn0(InputStream in);
private static native void setOut0(PrintStream out);
private static native void setErr0(PrintStream err);

从键盘输入字符串,要求将读取到的整行字符串转成大写输出。然后继续进行输入操作,直至当输入“e”或者“exit”时,退出程序。

System.out.println("请输入信息(退出输入e或exit):");
// 把"标准"输入流(键盘输入——读入到内存中的流)这个字节流包装成字符流(处理流之转换流),再包装成缓冲流
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
String s = null;
try {
    while ((s = br.readLine()) != null) { // 读取用户输入的一行数据 --> 阻塞程序
        if ("e".equalsIgnoreCase(s) || "exit".equalsIgnoreCase(s)) {
            System.out.println("安全退出!!");
            break;
        }
        // 将读取到的整行字符串转成大写输出
        System.out.println("-->:" + s.toUpperCase());
        System.out.println("继续输入信息");
    }
} catch (IOException e) {
    e.printStackTrace();
} finally {
    try {
        if (br != null) {
            br.close(); // 关闭过滤流时,会自动关闭它包装的底层节点流
        }
    } catch (IOException e) {
        e.printStackTrace();
    }
}

4.2 打印流PrintStream/PrintWriter

1、打印流作用:基本数据类型——》字符串输出

  • 可以输出多种数据类型:提供了一系列重载的print()和println()方法
  • PrintStream和PrintWriter的输出不会抛出IOException异常
  • PrintStream和PrintWriter有自动flush功能
  • PrintStream 打印的所有字符都使用平台的默认字符编码转换为字节。在需要写入字符而不是写入字节的情况下,应该使用 PrintWriter 类。
  • System.out返回的是PrintStream的实例

2、打印流的构造器

  • PrintStream(File file) :创建具有指定文件且不带自动行刷新的新打印流。
  • PrintStream(File file, String csn):创建具有指定文件名称和字符集且不带自动行刷新的新打印流。
  • PrintStream(OutputStream out) :创建新的打印流。
  • PrintStream(OutputStream out, boolean autoFlush):创建新的打印流。 autoFlush如果为 true,则每当写入 byte 数组、调用其中一个 println 方法或写入换行符或字节 (‘\n’) 时都会刷新输出缓冲区。
  • PrintStream(OutputStream out, boolean autoFlush, String encoding) :创建新的打印流。
  • PrintStream(String fileName):创建具有指定文件名称且不带自动行刷新的新打印流。
  • PrintStream(String fileName, String csn) :创建具有指定文件名称和字符集且不带自动行刷新的新打印流。

使用示例:自定义一个日志工具

/*
日志工具
 */
public class Logger {
    /*
    记录日志的方法。
     */
    public static void log(String msg) {
        try {
            // 指向一个日志文件
            PrintStream out = new PrintStream(new FileOutputStream("log.txt", true));
            // 改变输出方向
            System.setOut(out);
            // 日期当前时间
            Date nowTime = new Date();
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS");
            String strTime = sdf.format(nowTime);

            System.out.println(strTime + ": " + msg);
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
    }
}

测试日志工具

public class LogTest {
    public static void main(String[] args) {
        //测试工具类是否好用
        Logger.log("调用了System类的gc()方法,建议启动垃圾回收");
        Logger.log("调用了TeamView的addMember()方法");
        Logger.log("用户尝试进行登录,验证失败");
    }
}

4.3 Scanner类

1、构造方法

  • Scanner(File source) :构造一个新的 Scanner,它生成的值是从指定文件扫描的。
  • Scanner(File source, String charsetName) :构造一个新的 Scanner,它生成的值是从指定文件扫描的。
  • Scanner(InputStream source) :构造一个新的 Scanner,它生成的值是从指定的输入流扫描的。
  • Scanner(InputStream source, String charsetName) :构造一个新的 Scanner,它生成的值是从指定的输入流扫描的。

2、常用方法:

  • boolean hasNextXxx(): 如果通过使用nextXxx()方法,此扫描器输入信息中的下一个标记可以解释为默认基数中的一个 Xxx 值,则返回 true。
  • Xxx nextXxx(): 将输入信息的下一个标记扫描为一个Xxx
package com.atguigu.systemio;

import org.junit.Test;

import java.io.*;
import java.util.Scanner;

public class TestScanner {

    @Test
    public void test01() throws IOException {
        Scanner input = new Scanner(System.in);
        PrintStream ps = new PrintStream("1.txt");
        while(true){
            System.out.print("请输入一个单词:");
            String str = input.nextLine();
            if("stop".equals(str)){
                break;
            }
            ps.println(str);
        }
        input.close();
        ps.close();
    }
    
    @Test
    public void test2() throws IOException {
        Scanner input = new Scanner(new FileInputStream("1.txt"));
        while(input.hasNextLine()){
            String str = input.nextLine();
            System.out.println(str);
        }
        input.close();
    }
}

4.4 apache-common包的使用

1、作用:IO技术的工具类commonsIO,简化IO开发。
Oracle第一方,我们自己第二方,其他都是第卅安防

2、使用方式:导入commons-io-2.5.jar包,使用内部的API。

  • IOUtils类的使用
    • 静态方法:IOUtils.copy(InputStream in,OutputStream out)传递字节流,实现文件复制。
    • 静态方法:IOUtils.closeQuietly(任意流对象)悄悄的释放资源,自动处理close()方法抛出的异常。
  • FileUtils类的使用
    • 静态方法:void copyDirectoryToDirectory(File src,File dest):整个目录的复制,自动进行递归遍历
      参数:
      src:要复制的文件夹路径
      dest:要将文件夹粘贴到哪里去
    • 静态方法:void writeStringToFile(File file,String content):将内容content写入到file中
    • 静态方法:String readFileToString(File file):读取文件内容,并返回一个String
    • 静态方法:void copyFile(File srcFile,File destFile):文件复制

IOUtils类的使用

public class Test01 {
    public static void main(String[] args)throws Exception {
        //- 静态方法:IOUtils.copy(InputStream in,OutputStream out)传递字节流,实现文件复制。
        IOUtils.copy(new FileInputStream("E:\\Idea\\io\\1.jpg"),new FileOutputStream("E:\\Idea\\io\\file\\柳岩.jpg"));
        //- 静态方法:IOUtils.closeQuietly(任意流对象)悄悄的释放资源,自动处理close()方法抛出的异常。
       /* FileWriter fw = null;
        try {
            fw = new FileWriter("day21\\io\\writer.txt");
            fw.write("hahah");
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
           IOUtils.closeQuietly(fw);
        }*/
    }
}

FileUtils类的使用

public class Test02 {
    public static void main(String[] args) {
        try {
            //- 静态方法:void copyDirectoryToDirectory(File src,File dest);
            FileUtils.copyDirectoryToDirectory(new File("E:\\Idea\\io\\aa"),new File("E:\\Idea\\io\\file"));


            //- 静态方法:writeStringToFile(File file,String str)
            FileUtils.writeStringToFile(new File("day21\\io\\commons.txt"),"柳岩你好");

            //- 静态方法:String readFileToString(File file)
            String s = FileUtils.readFileToString(new File("day21\\io\\commons.txt"));
            System.out.println(s);
            //- 静态方法:void copyFile(File srcFile,File destFile)
            FileUtils.copyFile(new File("io\\yangm.png"),new File("io\\yangm2.png"));
            System.out.println("复制成功");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

五、企业真题

1、谈谈Java IO里面的常用类,字节流,字符流
File、字符流、字节流的抽象基类。

2、Java 中有几种类型的流?JDK为每种类型的流提供一些抽象类以供继承,请说出他们分别是哪些类?
InputStream \ OutputStream \ Reader \ Writer

3、流一般需不需要关闭?如果关闭的话用什么方法?处理流是怎么关闭的?
需要。close()
处理流在关闭过程中,也会关闭内部的流。

4、OutputStream里面的write()是什么意思?
数据写出的意思。理解为从内存写出到文件中

5、BufferedReader属于哪种流?他主要是用来做什么的?
缓冲流,作用:不直接连接到数据源或目的地,而是“连接”在已存在的流(节点流或处理流)之上,通过对数据的处理为程序提供更为强大的读写功能。提升文件读写的效率,减少与磁盘交互的次数。

6、什么是缓冲区?有什么作用?
内部提供了一个数组,将读取或要写出的数据,现在此数组中缓存。达到一定程度时,集中性的写出。
作用:减少与磁盘的交互,进而提升读写效率。

7、字节流和字符流是什么?怎么转换?
字节流:二进制流,字符流:一个一个字符
InputStreamReader、OutputStreamWriter
输入型字节流——》输入型字符流
输出型字符流——》输出型字节流
字符流-字节流-转换

8、什么是Java序列化,如何实现
对象序列化机制允许把内存中的Java对象转换成平台无关的二进制流,从而允许把这种二进制流持久地保存在磁盘上,
或通过网络将这种二进制流传输到另一个网络节点。//当其它程序获取了这种二进制流,就可以恢复成原来的Java对象。

9、Java有些类中为什么需要实现Serializable接口?
便于此类的对象实现序列化操作。标记接口,显式声明,当反序列化时回去找该对象,以便能够正确反序列化,如果不显式声明的化,那么当修改类后生成不同的class文件的uid会发生改变,导致反序列化不成功。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/746395.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

[leetcode]assign-cookies. 分发饼干

. - 力扣&#xff08;LeetCode&#xff09; class Solution { public:int findContentChildren(vector<int>& g, vector<int>& s) {sort(g.begin(), g.end());sort(s.begin(), s.end());int m g.size(), n s.size();int count 0;for (int i 0, j 0; i…

代码随想录算法训练营第50天(py)| 动态规划 | 1143.最长公共子序列、1035.不相交的线、53. 最大子序和、392.判断子序列

1143.最长公共子序列 力扣链接 给定两个字符串 text1 和 text2&#xff0c;返回这两个字符串的最长 公共子序列&#xff08;未必连续&#xff09; 的长度。如果不存在 公共子序列 &#xff0c;返回 0 。 思路 确定dp含义 dp[i][j]&#xff1a;长度为[0,i-1]和[0,j-1]的最长公…

Redis-实战篇-缓存雪崩

文章目录 1、缓存雪崩2、解决方案&#xff1a; 1、缓存雪崩 缓存雪崩是指在同一时段大量的缓存key同时失效或者Redis服务宕机&#xff0c;导致大量请求到达数据库&#xff0c;带来巨大压力。 2、解决方案&#xff1a; 给不同的key的TTL添加随机值利用Redis集群提高服务的可用性…

0.7 模拟电视标准 PAL 简介

0.7 模拟电视标准PAL PAL 是一种用于模拟电视的彩色编码系统&#xff0c;全名为逐行倒相&#xff08;Phase Alternating Line&#xff09;。它是三大模拟彩色电视标准之一&#xff0c;另外两个标准是 NTSC 和 SECAM。“逐行倒相”的意思是每行扫描线的彩色信号会跟上一行倒相&…

Axure 教程 | 雅虎新闻焦点

主要内容 在雅虎首页&#xff0c;新闻焦点大图和焦点小图同步切换轮播&#xff0c;本课程我们来学习如何实现这个效果。 交互说明 1.页面载入后&#xff0c;切换当前屏幕显示的5张焦点图&#xff0c;小图标处以横线提示当前焦点图。 2.鼠标移入焦点大图&#xff0c;新闻标题显示…

用两个钟,我又在VMWARE上搞了一套内部网配置

最近要学es&#xff0c;所以打算自己用虚拟机搞个NAT&#xff0c;又搞了两个钟。为了不再费劲尝试&#xff0c;也为了造福大众&#xff0c;所以选择搞一份NAT笔记&#xff01;&#xff01;&#xff01;&#xff01; 1.初始化网关和DNS 我们给网关配置一个地址192.168.96.1&…

发包真香之:scapy工具

scapy – python 可自由组包 参考学习&#xff1a;初识Scapy–Python的Scapy/Kamene模块学习之路 scapy 介绍 Scapy是基于Python语言的网络报文处理程序&#xff0c;它可以让用户发送、嗅探、解析、以及伪造网络报文&#xff0c;运用Scapy可以进行网路侦测、端口扫描、路由追…

【手眼标定】使用kalibr对imu和双目摄像头进行联合标定

使用kalibr对imu和双目摄像头进行联合标定 前言一、IMU标定二、双目摄像头标定三、手眼标定&#xff08;imu和双目摄像头的联合标定&#xff09; 前言 由于本文的imu、双目摄像头都是在ros2环境下开发&#xff0c;数据传输自然也是在ros2中。 但想要使用kalibr进行标定&#x…

Power BI 插件 DAX Studio 安装配置

1&#xff0c;dax studio 下载地址 DAX Studio | DAX Studio 2&#xff0c;安装配置&#xff08;几乎是默认&#xff09; 3&#xff0c;使用方法 打开DAX studio 默认支持Power povit, PBI/SSDT ,Tabular server。先打开PBI再打开DAX studio &#xff0c;不然如果只打开Dax …

ios18开发者预览,Beta 2升级新增镜像等功能

近日&#xff0c;苹果发布了 iOS 18 开发者预览版 Beta 2 升级&#xff0c;为 iPhone 用户带来了多项新功能。据了解&#xff0c;这些新功能包括 iPhone 镜像和 SharePlay 屏幕共享&#xff0c;以及其他新增功能。 据了解&#xff0c;iPhone镜像可以让Mac用户将iPhone屏幕镜像…

IPFoxy Tips:匿名海外代理IP的使用方法及注意事项

在互联网上&#xff0c;隐私和安全问题一直备受关注。为了保护个人隐私和数据安全&#xff0c;使用匿名代理IP是一种常用的方法。匿名代理IP可以隐藏用户的真实IP地址&#xff0c;使用户在访问网站时更加隐秘和安全。 本文将介绍匿名代理IP的基本原理和核心功能。 基本原则 匿…

【云原生】Docker可视化工具Portainer使用详解

目录 一、前言 二、docker可视化管理概述​​​​​​​ 2.1 什么是docker可视化管理 2.1.1 Docker可视化管理常用功能 2.2 为什么需要docker可视化管理工具 2.3 docker可视化工具带来的好处 三、常用的docker容器可视化管理工具解决方案 3.1 Portainer 3.2 Rancher 3…

作 业 二

cs与msf权限传递 1、进入cs界面,首先来到 Cobalt Strike 目录下&#xff0c;启动 Cobalt Strike 服务端 2、用客户端进 3、建立监听 4、生成脚本文件 5、开启服务&#xff0c;让win_2012 下载木马文件并运行 6、显示已经获取到了win的权限 转到Metasploit Framework 7、进去m…

6 序列数据和文本的深度学习

6.1 使用文本数据 文本是常用的序列化数据类型之一。文本数据可以看作是一个字符序列或词的序列。对大多数问题&#xff0c;我们都将文本看作词序列。深度学习序列模型(如RNN及其变体)能够从文本数据中学习重要的模式。这些模式可以解决类似以下领域中的问题&#xff1a; 自然…

vface贴图使用说明

第一部分&#xff1a;01_head 说明 geos&#xff1a;几何体正常导入DCC软件即可&#xff1b; maps&#xff1a;重点说明: ID_mask&#xff1a;做头部区域细节区分控制的遮罩图&#xff1b;官方有详细教程&#xff1b; XYZ_albedo_lin_srgb.1001.exr 颜色贴图正常使用即可&…

【曦灵平台】深度体验百度智能云曦灵平台之数字人3.0、声音克隆、直播等功能,AI加持就是不一样,快来一起体验

目录 资产数字人 2D数字人克隆声音克隆 AI卡片更多功能总结推荐文章 资产 可进行人像与声音的定制&#xff0c;让数字人形象和声音成为我们的专属资产&#xff0c;用于后续的内容生产工作 数字人 这里拍摄的视频分辨率和帧率必须要确保是官方要求&#xff0c;这里博主通过第…

再谈kettle两种循环之--调用http分页接口循环获取数据

再谈kettle两种循环之 – 调用http分页接口循环获取数据 1.场景介绍&#xff1a; 由于数据量比较大,接口有返回限制,需要用到循环分页获取数据 2.案例适用范围&#xff1a; 循环job可参考&#xff0c;变量运用可参考&#xff0c;调用http分页接口循环获取数据可参考&#…

【idea-jdk1.8】使用Spring Initializr 创建 Spring Boot项目没有JDK8

信息差真可怕&#xff01; 很久没创建springboot项目&#xff0c;今天使用idea的Spring Initializr 创建 Spring Boot项目时&#xff0c;发现java版本里&#xff0c;无法选择jdk1.8&#xff0c;只有17、21、22&#xff1b;前段时间也听说过&#xff0c;springboot将放弃java8&a…

Java面试问题(一)

一.Java语言具有的哪些特点 1.Java是纯面向对象语言&#xff0c;能够直接反应现实生活中的对象 2.具有平台无关性&#xff0c;利用Java虚拟机运行字节码文件&#xff0c;无论是在window、Linux还是macOS等其他平台对Java程序进行编译&#xff0c;编译后的程序可在其他平台上运行…

深入理解计算机系统 CSAPP 家庭作业7.13

用一下496页提到的工具咯 A: whereis libm.a file lidm.a gedit libm.a libm.a是个ASCII text文件打开一看原来 libm-2.27.a 和libmvec.a才是我们要看的 所以我们cd到目标地址后 ar -t libm-2.27.a ar -t libmvec.a B: gcc -Og bar5.c foo5.c 用之前的两个文件链接后生成…