递归
概述
递归:指在当前⽅法内调⽤⾃⼰的这种现象。
递归的分类:
递归分为两种,直接递归和间接递归。
直接递归称为⽅法⾃身调⽤⾃⼰。
间接递归可以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。
分析:
- 接⼝作为参数,需要传递⼦类对象,重写其中⽅法。我们选择匿名内部类⽅式,⽐较简单。
- accept ⽅法,参数为File,表示当前File下所有的⼦⽂件和⼦⽬录。保留住则返回true,过
滤掉则返回false。保留规则:
1. 要么是.java⽂件。
2. 要么是⽬录,⽤于继续遍历。 - 通过过滤器的作⽤, 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);
}
}
}