java的File类、递归

File类、递归

File类

概述

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

File的静态成员变量

import java.io.File;

/*
    java.io.File类
    文件和目录路径名的抽象表示形式。
    java把电脑中的文件和文件夹(目录)封装为了一个File类,我们可以使用File类对文件和文件夹进行操作
    我们可以使用File类的方法
        创建一个文件/文件夹
        删除文件/文件夹
        获取文件/文件夹
        判断文件/文件夹是否存在
        对文件夹进行遍历
        获取文件的大小
    File类是一个与系统无关的类,任何操作系统都可以使用这个类的方法

    重点:记住这三个单词
        file:文件
        directory:文件夹/目录
        path:路径
 */
public class Demo01File {
     
    public static void main(String[] args) {
     
        /*
            static string pathSeparator 与系统有关的路径分隔符,为了方便,它被表示为一个字符串。
            static char pathSeparatorChar 与系统有关的路径分隔符。

            static String separator 与系统有关的默认名称分隔符,为了方便,它被表示为一个字符串。
            static char separatorChar 与系统有关的默认名称分隔符。

            操作路径:路径不能写死了
            C:\develop\a\a.txt Windows
            C:/develop/a/a.txt Linux
            "C:" + File.separator + "develop" + File.separator + "a" + File.separator + "a.txt"
         */
        String pathSeparator = File.pathSeparator;
        System.out.println(pathSeparator);//路径分隔符 Windows:分号; Linux:冒号:

        String separator = File.separator;
        System.out.println(separator);//文件名称分隔符  Windows:反斜杠\ Linux:正斜杠/
    }
}

构造方法

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

import java.io.File;

/*
    路径:
        绝对路径:是一个完整的路径
            以盘符(C:,D:)开始的路径
        相对路径:是一个简化的路径
            相对指的是相对于当前项目的根目录
            如果使用当前项目的根目录,路径就可以简化书写
        注意:
            1.路径是不区分大小写的
            2.路径中的文件名称分割福Windows使用反斜杠,反斜杠是转义符,两个反斜杠代表一个普通的反斜杠
 */
public class Demo02File {
     
    public static void main(String[] args) {
     
        /*
            Filr类的构造方法
         */
        //show01();
        //show02("c:\\","a.txt");
        show03();
    }

    /*
        File(File parent, String child) 根据 parent 抽象路径名和 child 路径名字符串创建一个新 File 实例。
        参数:把路径分成两部分
            File parent:父路径
            String child:子路径
        好处:
            父路径和子路径,可以单独书写,使用起来非常灵活;父路径和子路径都可以变化
            父路径是File类型,可以使用File的方法对路径进行一些操作,再使用路径创建对象
     */
    private static void show03() {
     
        File parent = new File("c:\\");
        File file = new File(parent,"hello.java");
        System.out.println(file);//c:\hello.java
    }

    /*
        File(String parent, String child) 根据 parent 路径名字符串和 child 路径名字符串创建一个新 File 实例。
        参数:把路径分成两部分
            String parent:父路径
            String child:子路径
        好处:
            父路径和子路径,可以单独书写,使用起来非常灵活;父路径和子路径都可以变化
     */
    private static void show02(String parent,String child) {
     
        File file = new File(parent,child);
        System.out.println(file);//c:\a.txt
    }

    /*
            File(String pathname) 通过将给定路径名字符串转换为抽象路径名来创建一个新 File 实例。
            参数:
                String pathname:字符串的路径名称
                路径可以是以文件结尾,也可以是以文件夹结尾
                路径可以是相对路径,也可以是绝对路径
                路径可以是存在的,也可以是不存在的
                创建File对象,只是把字符串路径封装到File对象,不考虑路径的真假情况
         */
    private static void show01() {
     
        File f1 = new File("D:\\IDEA2019(中文版)\\ZXL\\a.txt");
        System.out.println(f1);//重写了Object类的toString方法 D:\IDEA2019(中文版)\ZXL\a.txt

        File f2 = new File("D:\\IDEA2019(中文版)\\ZXL");
        System.out.println(f2);//D:\IDEA2019(中文版)\ZXL

        File f3 = new File("b.txt");
        System.out.println(f3);//b.txt
    }
}

常用方法

获取功能的方法

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

