Java学习笔记——递归

递归

概述
递归:指在当前⽅法内调⽤⾃⼰的这种现象。
递归的分类:
递归分为两种,直接递归和间接递归。
直接递归称为⽅法⾃身调⽤⾃⼰。
间接递归可以A⽅法调⽤B⽅法,B⽅法调⽤C⽅法,C⽅法调⽤A⽅法。
注意事项:
递归⼀定要有条件限定,保证递归能够停⽌下来,否则会发⽣栈内存溢出。
在递归中虽然有限定条件,但是递归次数不能太多。否则也会发⽣栈内存溢出。
构造⽅法,禁⽌递归

举例

public class Demo01DiGui {
 public static void main(String[] args) {
 // a();
 b(1);
 }

 /*
 * 3.构造⽅法,禁⽌递归
 * 编译报错:构造⽅法是创建对象使⽤的,不能让对象⼀直创建下去
 */
 public Demo01DiGui() {
 // Demo01DiGui();
 }
 /*
 * 2.在递归中虽然有限定条件,但是递归次数不能太多。否则也会发⽣栈内存溢出。
 * 4993
 * Exception in thread "main" java.lang.StackOverflowError
 */
 private static void b(int i) {
 System.out.println(i);
 //添加⼀个递归结束的条件,i==5000的时候结束
 if (i == 5000) {
 return; // 结束⽅法
 }
 b(++i);
 }
 /*
 * 1.递归⼀定要有条件限定,保证递归能够停⽌下来,否则会发⽣栈内存溢出。
Exception in thread "main"
 * java.lang.StackOverflowError
 */
 private static void a() {
 System.out.println("a⽅法");
 a();
 }
}

递归累加求和
计算1 ~ n的和
分析:num的累和 = num + (num-1)的累和,所以可以把累和的操作定义成⼀个⽅法,递归调
⽤。

举例

public class DiGuiDemo {
 public static void main(String[] args) {
 //计算1~num的和,使⽤递归完成
 int num = 5;
 // 调⽤求和的⽅法
 int sum = getSum(num);
 // 输出结果
 System.out.println(sum);
 }
 /*
 * 通过递归算法实现.
 * 参数列表:int
 * 返回值类型:int
 */
 public static int getSum(int num) {
 /*
 * num为1时,⽅法返回1,
 * 相当于是⽅法的出⼝,num总有是1的情况
 */
 if (num == 1) {
 return 1;
 }
 /*
 * num不为1时,⽅法返回 num +(num-1)的累和
 * 递归调⽤getSum⽅法
 */
 return num + getSum(num - 1);
 }
}

代码执⾏图解
在这里插入图片描述
注:递归求阶乘也是类似就不举例了。

递归打印多级⽬录

分析:多级⽬录的打印,就是当⽬录的嵌套。遍历之前,⽆从知道到底有多少级⽬录,所以我们还是要使⽤递归实现。

举例

public class DiGuiDemo2 {
 public static void main(String[] args) {
 // 创建File对象
 File dir = new File("D:\\aaa");
 // 调⽤打印⽬录⽅法
 printDir(dir);
 }
 public static void printDir(File dir) {
 // 获取⼦⽂件和⽬录
 File[] files = dir.listFiles();
 // 循环打印
 /*
 * 判断:
 * 当是⽂件时,打印绝对路径.
 * 当是⽬录时,继续调⽤打印⽬录的⽅法,形成递归调⽤.
 */
 for (File file : files) {
 // 判断
 if (file.isFile()) {
 // 是⽂件,输出⽂件绝对路径
 System.out.println("⽂件名:" + file.getAbsolutePath());
 } else {
 // 是⽬录,输出⽬录绝对路径
 System.out.println("⽬录:" + file.getAbsolutePath());
 // 继续遍历,调⽤printDir,形成递归
 printDir(file);
 }
 }
 }
}

⽂件过滤器优化
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);
 }
 }
 }
}

Lambda优化

分析: FileFilter 是只有⼀个⽅法的接⼝,因此可以⽤lambda表达式简写。
lambda格式:

() -> { }

举例

public static void printDir3(File dir) {
 // lambda的改写
 File[] files = dir.listFiles(f ->{
 return f.getName().endsWith(".java") || f.isDirectory();
 });
 // 循环打印
 for (File file : files) {
 if (file.isFile()) {
 System.out.println("⽂件名:" + file.getAbsolutePath());
  } else {
 printDir3(file);
 }
 }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值