无向红黑图-回溯法

题目

有一个无向红黑图,我们对这张图的每个节点进行染色,只能染成黑色或者红色,要求相邻节点不能同为红色,求有几种染法?

在这里插入图片描述

思路

回溯

代码解析

color_list用来保存每个节点染的颜色,0表示没有染色,1表示黑色,2表示红色

neighbor_list保存每个节点的所有相邻节点

回溯核心代码:
input:节点索引,从0号节点开始染色
base_case:最后一个节点成功染色,返回一种方法
分别遍历两种颜料(1为黑色和2为红色),染过一次黑色后,使用black_color标记当前节点已经染过黑色,当前层不再染黑色;

Python代码

class Solution:
    def can_dyeing(self, index, neighbor_list, color_list):
        for neighbor in neighbor_list[index]:
            if color_list[neighbor] == 2:
                return False
        return True

    def to_color_map(self, number, edge, relationship):
        ans = []
        color_list = [0] * number  # 每个点的染色情况,0表示没有染色,1表示染了黑色,2表示染了红色
        neighbor_list = [[] for _ in range(number)]  # 点的所有邻居
        for a, b in relationship:
            neighbor_list[a].append(b)
            neighbor_list[b].append(a)
        def backtrace(node_index):
            nonlocal neighbor_list, color_list, number, ans
            """
            node_index:当前待染色点的索引
            :param node_index:
            :return:
            """
            if node_index == number:
                ans.append(color_list[:])
                return
            black_color = 0  # 1表示当前层已经染过黑色,0表示还没有染过黑色
            for c in range(1, 3):  # 染色
                if c == 2 and self.can_dyeing(node_index, neighbor_list, color_list):
                    # 当前节点可以染红色
                    color_list[node_index] = c
                else:
                    if black_color:
                        # 剪枝,当前层已经染过黑色
                        continue
                    color_list[node_index] = 1
                    black_color = 1
                backtrace(node_index+1)
                color_list[node_index] = 0  # 回溯

        backtrace(0)
        print(ans)
        return len(ans)

if __name__ == '__main__':
    number = 4
    edge = 3
    relationship = [[0, 1], [0, 2], [1, 2], [3, 3]]
    obj = Solution()
    res = obj.to_color_map(number, edge, relationship)
    print(res)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值