算法之旅(九)算法核心思想之一 递归思想

递归思想

定义
递归是一种算法思想,其中一个函数直接或间接地调用自身来解决问题。递归通常包含一个基本情况(结束条件)和一个或多个递归情况(函数调用自身)。

注意点:

  • 1.调用自身
  • 2.设置终止条件

作用
递归可以使复杂的问题简化为更简单的子问题,便于求解。它特别适用于具有重叠子问题和自相似结构的问题,如树、图、排序等。

递归的常规应用

我以数据结构树和栈作为演示例子

  1. 树的遍历
    在树结构中,递归通常用于深度优先遍历(DFS)
    如前序遍历、中序遍历和后序遍历。

    • Demo Code
    class TreeNode {
        int val;
        TreeNode left;
        TreeNode right;
        TreeNode(int x) { val = x; }
    }
    
    public void preorderTraversal(TreeNode root) {
        if (root == null) return;
        System.out.print(root.val + " ");
        preorderTraversal(root.left);
        preorderTraversal(root.right);
    }
    
  2. 栈的实现

    • 递归可以用于实现栈的基本操作,例如入栈和出栈。
      通过递归可以实现栈的反转或深拷贝等操作。
    • Demo Code
    public void reverseStack(Stack<Integer> stack) {
        if (stack.isEmpty()) return;
        int top = stack.pop();
        reverseStack(stack);
        insertAtBottom(stack, top);
    }
    
    private void insertAtBottom(Stack<Integer> stack, int value) {
        if (stack.isEmpty()) {
            stack.push(value);
        } else {
            int top = stack.pop();
            insertAtBottom(stack, value);
            stack.push(top);
        }
    }
    

递归在技术栈中的应用

1. 微服务的循环依赖

  • 原理:在Spring Cloud的微服务架构中,递归可以用于处理服务间的调用关系。例如,当一个服务需要调用另一个服务,而这个被调用的服务又需要回调原服务时,就可能形成递归调用。
  • 示例:假设有两个服务 A 和 B,A 调用 B 的某个接口,B 在处理完请求后需要再调用 A 的另一个接口。可以使用递归来处理这种情况。
@RestController
public class ServiceAController {
    @Autowired
    private RestTemplate restTemplate;

    @GetMapping("/callB")
    public String callB() {
        // 调用服务 B
        return restTemplate.getForObject("http://service-b/callA", String.class);
    }
}

@RestController
public class ServiceBController {
    @Autowired
    private RestTemplate restTemplate;

    @GetMapping("/callA")
    public String callA() {
        // 递归调用服务 A
        return restTemplate.getForObject("http://service-a/callB", String.class);
    }
}

这里会有循环依赖问题,解决方案

解决循环依赖的方法

循环依赖常规解决方案一 使用异步调用
  • 将服务之间的同步调用改为异步消息传递,例如使用消息队列(如 RabbitMQ、Kafka)。这样,服务 A 可以将请求发送到消息队列,而不直接依赖于服务 B 的响应。

示例:使用消息队列解决循环依赖

假设有服务 A 和服务 B,服务 A 需要处理用户请求,并将结果发送到服务 B,而服务 B 又需要对服务 A 的结果进行反馈。

// 服务 A
@RabbitListener(queues = "requestQueue")
public void handleRequest(String request) {
    // 处理请求
    String result = processRequest(request);
    // 发送结果到队列
    rabbitTemplate.convertAndSend("responseQueue", result);
}

// 服务 B
@RabbitListener(queues = "responseQueue")
public void handleResponse(String response) {
    // 处理服务 A 的结果
    processResponse(response);
}

在这个例子中,服务 A 和服务 B 通过消息队列进行交互,消除了直接的循环依赖,从而避免了潜在的问题。同时,通过懒加载的方式,进一步降低了服务间的耦合度,使得服务的初始化更加灵活。

延迟加载(懒加载)

  • 利用懒加载的方式来处理循环依赖。在服务启动时,不立即初始化某些依赖,而是在需要使用时再进行初始化。可以通过 Spring 的 @Lazy 注解来实现懒加载。
@Service
public class ServiceA {
    private final ServiceB serviceB;

    @Autowired
    public ServiceA(@Lazy ServiceB serviceB) {
        this.serviceB = serviceB; // 懒加载 ServiceB
    }

    public void someMethod() {
        serviceB.performAction();
    }
}

@Service
public class ServiceB {
    private final ServiceA serviceA;

    @Autowired
    public ServiceB(@Lazy ServiceA serviceA) {
        this.serviceA = serviceA; // 懒加载 ServiceA
    }

    public void performAction() {
        serviceA.someMethod();
    }
}
循环依赖常规解决方案三 配置中心
  • 在 Spring Cloud 中,可以利用配置中心(如 Spring Cloud Config)来集中管理配置和依赖,减少服务间的直接依赖。

