LeetCode 第141场双周赛个人题解

目录

[构造最小位运算数组 I](#构造最小位运算数组 I)

原题链接

思路分析

AC代码

[3315. 构造最小位运算数组 II](#3315. 构造最小位运算数组 II)

原题链接

思路分析

AC代码

[3316. 从原字符串里进行删除操作的最多次数](#3316. 从原字符串里进行删除操作的最多次数)

原题链接

思路分析

AC代码

[3317. 安排活动的方案数](#3317. 安排活动的方案数)

原题链接

思路分析

AC代码


构造最小位运算数组 I

原题链接

构造最小位运算数组 I

思路分析

见Q2

AC代码

python 复制代码
class Solution:
    def minBitwiseArray(self, nums: list[int]) -> list[int]:
        n = len(nums)
        ans = [-1] * n
        for i, x in enumerate(nums):
            for b in range(x.bit_length() - 1, -1, -1):
                if (x >> b) & 1:
                    t = x ^ (1 << b)
                    if ((t + 1) | t) == x:
                        ans[i] = t
                        break
        return ans

3315. 构造最小位运算数组 II

原题链接

3315. 构造最小位运算数组 II

思路分析

考虑枚举二进制位

如果存在 的话,ansi 一定有一位为0 而原数该位为1,不可能存在两个位为0,因为我们+1最多贡献一个1

所以对于每个numsi,我们枚举二进制位为1的位b,令t = numsi ^ (1 << b)判断是否满足 t | (t + 1) = numsi,取最大的t即可

时间复杂度: O(NlogN)

AC代码

python 复制代码
class Solution:
    def minBitwiseArray(self, nums: list[int]) -> list[int]:
        n = len(nums)
        ans = [-1] * n
        for i, x in enumerate(nums):
            for b in range(x.bit_length() - 1, -1, -1):
                if (x >> b) & 1:
                    t = x ^ (1 << b)
                    if ((t + 1) | t) == x:
                        ans[i] = t
                        break
        return ans

3316. 从原字符串里进行删除操作的最多次数

原题链接

3316. 从原字符串里进行删除操作的最多次数

思路分析

编辑距离问题

很典的一类dp,我们定义 dp(i, j) 为 source 前 i 个字符 和 pattern 前 j 个字符匹配能够删除的最大字符数目

那么 枚举 i,j

如果 i - 1 在 target 内,可分为两种情况:

sourcei - 1 == patternj - 1, dpij 可从 dpi - 1j - 1转移,即不删除

然后dpij 可从 dpi - 1j + 1转移,即删除

剩下的情况类似,无非就是 不+ 1 了,因为 i - 1不在target 内

时间复杂度:O(NM)

AC代码

python 复制代码
class Solution:
    def maxRemovals(self, source: str, pattern: str, targetIndices: List[int]) -> int:
        n = len(source)  
        m = len(pattern)  
        st = set(targetIndices)  
        
        dp = [[-inf] * (m + 1) for _ in range(n + 1)]  
        dp[0][0] = 0
        cnt = 0
        for i in range(1, n + 1):  
            if (i - 1) in st:
                cnt += 1
            dp[i][0] = cnt
            for j in range(1, m + 1):  
                if i - 1 in st:
                    dp[i][j] = dp[i - 1][j] + 1
                    if source[i - 1] == pattern[j - 1]:  
                        dp[i][j] = max(dp[i][j], dp[i - 1][j - 1])  
                elif source[i - 1] == pattern[j - 1]:  
                    dp[i][j] = max(dp[i - 1][j - 1], dp[i - 1][j])
                else:  
                    dp[i][j] = dp[i - 1][j]

        return max(dp[i][m] for i in range(n + 1))

3317. 安排活动的方案数

原题链接

3317. 安排活动的方案数

思路分析

第二类斯特林数

考虑 将 n 个人扔进 x 个集合,这是第二类斯特林数讨论的问题,即集合的分划

由于 集合互异,所以斯特林数应该乘上 factorial(x)

我们枚举 划分数目 i

那么划分方案就是 S(n, i) * fac(i),此时打分可以随便打,即 y ^ i

时间复杂度:O(N^2)

AC代码

python 复制代码
P = 1_000_000_007
N = 1000
memo = [[0] * (N + 1) for _ in range(N + 1)]
memo[0][0] = memo[1][1] = 1
for i in range(2, N + 1):
    memo[i][1] = 1
    for j in range(2, N + 1):
        memo[i][j] = memo[i - 1][j - 1] + j * memo[i - 1][j]
fac = [1] * (N + 1)
for i in range(2, N + 1):
    fac[i] = fac[i - 1] * i % P
comb = cache(comb)
class Solution:
    def numberOfWays(self, n: int, x: int, y: int) -> int:
        res = 0
        for i in range(1, min(n, x) + 1):
            res += memo[n][i] * comb(x, i) % P * pow(y, i, P) % P * fac[i] % P
            if res >= P: res -= P
        return res
相关推荐
wabs66618 小时前
关于贪心算法的思考
算法·贪心算法
社交怪人18 小时前
【判断大小】信息学奥赛一本通C语言解法(题号1043)
算法
Snasph19 小时前
GNU Make 用户手册(中文版)
服务器·算法·gnu
江澎涌19 小时前
拆解与 AI 的一次对话
人工智能·算法·程序员
sheeta199819 小时前
LeetCode 每日一题笔记 日期:2026.06.02 题目:3635. 最早完成陆地和水上游乐设施的时间 II
笔记·算法·leetcode
Lsk_Smion20 小时前
力扣实训 _ [102].层序遍历--前序--后续_递归与非递归的实现
数据结构·算法·leetcode
小欣加油21 小时前
leetcode3751 范围内总波动值I
java·数据结构·c++·算法·leetcode
Halo_tjn1 天前
反射与设计模式1
java·开发语言·算法
V搜xhliang02461 天前
临床科研新范式:从选题到投稿,AI智能体如何接管全流程?
运维·数据结构·人工智能·算法·microsoft·数据挖掘·自动化
计算机安禾1 天前
【算法分析与设计】第46篇:近似难度与不可近似性理论
网络协议·算法·ssl