天天看点

File类、递归一、File类二、递归

一、File类

1.1 概述

java.io.File类是文件和目录路径名的抽象表示,主要用于文件和目录的创建、查找和删等操作。
           

1.2 构造方法

* public File(String pathname):通过将给定的路径名字符串转换为抽象路径名来创建新的File实例。
* public File(String parent,String child):从父路径名字符串和子路径名字符串创建新的File实例。
* public File(File parent, String child):从父抽象路径名和子路径名字符串创建新的File实例。
           

构造举例,代码如下:

//文件路径名
	String pathname = "D:\\aaa.txt";
	File filel = new File(pathname);

//文件路径名
	String pathname2 = "D:\\aaa\\bbb.txt";
	File file2 = new File(pathname2);
	
//通过父路径和子路径字符串
	String parent = "D:\\aaa";
	String child = "bbb.txt";
	File file3 = new File(parent,child);

//通过父级File对象和子路径字符串
	File parentDir = newFile("d:\\aaa");
	String child = "bbb.txt";
	File file4 = new File(parentDir, child);

备注:
	1. 一个File对象代表硬盘中实际存在的一个文件或者目录。
	2. 无论该路径下是否存在文件或者目录,都不影响File对象的创建。
           

1.3 常用方法

获取功能的方法

* public String getAbsolutePath():返回此File的绝对路径名字符串。
* public String getPath():将此File转换为路径名字符串。
* public String getName():返回由此File表示的文件或目录的名称。
* public long length():返回由此File表示的文件的长度。
           

方法演示,代码如下:

public class FileGet{
	public static void main(String[] args){
		File f = new File("d://aaa/bbb/java");
		System.out.println("文件绝对路径:"+f.getAbsolutePath());
		System.out.println("文件构造路径:"+f.getPath());
		System.out.println("文件名称:"+f.getname());
		System.out.println("文件长度:"+f.length());
		
		File f2 = new File("d:/aaa");
		System.out.println("目录绝对路径:"+f2.getAbsolutePath());
		System.out.println("目录构造路径:"+f2.getPath());
		System.out.println("目录名称:"+f2.getName());
		System.out.println("目录长度:"+f2.length());
	}
}
输出结果:
文件绝对路径:d:\aaa\bbb.java
文件构造路径:d:\aaa\bbb.java
文件名称:bbb.java
文件长度:636字节

目录绝对路径:d:\aaa
目录构造路径:d:\aaa
目录名称:aaa
目录长度:4096
           

绝对路径和相对路径

  • 绝对路径:从盘符开始的路径,这是一个完整的路径。
  • 相对路径:相对于项目目录的路径,这是一个便捷的路径,开发中经常使用。
public class FilePath {
    public static void main(String[] args) {
      	// D盘下的bbb.java文件
        File f = new File("D:\\bbb.java");
        System.out.println(f.getAbsolutePath());
      	
		// 项目下的bbb.java文件
        File f2 = new File("bbb.java");
        System.out.println(f2.getAbsolutePath());
    }
}
输出结果:
D:\bbb.java
D:\idea_project_test4\bbb.java
           

判断功能的方法

  • public boolean exists():此File表示的文件或目录是否实际存在。
  • public boolean isDirectory():此File表示的是否为目录。
  • public boolean isFile():此File表示的是是否为文件。

方法演示,代码如下:

public class FileIs{
	public static void main(String[] args){
		 File f = new File("d:\\aaa\\bbb.java");
		 File f2 = new File("d:\\aaa");
		 //判断是否存在
		 System.out.println("d:\\aaa\\bbb.java是否存在:"+f.exists());
		 System.out.println("d:\\aaa是否存在"+f2.exists());
		 //判断是文件还是目录
		 System.out.println("d:\\aaa 是文件?"+f2.isFile());
		 System.out.println("d:\\aaa 是目录?"+f2.isDirectory());
	}
}
输出结果:
d:\aaa\bbb.java 是否存在:true
d:\aaa 是否存在:true
d:\aaa 文件?:false
d:\aaa 目录?:true
           

