leetcode力扣刷题系列——【找到按位或最接近 K 的子数组】

题目

给你一个数组 nums 和一个整数 k 。你需要找到 nums 的一个子数组,满足子数组中所有元素按位或运算 OR 的值与 k 的 绝对差 尽可能 小 。换言之,你需要选择一个子数组 nums[l...r] 满足 |k - (nums[l] OR nums[l + 1] ... OR nums[r])| 最小。

请你返回 最小 的绝对差值。

子数组是数组中连续的 非空 元素序列。

示例 1
输入:

nums = [1,2,4,5], k = 3
输出

0
解释:

子数组 nums[0...1] 的按位 OR 运算值为 3 ,得到最小差值 |3 - 3| = 0 。

示例 2:

输入:

nums = [1,3,1,3], k = 2
输出:

1
解释:

子数组 nums[1...1] 的按位 OR 运算值为 3 ,得到最小差值 |3 - 2| = 1 。

示例 3:
输入:

nums = [1], k = 10

输出:

9
解释:

只有一个子数组,按位 OR 运算值为 1 ,得到最小差值 |10 - 1| = 9 。

提示:

1 <= nums.length <= 105

1 <= nums[i] <= 109

1 <= k <= 109

答案

我的方法

方法有问题,这道题在力扣中属于难题,对我而言也不简单,我就是简单的先找到所有的子列表,再找到所有的or值,在进行绝对值的运算,这样子确实简单易懂,但是一旦nums足够大,我们的程序就会超出内存大小。

今天也是没有完成今天的题目,还是因为我太菜啦,想不到更牛的方法,所以今天还是分享一下官方大大的解题思路和方法吧。

python 复制代码
class Solution(object):
    def minimumDifference(self, nums, k):
        """
        :type nums: List[int]
        :type k: int
        :rtype: int
        """
        # 获取数组中所有子数组的值
        subnums=[]
        result = []
        for i in range(len(nums)):
            for j in range(i,len(nums)):
                subnum = nums[i:j+1]
                subnums.append(subnum)
        # 算出每一个或运算的值
        for sublist in subnums:
            if len(sublist) == 1:
                result.append(sublist[0])
            else:
                or_result = sublist[0]
                for num in sublist[1:]:
                    or_result = or_result | num
                result.append(or_result)
        absnums = [abs(num - k) for num in result]
        return min(absnums)

注:我写的方法没啥用,并没有能通过测试

官方大大的方法
思路与算法

题目给定一个数组 nums 和一个整数 k。我们需要找到 nums 的一个非空子数组,要求其所有元素的或运算结果与 k 值尽可能接近。

根据或运算的性质,当我们固定子数组的右端点,不断地向左延伸左端点时,子数组或运算结果逐渐增加,并且结果种类数不超过 log(max(nums))+1 种。因为或运算结果每次增加对应于二进制表示中某些位上由 0 变为 1,因此增加的种类数受到数值范围的限制。

因此,我们从左到右遍历 nums,并在过程中记录每个二进制上的 1 出现的最晚的位置。这些位置用于我们在延伸左端点时,遍历所有种类的或运算结果。具体的,我们用 bits_max_pos[j] 来表示第 j 个二进制为 1 出现的最晚位置,在固定右端点后,将所有的二元组 (bits_max_pos[j],j) 从大到小排序,然后依次遍历这些二元组。遍历时将或运算结果与 2^j做或运算,得到新的结果,并计算其与 k 的差值。最终答案取所有差值的最小值。

需要注意的是,对于不同的 j,其 bits_max_pos[j] 可能相同,在计算区间或运算结果时需要将它们都考虑到,因此这里需要双指针去更新或运算结果。

代码

python 复制代码
n = len(nums)
        bits_max_pos = [-1] * 31
        res = inf
        
        for i in range(n):
            for j in range(31):
                if nums[i] >> j & 1:
                    bits_max_pos[j] = i
            pos_to_bit = [(bits_max_pos[j], j) for j in range(31) if bits_max_pos[j] != -1]
            pos_to_bit.sort(reverse = True, key = lambda x: x[0])
            
            j, val = 0, 0
            while j < len(pos_to_bit):
                p = j
                while j < len(pos_to_bit) and pos_to_bit[j][0] == pos_to_bit[p][0]:
                    val |= 1 << pos_to_bit[j][1]
                    j += 1
                res = min(res, abs(val - k))
        
        return res

作者:力扣官方题解
我是链接哦

来源:力扣(LeetCode)

著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

相关推荐
努力学习编程的伍大侠6 分钟前
基础排序算法
数据结构·c++·算法
ThisIsClark10 分钟前
【后端面试总结】MySQL主从复制逻辑的技术介绍
mysql·面试·职场和发展
XiaoLeisj34 分钟前
【递归,搜索与回溯算法 & 综合练习】深入理解暴搜决策树:递归,搜索与回溯算法综合小专题(二)
数据结构·算法·leetcode·决策树·深度优先·剪枝
Jasmine_llq1 小时前
《 火星人 》
算法·青少年编程·c#
闻缺陷则喜何志丹1 小时前
【C++动态规划 图论】3243. 新增道路查询后的最短距离 I|1567
c++·算法·动态规划·力扣·图论·最短路·路径
Lenyiin1 小时前
01.02、判定是否互为字符重排
算法·leetcode
鸽鸽程序猿2 小时前
【算法】【优选算法】宽搜(BFS)中队列的使用
算法·宽度优先·队列
Jackey_Song_Odd2 小时前
C语言 单向链表反转问题
c语言·数据结构·算法·链表
Watermelo6172 小时前
详解js柯里化原理及用法,探究柯里化在Redux Selector 的场景模拟、构建复杂的数据流管道、优化深度嵌套函数中的精妙应用
开发语言·前端·javascript·算法·数据挖掘·数据分析·ecmascript