B. And It‘s Non-Zero

题目链接:https://codeforces.com/problemset/problem/1615/B

位运算之前没怎么写过,所以不会写。留一份题解,作为复习使用。

题解:按位与的结果不为0,则至少有一列全为1.要求删除的数最少,即要求该列原本含有的1最多。则统计每一位的1的个数,找出个数最大值,用数组数字的总数,减去1的个数,即需要删除的个数。

第一版写的是枚举每一个数,对每一个数的每一位用&(1<<j)判断是不是1,用一个数组来统计每一位的结果。然而是TLE...

第二版,借助AI工具,发现了从0开始到某一位的数据的二进制排列规律,用前缀和求解可得。

背景知识

在二进制表示中,每一位(bit)会以固定的周期出现 0 和 1。例如:

  • 第 0 位(最低位):0, 1, 0, 1, 0, 1, ...,周期为 2。
  • 第 1 位:0, 0, 1, 1, 0, 0, 1, 1, ...,周期为 4。
  • 第 2 位:0, 0, 0, 0, 1, 1, 1, 1, ...,周期为 8。

我们可以直接计算某一位为 1 的数量,而不需要逐个遍历每个数字。


公式解释

1. 完整周期的贡献

对于第 j 位:

  • (1 << (j + 1)) 个数字中,第 j 位会有连续 (1 << j) 个 1。
  • 例如:
    • 第 0 位:每 2 个数字中有 1 个 1。
    • 第 1 位:每 4 个数字中有 2 个 1。
    • 第 2 位:每 8 个数字中有 4 个 1。

完整的 (1 << (j + 1)) 块的数量为:(r + 1)/(1 << (j + 1)) +1是因为0也要算进去

这些完整块中,第 j 位的 1 的总数为:(r + 1)/(1 << (j + 1)) * (1 << j)

2. 剩余部分的贡献

可能还有一部分数字不足一个完整的 (1 << (j + 1)) 块,这部分的长度为 (r + 1) % (1 << (j + 1))

在这部分中,第 j 位为 1 的数量为:

  • 如果剩余部分长度大于 (1 << j),那么第 j 位会有 (r + 1) % (1 << (j + 1)) - (1 << j)个 1。
  • 如果剩余部分长度小于等于 (1 << j),那么第 j 位会有 0 个 1。
3. 总计第 j 位的 1 的数量

将完整周期和剩余部分的贡献相加

代码如下:

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;


void count(int r,int*arr)
{
    for(int i=0;i<20;i++)
    {
        int num=0;

        int period=(1<<(i+1));
        int full_p=(r+1)/period;
        int remain=(r+1)%period;
        if(remain>(1<<i))
        {
            num=full_p*(1<<i)+remain-(1<<i);
        }
        else
        {
            num=full_p*(1<<i);
        }
        arr[i]=num;
    }
}



int main()
{
    int t;
    cin>>t;
    int ans[t];
    int k=0;
    while(t--)
    {
        int l,r;
        cin>>l>>r;
        
        int arr_l[20];
        int arr_r[20];
        count(l-1,arr_l);
        count(r,arr_r);
        
        int max=0;
        for(int i=0;i<20;i++)
        {
            if((arr_r[i]-arr_l[i])>max)
            {
                max=arr_r[i]-arr_l[i];
            }
        }
        ans[k++]=r-l+1-max;

    }
    for(int i=0;i<k;i++)
    {
        cout<<ans[i]<<"\n";
    }
    return 0;
}
相关推荐
dulu~dulu4 分钟前
算法---寻找和为K的子数组
笔记·python·算法·leetcode
moonsea020314 分钟前
【无标题】
算法
佑白雪乐35 分钟前
<ACM进度212题>[2026-3-1,2026-3-26]
算法·leetcode
穿条秋裤到处跑39 分钟前
每日一道leetcode(2026.03.26):等和矩阵分割 II
算法·leetcode·矩阵
平凡灵感码头43 分钟前
C语言 printf 数据打印格式速查表
c语言·开发语言·算法
哔哔龙1 小时前
Android OpenCV 实战:图片轮廓提取与重叠轮廓合并处理
android·算法
hz_zhangrl1 小时前
CCF-GESP 等级考试 2026年3月认证C++三级真题解析
c++·算法·程序设计·gesp·gesp2026年3月·gesp c++三级
x_xbx1 小时前
LeetCode:1. 两数之和
数据结构·算法·leetcode
x_xbx1 小时前
LeetCode:49. 字母异位词分组
算法·leetcode·职场和发展
玲娜贝儿--努力学习买大鸡腿版1 小时前
hot 100 刷题记录(1)
数据结构·python·算法