递归概念&分类&注意事项:

递归:指在当前方法中调用自己的这种现象

递归的分类:直接递归和间接递归

直接递归:方法自身调用自己

间接递归:A方法调用B方法,B方法调用C方法,C方法调用A方法

注意事项:

递归一定要有条件限定,保证递归能够停下来,否则会发生栈内存溢出

在递归中虽然有限定条件,但是递归次数不能太多哦,否则也会发生栈内存溢出

构造方法,禁止递归。


public class Demo01Recursion {
    public static void main(String[] args) {
        //a();
        b(1);
    }
    //递归中虽然有限定条件,但是递归次数不能太多哦,否则也会发生栈内存溢出
    private static void b(int i) {
        System.out.println(i);
        if(i==20000){
            return;//结束方法
        }
        b(++i);
    }
    //构造方法,禁止递归。编译报错:构造方法是创建对象使用的,一直递归会导致内存中有无数个对象,直接编译报错
    public Demo01Recursion() {
        //Demo01Recursion();
    }

    //递归一定要有条件限定,保证递归能够停下来,否则会发生栈内存溢出
    private static void a() {
        System.out.println("a方法");//栈内存溢出异常
        //a方法会在栈内存中,一只调用a方法,会导致栈内存中有无数个a方法,方法太多了就会超出栈内存的大小,就会导致内存溢出的错误
        //当一个方法调用其他方法的时候,被调用的方法没有执行完毕,当前方法会一直等待调用的方法
        //执行完毕才会继续执行
        a();
    }
}


递归使用前提:

当调用方法的时候,方法的主体不变,每次调用方法的参数不同,可以使用递归

练习-1-使用递归计算1-n之间的和:


/*
* 练习:
* 使用递归计算1-n的和
* */
public class Demo02Recursion {
    public static void main(String[] args) {
        int s = sum(100);
        System.out.println(s);//5050
    }
    /*
    * 定义一个方法,使用递归计算1-n的和。
    * 使用递归必须明确:
    * 1.递归的结束条件
    * 获取到1的时候结束
    * 2.递归的目的
    * 获取下一个被加的数字(n-1)
    * */
    public static int sum(int n){
        if(n==1) {
            return 1;
        }
        return n+sum(n-1);
    }
}
注意:使用递归求和,main方法调用sum方法,sum方法会一直调用sum方法,导致在内存中有多个sum方法(频繁的创建方法,调用方法,销毁方法),效率低下。所以仅仅是计算1-n之间的和,不推荐使用递归,使用for循环即可


练习2-使用递归计算阶乘:


/*
* 递归求阶乘
* */
public class Demo03Recursion {
    public static void main(String[] args) {
        int j = jieCheng(5);
        System.out.println(j);
    }
    /*
    * 递归结束的条件:
    * 获取到1的时候结束
    * 递归的目的:
    * 获取下一个被乘的数字n-1
    * */
    public static int jieCheng(int n){
        if(n==0) {
            return 1;
        }
        return n*jieCheng(n-1);
    }
}


练习3-递归打印多级目录:

public class Demo04Recursion {
    public static void main(String[] args) {
        File file=new File("D:\\abc");
        getAllFile(file);
    }
    /*
    * 定义一个方法,参数传递File类型的目录
    * 方法中对目录进行遍历
    * */
    public static void getAllFile(File dir){
        System.out.println(dir);//打印被遍历的目录名称
        File[] files = dir.listFiles();
        for (File file : files) {
            //对遍历得到的File对象 file进行判断
            if(file.isDirectory()){
                //file是一个文件夹,则继续遍历这个文件夹
                //直接调用getAllFile方法即可:递归(自己调用自己)。
                getAllFile(file);
            }else {
                System.out.println(file);
            }
        }
    }
}

综合案例:文件搜索:


public class Demo05Recursion {
    public static void main(String[] args) {
        File file=new File("D:\\abc");
        getAllFile(file);
    }
    /*
    * 定义一个方法,参数传递File类型的目录
    * 方法中对目录进行遍历
    * 只要.txt结尾的文件
    * */
    public static void getAllFile(File dir){
        //System.out.println(dir);//打印被遍历的目录名称
        File[] files = dir.listFiles();
        for (File file : files) {
            //对遍历得到的File对象 file进行判断
            if(file.isDirectory()){
                //file是一个文件夹,则继续遍历这个文件夹
                //直接调用getAllFile方法即可:递归(自己调用自己)。
                getAllFile(file);
            }else {
                //只要.txt结尾的文件
                /*
                * 1.把File对象f,转化为字符串对象
                * */
                //String name = file.getName();
                //String path = file.getPath();
                //String s = file.toString();
                //2.调用String类中的方法endWith判断字符串是否以.txt结尾
                //boolean b = s.endsWith(".txt");
                //3.如果是以.txt结尾的文件,则输出
                /*if(b){
                    System.out.println(file);
                }*/
                if(file.toString().toLowerCase().endsWith(".txt")){
                    System.out.println(file);
                }
            }
        }
    }
}