题解:CF1829H Don‘t Blame Me

一:思路:

在本题,我们可以先设dpi,j为选到第 i 个数时,按位与结果为 j 的方案数

接下来分两种情况分类讨论:

**- 如果不选:加上选到第 i−1 个数的方案数,也就是dpi,j = dpi,j + dpi-1,j

  • 如果选:与上第 i 个数,也就是:dp i,j & a i = dp i,j & ai + dp i-1,j**
  1. 由于题目给出的 k 表示二进制位有 k 个 1,那我们就要在 0-63 中找到所有二进制位中有 k 个 1 的数,并将方案数累加。

  2. 这里的方法找是二进制中有多少个 1,不停的与比当前数少 1 的数进行按位与,这样当目前的数变成 0 时,二进制位 1 的个数也就统计出来了。

代码:

cpp 复制代码
#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int N=2e5+10,mod=1e9+7;
ll t,n,m,a[N];
ll dp[N][80];
ll ldnsbshljbl(ll x){
    ll cnt=0;
    while(x!=0){
        cnt++;
        x&=x-1;
    }
    return cnt;
    }
    int main(){
    scanf("%lld",&t);
    while(t--){
        scanf("%lld%lld",&n,&m);
        for(int i=1;i<=n;i++){
            scanf("%lld",&a[i]);
            for(int j=0;j<64;j++){
                dp[i][j]=0;
            }
            dp[i][a[i]]=1;
        }
        for(int i=1;i<=n;i++){
            for(int j=0;j<64;j++){
                dp[i][j]=(dp[i][j]+dp[i-1][j])%mod;
                dp[i][j&a[i]]=(dp[i][j&a[i]]+dp[i-1][j])%mod;
            }
        }
        ll ans=0;
        for(int i=0;i<64;i++){
            if(ldnsbshljbl(i)==m){
                ans=(ans+dp[n][i])%mod;
            }
        }
        printf("%lld\n",ans);
    }
    return 0;
}
相关推荐
wuweijianlove12 小时前
算法优化的多目标平衡与性能建模研究的技术7
算法
_深海凉_12 小时前
LeetCode热题100-两两交换链表中的节点
算法·leetcode·链表
啊罗罗12 小时前
windows下,c++的axv2+fma/avx-vnni加速计算demo
c++·windows·算法
qq_2837200512 小时前
Embedding 调优实战技巧:从原理到落地,打造高精度向量检索
python·算法·词嵌入·调优
Xpower 1712 小时前
OpenClaw近一月版本更替讲解
人工智能·学习·算法
Ulyanov12 小时前
《从质点到位姿:基于Python与PyVista的导弹制导控制全栈仿真》: 刚体觉醒——6-DOF刚体动力学、四元数与全姿态解算
开发语言·人工智能·python·算法·系统仿真·雷达电子对抗仿真
Dillon Dong12 小时前
机侧同步定子电压环的PI整定:从“对齐三个量“到“闭环四步法“
算法·变流器·风电控制·dfig
Languorous.12 小时前
C++数据结构进阶|图(Graph)详解:从存储到面试高频算法实战
算法
我头发还没掉光~12 小时前
P4147 玉蟾宫
数据结构·c++·算法
枕星而眠13 小时前
栈(Stack)与队列(Queue)核心知识总结
c语言·数据结构·后端·链表