/*
    File类获取功能的方法:
        - public String getAbsolutePath():返回此File的绝对路径名字符串。
        - public Stirng getPath():将此File转换为路径字符串
        - public String getName()`返回由此File表示的文件或目录的名称
        - public long length():返回此File表示的文件的长度
 */
public class Demo03File {
     
    public static void main(String[] args) {
     
        show04();
    }
    /*
        public long length():返回此File表示的文件的长度
        后偶去的是构造方法指定的文件大小,以字节为单位
        注意:
            文件夹是没有大小概念的,不能获取文件夹的带下
            如果构造方法中给出的路径不存在,那么length方法返回0
     */
    private static void show04() {
     
        File f1 = new File("D:\\朱相龙\\java学习\\笔记\\图片\\对象数组.png");
        long l1 = f1.length();
        System.out.println(l1);//477069字节

        File f2 = new File("D:\\朱相龙\\java学习\\笔记\\图片\\对象数组2.png");
        long l2 = f2.length();
        System.out.println(l2);//0
    }

    /*
        public String getName()`返回由此File表示的文件或目录的名称
        获取的就是构造方法传递路径的结尾部分(文件/文件夹)
     */
    private static void show03() {
     
        File f1 = new File("D:\\IDEA2019(中文版)\\ZXL\\a.txt");
        String name1 = f1.getName();
        System.out.println(name1);//a.txt

        File f2 = new File("D:\\IDEA2019(中文版)\\ZXL");
        String name2 = f2.getName();
        System.out.println(name2);//ZXL
    }

    /*
        public Stirng getPath():将此File转换为路径字符串
        获取的构造方法中传递的路径

        toString方法调用的就是getPath方法
        源码:
            public String toString() {
                return getPath();
            }
     */
    private static void show02() {
     
        File f1 = new File("D:\\IDEA2019(中文版)\\ZXL\\a.txt");
        File f2 = new File("a.txt");
        String path1 = f1.getPath();
        System.out.println(path1);//D:\IDEA2019(中文版)\ZXL\a.txt
        String path2 = f2.getPath();
        System.out.println(path2);//a.txt

        System.out.println(f1);//D:\IDEA2019(中文版)\ZXL\a.txt
        System.out.println(f1.toString());//D:\IDEA2019(中文版)\ZXL\a.txt

    }

    /*
        public String getAbsolutePath():返回此File的绝对路径名字符串。
        获取的构造方法中传递的路径
        无论路径是绝对的还是相对的,getAbsolutePath方法返回的都是绝对路径
     */
    private static void show01() {
     
        File f1 = new File("D:\\IDEA2019(中文版)\\ZXL\\a.txt");
        String absolutePath1 = f1.getAbsolutePath();
        System.out.println(absolutePath1);//D:\IDEA2019(中文版)\ZXL\a.txt

        File f2 = new File("a.txt");
        String absolutePath2 = f2.getAbsolutePath();
        System.out.println(absolutePath2);//D:\IDEA2019(中文版)\ZXL\a.txt
    }
}

判断功能的方法

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

/*
    File类判断功能的方法:
        - public boolean exists():此File表示的文件或目录是否实际存在
        - public boolean isDirectory:此File表示的是否为目录
        - public boolean isFile:此File表示的是否为文件
 */
public class Demo04File {
     
    public static void main(String[] args) {
     
        show02();
    }
    /*
        public boolean isDirectory:此File表示的是否为目录
            判断构造方法中给定的路径是否已文件夹结尾
                是:true
                否:false
        public boolean isFile:此File表示的是否为文件
            判断构造方法中给定的路径是否已文件结尾
                是:true
                否:false
        注意:
            电脑的硬盘中只有文件/文件夹,两个方法是互斥
            这两个方法使用前提,路径必须是存在的,否则都返回false
     */
    private static void show02() {
     
        File f1 = new File("D:\\IDEA2019(中文版)\\Z");
        //不存在,就没有必要获取
        if (f1.exists()){
     
            System.out.println(f1.isDirectory());
            System.out.println(f1.isFile());
        }

        File f2 = new File("D:\\IDEA2019(中文版)\\ZXL");
        //不存在,就没有必要获取
        if (f2.exists()){
     
            System.out.println(f2.isDirectory());//true
            System.out.println(f2.isFile());//false
        }

        File f3 = new File("D:\\IDEA2019(中文版)\\ZXL\\ZXL.iml");
        //不存在,就没有必要获取
        if (f3.exists()){
     
            System.out.println(f3.isDirectory());//false
            System.out.println(f3.isFile());//true
        }
    }