创建删除功能的方法

  • public boolean createNewFile():当且仅当具有该名称的文件尚不存在时,创建一个新的文件。
  • public boolean delete():删除由此File表示的文件或目录。
  • public boolean mkdir():创建由此File表示的目录。
  • public boolean mkdir():创建由此File表示的目录,包括任何必须但不存在的父目录。

方法演示,代码如下:

public class FileCreateDelete {
    public static void main(String[] args) throws IOException {
        // 文件的创建
        File f = new File("aaa.txt");
        System.out.println("是否存在:"+f.exists()); // false
        System.out.println("是否创建:"+f.createNewFile()); // true
        System.out.println("是否存在:"+f.exists()); // true
		
     	// 目录的创建
      	File f2= new File("newDir");	
        System.out.println("是否存在:"+f2.exists());// false
        System.out.println("是否创建:"+f2.mkdir());	// true
        System.out.println("是否存在:"+f2.exists());// true

		// 创建多级目录
      	File f3= new File("newDira\\newDirb");
        System.out.println(f3.mkdir());// false
        File f4= new File("newDira\\newDirb");
        System.out.println(f4.mkdirs());// true
      
      	// 文件的删除
       	System.out.println(f.delete());// true
      
      	// 目录的删除
        System.out.println(f2.delete());// true
        System.out.println(f4.delete());// false
    }
}
           

1.4 目录的遍历

  • public String[] list():返回一个String数组,表示该File目录中的所有子文件或目录。
  • public FIle[] listFiles():返回一个File数组,表示该File目录中的所有的子文件或目录。
public class FileFor{
	public static void main(String args){
		File dir = new File("d:\\java_code");
		
		//获取当前目录下的文件以及文件夹的名称
		String[] names = dir.list();
		for(String name : names){
			System.out.println(name);
		}
		//获取当前目录下的文件以及文件夹对象,只要拿到了对象,那么就可以获取更多信息
		File[] files = dir.listFiles();
		for(File file : files){
			System.out.println(file);
		}
	}
}
           

二、递归

* 递归:方法自己调用自己
    - 递归的分类:
        - 递归分为两种,直接递归和间接递归。
        - 直接递归称为方法自身调用自己
        - 间接递归可以A方法调用B方法,B方法调用C方法,C方法调用A方法。
    - 注意事项:
        - 递归一定要有条件限制,保递归能够停下来,否则会发生栈内存溢出。
        - 在递归中虽然有限定条件,但递归次数不能太多。否则会发生栈内存溢出。
        - 构造方法,禁止递归
    - 递归的使用前提:
        当调用方法的时候,方法的主题不变,每次调用方法的参数不同,可以使用递归
           
public class Demo01Recurison {
    public static void main(String[] args) {
        //a();
        b(1);
    }
    
    /*
        构造方法,禁止递归
            编译报错:构造方法是创建对象使用的,一直递归会导致内存中有无数的对象,所以直接编译报错
     
    public Demo01Recurison(){
        Demo01Recurison();
    }*/
    
    
    private static void b(int i) {
        System.out.println(i);
        if(i==100000){
            return; //结束方法
        }       //StackOverflowError
        b(++i);
    }

    private static void a() {
        System.out.println("a方法!");
        a();
    }
}
           

递归求和

/*
    练习:
        使用递归计算1-n之间的和
 */
public class Demo02Recurison {
    public static void main(String[] args) {
        int s = sum(100);
        System.out.println(s);
    }

    /*
        定义一个方法,使用递归计算1-n之间的和
        1+2+3+...+n
        n+(n-1)+(n-2)+...+2+1
        已知:
            最大值:n
            最小值: 1
        使用递归必须明确:
            1.递归的结束条件
                获取到1的时候结束
            2.递归的目的
                获取下一被加的数字(n-1)
     */
    private static int sum(int n) {
        //获取下一被加的数字(n-1)
        if (n == 1) {
            return 1;
        }
        //获取下一个被加的数字(n-1)
        return n + sum(n - 1);
    }
}
           
File类、递归一、File类二、递归

递归求阶乘

