递归概念&分类&注意事项:
递归:指在当前方法中调用自己的这种现象
递归的分类:直接递归和间接递归
直接递归:方法自身调用自己
间接递归: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); } } } } }