    /*
        public boolean exists():此File表示的文件或目录是否实际存在
        用于判断构造方法中的路径是否存在
            存在:true
            不存在:false
     */
    private static void show01() {
     
        File f1 = new File("D:\\IDEA2019(中文版)\\ZXL");
        System.out.println(f1.exists());//true

        File f2 = new File("D:\\IDEA2019(中文版)\\Z");
        System.out.println(f2.exists());//false

        File f3 = new File("ZXL.iml");//相对路径    D:\IDEA2019(中文版)\ZXL\\ZXL.iml
        System.out.println(f3.exists());//true

        File f4 = new File("a.txt");
        System.out.println(f4.exists());//false
    }
}

创建删除功能的方法

  • public boolean createNewFile() :当且仅当具有该名称的文件尚不存在时,创建一个新的空文件
  • public boolean delete(): 删除由此File表示的文件或目录。
  • public boolean mkdir() :创建单级文件夹。
  • public boolean mkdirs():既可以创建单级空文件夹,也可以创建多级
import java.io.File;
import java.io.IOException;

/*
    File类创建删除功能的方法
        - public boolean createNewFile(): 当且仅当具有该名称的文件尚不存在时,创建一个新的空文件
        - public boolean delete(): 删除由此File表示的文件或目录。
        - public boolean mkdir(): 创建由此File表示的目录。
        - public boolean mkdirs(): 创建由此File表示的目录,包括任何必需但不存在的父目录
 */
public class Demo05File {
    public static void main(String[] args) throws IOException {
        show03();
    }
    /*
        public boolean delete(): 删除由此File表示的文件或目录
        此方法,可以删除构造方法路径中给出的文件/文件夹
        返回值:布尔值
            true:文件/文件夹删除承购,返回true
            false:文件夹中有内容,不会删除返回false;构造方法中不存在false
        注意:
            delete方法是直接在硬盘删除文件/文件夹,不走回收站,删除要谨慎
     */
    private static void show03() {
        File f1 = new File("新建文件夹");
        System.out.println(f1.delete());

        File f2 = new File("2.txt");
        System.out.println(f2.delete());

        File f3 = new File("111");
        System.out.println(f3.delete());//文件夹中有内容,不会删除返回false

        File f4 = new File("bbb");
        System.out.println(f4.delete());//构造方法中不存在false
    }

    /*
        public boolean mkdir(): 创建单级文件夹
        public boolean mkdirs(): 既可以创建单级空文件夹,也可以创建多级
        创建文件夹的路径和名称在构造方法中给出(构造方法的参数)
        返回值:布尔值
            true:文件夹不存在,创建文件,返回true
            false:文件夹存在,不会创建,返回false;构造方法中给出的路径不存在,也返回false
       注意:
            1.此方法只能创建文件夹,不能创建文件
     */
    private static void show02() {
        File f1 = new File("aaa");
        System.out.println(f1.mkdir());

        File f2 = new File("111\\222\\333\\444");
        System.out.println(f2.mkdirs());

        File f3 = new File("aaa.txt");
        System.out.println(f3.mkdirs());//看类型,是一个文件

        File f4 = new File("D:\\IDEA2019(中文版)\\ZXL\\src\\com\\company\\pratise\\FileAndRecur\\aaa.txt");
        System.out.println(f4.mkdirs());//不会抛出异常,但是路径不存在,不会创建
    }

