3095. 或值至少 K 的最短子数组 I

【算法题解】最短特别子数组的长度(按位或运算 OR ≥ k)

🧩 题目描述

给你一个 非负整数数组 nums 和一个整数 k。如果某个子数组中所有元素按位或(bitwise OR)之后的结果 大于等于 k,我们就称它为“特别的”。

现在,请你返回 nums最短的特别非空子数组的长度。如果不存在这样的子数组,返回 -1


🔍 示例

输入: nums = [1, 2, 3], k = 3
输出: 1
解释: 子数组 [3] 的 OR = 3,满足条件,长度为 1。

🚀 解题思路

思路一:暴力枚举(TLE 警告)

最直接的方法是:

  • 枚举所有子数组 nums[i:j+1]
  • 对每个子数组计算 OR 值,如果 ≥ k,记录长度并尝试更新最短答案。

时间复杂度:O(n²)
缺点:在大数组下容易超时。


思路二:滑动窗口 + 重计算 OR(中等优化)

我们观察到,按位或是 单调递增操作

给你一个整数 n 和一个在范围 [0, n - 1] 以内的整数 p ,它们表示一个长度为 n 且下标从 0 开始的数组 arr ,数组中除了下标为 p 处是 1 以外,其他所有数都是 0 。 同时给你一个整数数组 banned ,它包含数组中的一些位置。banned 中第 i 个位置表示 arr[banned[i]] = 0 ,题目保证 banned[i] != p 。 你可以对 arr 进行 若干次 操作。一次操作中,你选择大小为 k 的一个 子数组 ,并将它 翻转 。在任何一次翻转操作后,你都需要确保 arr 中唯一的 1 不会到达任何 banned 中的位置。换句话说,arr[banned[i]] 始终 保持 0 。 请你返回一个数组 ans ,对于 [0, n - 1] 之间的任意下标 i ,ans[i] 是将 1 放到位置 i 处的 少 翻转操作次数,如果无法放到位置 i 处,此数为 -1 。 子数组 指的是一个数组里一段连续 非空 的元素序列。 对于所有的 i ,ans[i] 相互之间独立计算。 将一个数组中的元素 翻转 指的是将数组中的变成 相反顺序 。 嗯,这个问题看起来有点挑战性。让我仔细读题目。题目大意是说,我们有一个长度为n的数组arr,其中只有下标p的位置是1,其他都是0。还有一个banned数组,里面的位置不允许在翻转后出现1。我们的操作是每次选择一个长度为k的子数组进行翻转,翻转后必须保证1不能出现在banned的位置。对于每个位置i,要找出将1移动到i处所需的少翻转次数,如果不可能就返回-1。终要返回一个数组ans,每个位置对应结果。 首先,我需要理解翻转操作对1的位置的影响。假设当前1的位置是pos,翻转一个包含pos的子数组,那么翻转后的1的位置会改变。因为翻转子数组会改变元素的顺序,比如子数组从i到i+k-1,那么原来的位置i和i+k-1会被交换,依此类推。假设当前1的位置在子数组中的位置是相对位置m(相对于子数组起始位置),那么翻转后它的新位置会是子数组起始位置 + (k-1 - m)。 举个例子,假设子数组是位置3到5(k=3),当前1在位置4。在子数组中,4的位置是第1位(从0开始的话,3是0,4是1,5是2)。翻转后,这个元素会被放到子数组的起始位置 + (2 -1) =3+1=4的位置?者是不是位置变成起始位置+ (k-1 -m)。比如子数组起始位置是s,长度k,那么原来的位置s+i翻转后会变成s + (k-1 -i)。所以如果原来的1在位置s+i,翻转后变成s + (k-1 -i)。所以每次翻转操作可以看作是移动1的位置到新的位置,这个新的位置的计算方式需要明确。 比如,当前pos在某个子数组内部,翻转后pos会被移动到新的位置。比如子数组s到s+k-1,原来的pos是s+i,翻转后变为s + (k-1 -i) = s +k-1 -i。例如,子数组长度k=3,起始位置s=2,那么子数组是2,3,4。假设pos是3,即i=1,那么翻转后变成s + (2-1)=2+1=3?者原数组的3在子数组中的位置是i=1,翻转后的位置是s + (k-1 -i)=2+2-1=3?所以pos的位置不变?这可能吗? 者应该举具体例子。比如子数组是[0,1,2](k=3,s=0),翻转后变成2,1,0。原来的1在位置0的话,翻转后变成位置2。者原位置0的元素会移动到位置2,原位置2的元素到位置0。所以,假设当前1的位置在子数组的某个位置,翻转后它会被移动到子数组中对应的对称位置。 所以,假设当前1的位置是pos,被包含在选择的子数组[s, s+k-1]中。那么翻转后,新的位置是 s + (s +k-1 - pos) = 2s +k-1 - pos。这可能吗?比如s=0,k=3,pos=0,那么新位置是0 + (0+2-0) =2?是的,因为子数组翻转后原来的0号位置变成2号。那此时,新的pos是2。 所以,每次操作可以将pos移动到某个新的位置,这取决于选择的子数组的位置。那么,问题转化为,从初始位置p出发,每次移动pos到新的位置,且每次移动后的位置不能在banned的位置,同时终要到达目标i的少步数。这看起来像是一个图的路径问题,其中节点是位置,边是可以通过一次翻转操作到达的位置变化。 那这样的话,每个位置可以看作图中的节点,每个节点可以连接到多个其他节点,连接的条件是存在一个长度为k的子数组,使得翻转后能够从当前pos移动到另一个位置,并且移动后的位置不在banned中。然后我们需要找出从初始节点p到其他所有节点的路径。 接下来需要分析,如何生成每个节点的邻接节点。也就是说,对于当前pos,可以翻转哪些子数组,使得翻转后的新位置不被禁止,并且没有被访问过。 但这里的问题可能比较复杂,因为k的长度可能不同,但题目中给出的k是固定的?者题目中的k是否是固定的?原问题中的操作是每次选择一个大小为k的子数组进行翻转。这里可能题目中的k是固定的?原问题描述中的参数是否有k?原题的问题描述中的参数中有一个k吗?者用户的问题是否漏掉了? 仔细看问题描述中的输入参数:用户给出的是整数n,p,banned数组。然后问题描述中说,每次操作选择大小为k的一个子数组。这里的k是否是输入的参数?者题目中是否有其他参数? 根据用户给出的问题描述:“给你一个整数n和一个在范围[0, n-1]的整数p…同时给你一个整数数组banned…你可以进行若干次操作。一次操作中,你选择大小为k的一个子数组…”这里的k应该是固定的,即每次操作必须选择相同长度的k的子数组。但是问题中的参数是否包含k?比如,用户的问题中是否应该包含k作为输入?者题目是否漏掉了? 这可能是原题中的一个参数。比如,原题的输入应该包括k。可能用户的问题中可能将问题描述中的参数遗漏了。者可能用户的问题中的输入参数是正确的? 例如,用户的问题中给出的输入参数是:整数n,整数p,数组banned。但根据问题中的操作描述,每次翻转的子数组大小是k。所以k应该是一个输入参数。但用户的问题中没有提到k,这可能是一个错误。例如,可能用户的问题描述中是否有错误? 现在,假设问题中的k是给定的输入参数。例如,用户的问题可能应该有一个k参数,但是可能被遗漏了?者可能题目中的k是固定的,比如所有操作的子数组长度k是预先确定的? 此时,用户的问题可能存在描述错误,者我在理解上有问题。需要再次确认问题描述。 根据用户的问题描述:在操作中选择大小为k的子数组,那么k应该是一个输入的参数。否则问题无法解决。所以,可能用户在问题描述中漏掉了参数k。者可能原题中的k是另一个参数? 例如,正确的题目应该给出参数k,但用户的问题中没有提到,这会影响整个思路。者可能用户的问题中的k是其他参数? 比如,假设原题的输入是n, p, banned,还有k,那么问题的解答需要考虑k。但在用户的问题中没有提到k,这导致问题描述可能不完整? 者可能用户的问题中给出的参数是正确的,而k可能作为其他方式给出?比如,可能题目中的k是动态变化的?这似乎不太可能。此时,这可能是一个问题描述的错误,者我可能误解了问题。 这时候可能需要假设用户的问题中存在参数k,可能用户的问题中的参数列表有遗漏。例如,原题的正确输入应包括n,p,banned,k。否则无法解题。所以,可能用户在问题描述时漏掉了k。例如,正确的参数应该是:整数n,p,整数k,以及数组banned。 基于此,我必须假设k是给定的输入参数之一,否则问题无法解答。因此,可能用户在问题描述中漏掉了k,者我误解了问题。此时,可能需要假设问题中的每次操作的子数组长度k是给定的。否则,问题将无法处理,因为每次可以选任意长度的子数组,这会使得问题变得非常复杂,甚至可能无法解决。 所以,我需要假设k是作为输入的参数,而用户的问题中可能漏掉了。例如,正确的输入应该包括k,否则问题无法解答。例如,根据原题的描述,用户的问题中的参数可能应该是:整数n,p,以及一个整数k,和数组banned。这样才能确定每次 继续生成 给你一个整数 n 和一个在范围 [0, n - 1] 以内的整数 p ,它们表示一个长度为 n 且下标从 0 开始的数组 arr ,数组中除了下标为 p 处是 1 以外,其他所有数都是 0 。 同时给你一个整数数组 banned ,它包含数组中的一些位置。banned 中第 i 个位置表示 arr[banned[i]] = 0 ,题目保证 banned[i] != p 。 你可以对 arr 进行 若干次 操作。一次操作中,你选择大小为 k 的一个 子数组 ,并将它 翻转 。在任何一次翻转操作后,你都需要确保 arr 中唯一的 1 不会到达任何 banned 中的位置。换句话说,arr[banned[i]] 始终 保持 0 。 请你返回一个数组 ans ,对于 [0, n - 1] 之间的任意下标 i ,ans[i] 是将 1 放到位置 i 处的 少 翻转操作次数,如果无法放到位置 i 处,此数为 -1 。 子数组 指的是一个数组里一段连续 非空 的元素序列。 对于所有的 i ,ans[i] 相互之间独立计算。 将一个数组中的元素 翻转 指的是将数组中的变成 相反顺序 。 用C语言解决
03-21
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值