图论解法:哈密顿通路问题 Leetcode 2741. 特别的排列

描述

给你一个下标从 0 开始的整数数组 nums ,它包含 n 个 互不相同 的正整数。如果 nums 的一个排列满足以下条件,我们称它是一个特别的排列:

对于 0 <= i < n - 1 的下标 i ,要么 nums[i] % nums[i+1] == 0 ,要么 nums[i+1] % nums[i] == 0 。

请你返回特别排列的总数目,由于答案可能很大,请将它对 10^9 + 7 取余 后返回。

示例 1:

输入:nums = [2,3,6]

输出:2

解释:[3,6,2] 和 [2,6,3] 是 nums 两个特别的排列。

示例 2:

输入:nums = [1,4,3]

输出:2

解释:[3,1,4] 和 [4,1,3] 是 nums 两个特别的排列。

提示:

2 <= nums.length <= 14
1 <= nums[i] <= 10^9

思路

可以用图中的边表示两个数(结点)能满足题目描述的排列关系。因此问题转化为求出无向图中不重复经过所有节点的路径。

这个问题其实就是哈密顿通路问题。

一开始本人建图后,用dfs搜,超时了。然后改成了记忆化搜索。

进一步可改进地,可以用一个mask来表示搜索的状态(表示visited_map)(题目的2 <= nums.length <= 14其实就有暗示,结点数量有限)

代码

本人写的记忆化搜索

python 复制代码
class Solution:
    def specialPerm(self, nums: List[int]) -> int:
        n = len(nums)
        self.adj_map = {}
        for i,num1 in enumerate(nums):
            for j in range(i+1,n):
                num2 = nums[j]
                if(num1 % num2 != 0 and num2 % num1 != 0): continue
                if(num1 not in self.adj_map): self.adj_map[num1] = set()
                if(num2 not in self.adj_map): self.adj_map[num2] = set()
                self.adj_map[num1].add(num2)
                self.adj_map[num2].add(num1)
        
        if(len(self.adj_map) != n): return 0
        head_nodes = []

        self.remain_nodes_set = set(nums)
        ret = 0
        self.memory_map = {}
        for node in nums:
            ret += self.dfs(node)
        return ret % 1000000007
    
    def dfs(self,head_node):
        status = (head_node,tuple(sorted(list(self.remain_nodes_set))))
        if(status in self.memory_map):
            return self.memory_map[status]
        if(len(self.remain_nodes_set) == 1):
            self.memory_map[status] = 1
            return 1
        ret = 0
        self.remain_nodes_set.remove(head_node)
        for n in self.adj_map[head_node]:
            if(n not in self.remain_nodes_set): continue
            ret += self.dfs(n)
        self.remain_nodes_set.add(head_node)
        self.memory_map[status] = ret
        return ret

这里给出评论区一位大佬写的,也是图的思路

python 复制代码
class Solution:
    def specialPerm(self, nums: List[int]) -> int:
        n = len(nums)
        es = [[] for _ in range(n)]
        for i in range(n):
            for j in range(i+1,n):
                if nums[i]%nums[j]==0 or nums[j]%nums[i]==0:
                    es[i].append(j)
                    es[j].append(i)
        @cache
        def dfs(state, last):
            return sum([dfs(1<<last^state,pre) for pre in es[last] if 1<<pre&state]) if 1<<last^state else 1
        return sum(dfs((1<<n)-1,i) for i in range(n))%int(1e9+7)
相关推荐
A懿轩A15 分钟前
C/C++ 数据结构与算法【数组】 数组详细解析【日常学习,考研必备】带图+详细代码
c语言·数据结构·c++·学习·考研·算法·数组
古希腊掌管学习的神15 分钟前
[搜广推]王树森推荐系统——矩阵补充&最近邻查找
python·算法·机器学习·矩阵
云边有个稻草人19 分钟前
【优选算法】—复写零(双指针算法)
笔记·算法·双指针算法
半盏茶香20 分钟前
在21世纪的我用C语言探寻世界本质 ——编译和链接(编译环境和运行环境)
c语言·开发语言·c++·算法
忘梓.1 小时前
解锁动态规划的奥秘:从零到精通的创新思维解析(3)
算法·动态规划
tinker在coding3 小时前
Coding Caprice - Linked-List 1
算法·leetcode
XH华8 小时前
初识C语言之二维数组(下)
c语言·算法
南宫生8 小时前
力扣-图论-17【算法学习day.67】
java·学习·算法·leetcode·图论
不想当程序猿_8 小时前
【蓝桥杯每日一题】求和——前缀和
算法·前缀和·蓝桥杯
落魄君子8 小时前
GA-BP分类-遗传算法(Genetic Algorithm)和反向传播算法(Backpropagation)
算法·分类·数据挖掘