    /*
        public boolean createNewFile(): 当且仅当具有该名称的文件尚不存在时,创建一个新的空文件
        创建文件的路径和名称在构造方法中给出(构造方法的参数)
        返回值:布尔值
            true:文件不存在,创建文件,返回true
            false:文件存在,不会创建,返回false
       注意:
            1.此方法只能创建文件,不能创建文件夹
            2.创建文件的路径必须存在,否则会抛出异常

       public boolean createNewFile() throws IOException
       createNewFile声明抛出IOException,我们调用这个方法,就必须的处理这个异常,要么throws,要么try...catch
     */
    private static void show01() throws IOException {
        File f1 = new File("D:\\IDEA2019(中文版)\\ZXL\\src\\com\\company\\pratise\\FileAndRecursion\\1.txt");
        boolean b1 = f1.createNewFile();
        System.out.println("b1:" + b1);

        File f2 = new File("2.txt");
        System.out.println(f2.createNewFile());

        File f3 = new File("新建文件夹");
        System.out.println(f3.createNewFile());//不要被名称迷惑,要看类型

        File f4 = new File("D:\\IDEA2019(中文版)\\ZXL\\src\\com\\company\\pratise\\FileAndRecur\\3.txt");
        System.out.println(f4.createNewFile());//路径不存在,抛出IOException
    }
}

目录的遍历

  • public String[] list():返回一个String数组,表示该File目录中的所有子文件或目录
  • public File[] listFiles():返回一个File数组,表示该File目录中的所有子文件或目录

递归

概述

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

    /*
            在递归中虽然有限定条件,但是递归次数不能太多。否则也会发生栈内存溢出
            23322
                Exception in thread "main" java.lang.StackOverflowError
         */
    private static void b(int i) {
     
        System.out.println(i);
        if(i == 30000){
     
            return;//结束方法
        }
        b(++i);
    }

    /*
        递归一定要有条件限定保证递归能够停止下来,否则会发生栈内存溢出
        Exception in thread "main" java.lang.StackOverflowError
     */
    private static void a() {
     
        System.out.println("a方法!");
        a();
    }
}

递归累加求和

/*
    练习:
        使用递归计算1-n之间的和
 */
public class Demo02Recursion {
     
    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)+...+1
        已知:
            最大值:n
            最小值:1
        使用递归必须明确:
            1.递归的结束条件
                获取到1的时候结束
            2.递归的目的
                获取下一个被加的数字(n-1)
     */
    public static int sum(int n){
     
        if(n == 1){
     
            return 1;
        }
        //获取下一个倍加的数字(n-1)
        return n + sum(n-1);
    }
}

递归求阶乘

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

    /*
        定义方法使用递归计算阶乘
     */
    public static int jc(int n){
        if(n == 1){
            return 1;
        }
        return n*jc(n-1);
    }
}

递归打印多级目录

import java.io.File;
/*
    练习:
        递归打印多级目录
 */
public class Demo04Recursion {
     
    public static void main(String[] args) {
     
        File file = new File("D:\\IDEA2019(中文版)\\ZXL\\src\\com\\company\\pratise\\Thread");
        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是一个文件,直接打印即可
                System.out.println(f);
            }
        }
    }
}

综合案例

文件搜索

import java.io.File;
/*
    练习:
        递归打印多级目录
    需求:
        只要.java文件
 */
public class Demo05Recursion {
     
    public static void main(String[] args) {
     
        File file = new File("D:\\IDEA2019(中文版)\\ZXL\\src\\com\\company\\pratise\\Thread");
        getAllFile(file);
    }
    /*
        定义一个方法,参数床底File类型目录
        方法中对目录进行遍历
     */
    public static void getAllFile(File 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 name = f.getName();
                //String path = f.getPath();
                //String s = f.toString();

                //把字符串,转换为小写
                //s = s.toLowerCase();

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

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

                if(f.getName().toLowerCase().endsWith(".java")){
     
                    System.out.println(f);
                }
            }
        }
    }
}

文件过滤器优化

java.io.FileFilter是一个接口,是File的过滤器。该接口的对象可以传递给File类的listFiles(FileFilter)作为参数,接口只有一个方法。

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

import java.io.File;
/*
    练习:
        递归打印多级目录
    我们可以使用过滤器来实现
    在File类中有两个和listFiles重载的方法,方法的参数传递的就是过滤器
    File[] listFiles(FileFilter filter)
    java.io.FileFilter接口:用于抽象路径名(File对象)的过滤器
        作用:用来过滤文件(File对象)
        抽象方法:用来过滤文件的方法
            boolean accept(File pathname) 测试指定抽象路径名是否应该包含在某个路径名列表中
            参数:
                File pathname:使用listFiles方法遍历目录,得到的每一个文件对象
    File[] listFiles(FilenameFilter filter)
    java.io.FilenameFilter接口:实现此接口的类实例可用于过滤器文件名
        作用:用于过滤文件名称
        抽象方法:用来过滤文件的方法
            boolean accept(File dir , String name) 测试指定文件是否应该包含在某一文件列表中
            参数:
                File dir:构造方法中传递的被遍历的目录
                String name:使用listFiles方法遍历目录,获取的每一个文件/文件夹名称
    注意:
        两个过滤器接口是没有实现类的,需要我们自己写实现类,重写过滤的方法accept,在方法中自己定义过滤的规则
 */