2. 中间件RabbitMQ的应用:只是场景之一其他队列中间件同理

  • 原理:在消息队列的处理中,递归可以用于实现消息的重试机制。例如,如果某个消息处理失败,可以通过递归调用来重新处理该消息,直到成功或达到最大重试次数。
  • 示例:在 RabbitMQ 的消费者中,可以使用递归来实现重试机制。
@RabbitListener(queues = "myQueue")
public void consumeMessage(String message) {
    try {
        // 处理消息
        processMessage(message);
    } catch (Exception e) {
        // 如果处理失败,递归重试
        retryConsumeMessage(message, 1);
    }
}

private void retryConsumeMessage(String message, int retryCount) {
    if (retryCount <= 3) { // 最大重试次数为 3
        try {
            // 处理消息
            processMessage(message);
        } catch (Exception e) {
            // 递归调用重试
            retryConsumeMessage(message, retryCount + 1);
        }
    } else {
        // 达到最大重试次数,记录日志或发送到死信队列
        log.error("Message failed after retries: " + message);
    }
}

3. 文件系统遍历

在操作系统中,文件系统可以被视为一个树形结构,递归地遍历文件夹和文件是常见的操作。

思路

  • 使用递归遍历每个文件夹,查找其中的文件和子文件夹。

Demo Code

import java.io.File;

public class FileSystemTraversal {
    public static void listFiles(String directoryName) {
        File directory = new File(directoryName);
        File[] files = directory.listFiles();

        if (files != null) {
            for (File file : files) {
                if (file.isDirectory()) {
                    System.out.println("Directory: " + file.getName());
                    listFiles(file.getAbsolutePath()); // 递归调用
                } else {
                    System.out.println("File: " + file.getName());
                }
            }
        }
    }

    public static void main(String[] args) {
        listFiles("C:\\example"); // 替换为你要遍历的目录
    }
}

复杂度

  • 时间复杂度:O(n),n 为文件数量。
  • 空间复杂度:O(h),h 为树的高度(递归栈的深度)。

4. JSON 解析

在前端开发和后端服务中,JSON 数据结构通常是嵌套的,递归解析 JSON 对象是一种常见的需求。

思路

  • 使用递归方法来遍历 JSON 对象的每个键值对,处理嵌套的情况。

Demo code

import org.json.JSONObject;

public class JsonParser {
    public static void parseJson(JSONObject jsonObject, String prefix) {
        for (String key : jsonObject.keySet()) {
            Object value = jsonObject.get(key);
            if (value instanceof JSONObject) {
                parseJson((JSONObject) value, prefix + key + "."); // 递归解析
            } else {
                System.out.println(prefix + key + ": " + value);
            }
        }
    }

    public static void main(String[] args) {
        String jsonString = "{\"name\":\"John\", \"age\":30, \"address\":{\"city\":\"New York\", \"zip\":\"10001\"}}";
        JSONObject jsonObject = new JSONObject(jsonString);
        parseJson(jsonObject, ""); // 开始解析
    }
}

复杂度

  • 时间复杂度:O(n),n 为 JSON 对象中键值对的数量。
  • 空间复杂度:O(h),h 为嵌套层数。

5. 爬虫程序

在网络爬虫中,递归地访问网页并提取信息是非常常见的做法,特别是当网页内容是多层嵌套时。

思路

  • 访问网页,提取其中的链接并递归访问。

Demo Code

import org.jsoup.Jsoup;
import org.jsoup.nodes.Document;
import org.jsoup.nodes.Element;

import java.io.IOException;

public class WebCrawler {
    public static void crawl(String url, int depth) {
        if (depth == 0) return; // 结束条件
        try {
            Document doc = Jsoup.connect(url).get();
            System.out.println("Crawling: " + url);
            for (Element link : doc.select("a[href]")) {
                crawl(link.attr("abs:href"), depth - 1); // 递归访问链接
            }
        } catch (IOException e) {
            System.err.println("Error crawling: " + url);
        }
    }

    public static void main(String[] args) {
        crawl("http://example.com", 2); // 从指定URL开始,限制深度
    }
}

复杂度

  • 时间复杂度:O(n),n 为访问的网页数量。
  • 空间复杂度:O(h),h 为递归栈的深度。

递归在各种技术栈中发挥着重要作用,特别是在处理树形结构、嵌套数据和复杂系统时。
理解递归的原理和应用场景对于开发者来说至关重要,可以帮助他们解决更复杂的问题。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,290评论 6 491
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,107评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,872评论 0 347
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,415评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,453评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,784评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,927评论 3 406
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,691评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,137评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,472评论 2 326
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,622评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,289评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,887评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,741评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,977评论 1 265
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,316评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,490评论 2 348

推荐阅读更多精彩内容