当前位置: 首页 > news >正文

蓝桥杯 之 LCA算法

文章目录

  • 习题
    • 1483.树节点的第K个祖先
      • 拓展:LCA

  • LCA问题,就是最近公共祖先的问题

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

习题

1483.树节点的第K个祖先

1483.树节点的第K个祖先

在这里插入图片描述

  • 普通的做法,当然是一个个往上面搜索,但是这样的话时间复杂度是o(k),那么能不能每次求解的是爷爷节点,这样就是按照二进制的步子进行寻找

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

class TreeAncestor:

    def __init__(self, n: int, parent: List[int]):
        # bit_length()表示二进制的位数,因为pa[x][0]直接就是父亲节点,所以只用少一位即可
        m = n.bit_length() - 1
        # 初始化
        pa = [[p] + [-1]*m for p in parent]
        # 注意这个遍历,先枚举这个i再枚举这个x,先算出全部节点的所有爷爷节点,再算出所有爷爷的爷爷节点
        for i in  range(m):
            for x in range(n):
                p = pa[x][i]
                if p != -1:
                    pa[x][i+1] = pa[p][i]
        self.pa = pa

    def getKthAncestor(self, node: int, k: int) -> int:
        for i in range(k.bit_length()):
            # 这个k>>i的判断就十分巧妙
            # 并且这个node 是全局的,i从0开始判断
            if (k >> i) & 1:
                node = self.pa[node][i]
                if node < 0 :
                    break
        return node

拓展:LCA

在这里插入图片描述
在这里插入图片描述

  • 初始化
# 
class TreeAncestor:
    def __init__(self, edges: List[List[int]]):
        n = len(edges) + 1
        m = n.bit_length()
        g = [[] for _ in range(n)]
        for x, y in edges:  # 节点编号从 0 开始
            g[x].append(y)
            g[y].append(x)

        depth = [0] * n
        pa = [[-1] * m for _ in range(n)]
        def dfs(x: int, fa: int) -> None:
            pa[x][0] = fa
            for y in g[x]:
                if y != fa:
                    depth[y] = depth[x] + 1
                    dfs(y, x)
        dfs(0, -1)

        for i in range(m - 1):
            for x in range(n):
                if (p := pa[x][i]) != -1:
                    pa[x][i + 1] = pa[p][i]
        self.depth = depth
        self.pa = pa

    def get_kth_ancestor(self, node: int, k: int) -> int:
        for i in range(k.bit_length()):
            if (k >> i) & 1:  # k 二进制从低到高第 i 位是 1
                node = self.pa[node][i]
        return node

    # 返回 x 和 y 的最近公共祖先(节点编号从 0 开始)
    def get_lca(self, x: int, y: int) -> int:
        if self.depth[x] > self.depth[y]:
            x, y = y, x
        # 使 y 和 x 在同一深度
        y = self.get_kth_ancestor(y, self.depth[y] - self.depth[x])
        if y == x:
            return x
        for i in range(len(self.pa[x]) - 1, -1, -1):
            px, py = self.pa[x][i], self.pa[y][i]
            if px != py:
                x, y = px, py  # 同时上跳 2**i 步
        return self.pa[x][0]

相关文章:

  • 搜广推校招面经六十一
  • 多线程案例-单例模式
  • Tcp套接字编程
  • go - grpc入门
  • 5G_WiFi_CE_杂散测试
  • C语言入门教程100讲(0)从了解C语言的发展史开始
  • 3月29日星期六今日早报简报微语报早读
  • 【Qt】Qt 类的继承与内存管理详解:QObject、信号槽与隐式共享
  • Conda配置Python环境
  • 实时目标检测新突破:AnytimeYOLO——随时中断的YOLO优化框架解析
  • 侯捷 C++ 课程学习笔记:C++ 中引用与指针的深度剖析
  • CS2 DEMO导入blender(慢慢更新咯)
  • Mayo Clinic Platform在人工智能医疗领域的现状及启示意义研究
  • 深度学习——图像余弦相似度
  • 基于华为设备技术的端口类型详解
  • 嵌入式八股RTOS与Linux--中断篇
  • vue如何实现前端控制动态路由
  • 基于pycatia的CATIA零部件激活状态管理技术解析
  • Centos7,tar包方式部署rabbitmq-3.7.6
  • C++ 初阶总复习 (16~30)
  • 人民日报:在大有可为的时代大有作为
  • 上海科创的三种品格
  • 宁夏民政厅原厅长欧阳艳已任自治区政府副秘书长、办公厅主任
  • 华侨城A:一季度营收53.63亿元,净利润亏损14.19亿元
  • 港理大公布多项AI+医工成果,助港建设国际医疗创新枢纽
  • 第五届全国医院人文管理路演在昆山举办:患者体验才是温度计