public class Demo01Filter {
     
    public static void main(String[] args) {
     
        File file = new File("D:\\IDEA2019(中文版)\\ZXL\\src\\com\\company\\pratise\\Thread");
        getAllFile(file);
    }
    /*
        定义一个方法,参数床底File类型目录
        方法中对目录进行遍历
     */
    public static void getAllFile(File dir){
     
        File[] files = dir.listFiles(new FileFilterImpl());//传递过滤器对象
        for (File f : files) {
     
            //对遍历得到的File对象f进行判断,判断是否为文件夹
            if (f.isDirectory()){
     
                //f是一个文件夹,则继续用遍历这个文件夹
                //我们发现getAllFile方法就是传递文件夹,遍历文件夹的方法
                //所以直接调用getAllFile方法即可
                getAllFile(f);
            }else {
     
                //f是一个文件,直接打印即可
                System.out.println(f);
            }
        }
    }
}

import java.io.File;
import java.io.FileFilter;

/*
    创建过滤器FileFilter的实现类,重写过滤的方法accept,定义过滤规则
 */
public class FileFilterImpl implements FileFilter {
     
    @Override
    public boolean accept(File pathname) {
     
        /*
            过滤的规则:
            在accept方法中,判断File对象是否是以.java结尾的
            是就返回true
            不是就返回false
         */
        //如果pathname是一个文件夹,返回true,继续遍历这个文件夹
        if(pathname.isDirectory()){
     
            return true;
        }
        return pathname.getName().toLowerCase().endsWith(".java");
    }
}

Lambda优化程序

import java.io.File;
import java.io.FileFilter;
import java.io.FilenameFilter;

public class Demo02Filter {
     
    public static void main(String[] args) {
     
        File file = new File("D:\\IDEA2019(中文版)\\ZXL\\src\\com\\company\\pratise\\Thread");
        getAllFile(file);
    }
    /*
        定义一个方法,参数床底File类型目录
        方法中对目录进行遍历
     */
    public static void getAllFile(File dir){
     
        //传递过滤器对象,使用匿名内部类
        /*File[] files = dir.listFiles(new FileFilter() {
            @Override
            public boolean accept(File pathname) {
                //过滤规则,pathname是文件夹或者是.java结尾的文件夹返回true
                return pathname.isDirectory() || pathname.getName().toLowerCase().endsWith(".java");
            }
        });*/

        //使用Lambda表达式优化匿名内部类(接口中只有一个抽象方法)
        /*File[] files = dir.listFiles((pathname) ->
                //过滤规则,pathname是文件夹或者是.java结尾的文件夹返回true
                pathname.isDirectory() || pathname.getName().toLowerCase().endsWith(".java")

        );*/

        /*File[] files = dir.listFiles(new FilenameFilter() {
            @Override
            public boolean accept(File dir, String name) {
                //过滤规则,pathname是文件夹或者是.java结尾的文件夹返回true
                return new File(dir,name).isDirectory() || name.toLowerCase().endsWith(".java");
            }
        });*/

        //使用Lambda表达式优化匿名内部类(接口中只有一个抽象方法)
        File[] files = dir.listFiles((d,name) ->
            //过滤规则,pathname是文件夹或者是.java结尾的文件夹返回true
            new File(d,name).isDirectory() || name.toLowerCase().endsWith(".java"));

        for (File f : files) {
     
            //对遍历得到的File对象f进行判断,判断是否为文件夹
            if (f.isDirectory()){
     
                //f是一个文件夹,则继续用遍历这个文件夹
                //我们发现getAllFile方法就是传递文件夹,遍历文件夹的方法
                //所以直接调用getAllFile方法即可
                getAllFile(f);
            }else {
     
                //f是一个文件,直接打印即可
                System.out.println(f);
            }
        }
    }
}

你可能感兴趣的:(java进阶,java,递归法)