/*
    练习:
        使用递归计算阶乘
        n的阶乘:n! = n*(n-1)*...*3*2*1
 */
public class Demo03Recurison {
    public static void main(String[] args) {
        int jiecheng = jc(5);
        System.out.println(jiecheng);
    }

    /*
        定义方法使用递归计算阶乘
        5的阶乘:5! = 5*(5-1)*(5-2)*(5-3)*(5-4)*(4-1)=5*4*3*2*1
        递归结束的条件:
            获取到1的时候结束
        递归的目的:
            获取下一个被乘的数字(n-1)

     */
    public static int jc(int n) {
        //获取到1的时候结束
        if(n==1){
            return 1;
        }
        //获取下一个被乘的数字(n-1)
        return n * jc(n-1);
    }
}
           

递归打印多级目录(文件搜索)

import java.io.File;

/*
    练习:
        递归打印多级目录
    需求:
        遍历C:\Users\32189\Desktop\考研资料\abc文件夹,以及abc文件夹的子文件夹
        C:\Users\32189\Desktop\考研资料\abc
        C:\Users\32189\Desktop\考研资料\abc\abc.java
        C:\Users\32189\Desktop\考研资料
        C:\Users\32189\Desktop\考研资料
        C:\Users\32189\Desktop\考研资料
        C:\Users\32189\Desktop\考研资料
        C:\Users\32189\Desktop\考研资料
 */
public class Demo04Recurison {
    public static void main(String[] args) {
        File file = new File("C:\\Users\\32189\\Desktop\\考研资料");
        getAllFile(file);
    }
    /*
        定义一个方法,参数传递File类型的目录
        方法对目录中进行遍历
     */
    public static void getAllFile(File dir){
        //System.out.println(dir);    //打印被遍历的目录名称
        File[] files = dir.listFiles();
        for(File f : files){
            //对遍历得到的File对象f进行判断,判断是否是文件夹
            if (f.isDirectory()){
                //f是一个文件夹,则继续遍历这个文件夹
                //我们发现getAllFile方法就是传递文件夹,遍历文件夹的方法
                //所以直接调用getAllFile方法即可:递归(自己调用自己)
                getAllFile(f);
            }else{
                //f是一个文件,直接打开即可
                /*
                    只要.java结尾的文件
                    1. 把File对象f,转换为字符串对象

                 */
                String s = f.getName();

                //2. 调用String类中的方法endWith判断字符串是否是以.java结尾
                boolean b = s.endsWith(".java");

                //3. 如果是以.java结尾的文件,则输出
                if (b) {
                    System.out.println(f);
                }
            }
        }
    }
}
           

文件过滤器优化

java.io.FileFilter

是一个接口,是File的过滤器。 该接口的对象可以传递给File类的

listFiles(FileFilter)

作为参数, 接口中只有一个方法。

boolean accept(File pathname)

:测试pathname是否应该包含在当前File目录中,符合则返回true。

分析:

  1. 接口作为参数,需要传递子类对象,重写其中方法。我们选择匿名内部类方式,比较简单。
  2. accept

    方法,参数为File,表示当前File下所有的子文件和子目录。保留住则返回true,过滤掉则返回false。保留规则:
    1. 要么是.java文件。
    2. 要么是目录,用于继续遍历。
  3. 通过过滤器的作用,

    listFiles(FileFilter)

    返回的数组元素中,子文件对象都是符合条件的,可以直接打印。
public class DiGuiDemo4 {
    public static void main(String[] args) {
        File dir = new File("D:\\aaa");
        printDir2(dir);
    }
  
    public static void printDir2(File dir) {
      	// 匿名内部类方式,创建过滤器子类对象
        File[] files = dir.listFiles(new FileFilter() {
            @Override
            public boolean accept(File pathname) {
                return pathname.getName().endsWith(".java")||pathname.isDirectory();
            }
        });
      	// 循环打印
        for (File file : files) {
            if (file.isFile()) {
                System.out.println("文件名:" + file.getAbsolutePath());
            } else {
                printDir2(file);
            }
        }
    }
}