题解: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;
}
相关推荐
程序员阿鹏2 小时前
56.合并区间
java·数据结构·算法·leetcode
_OP_CHEN3 小时前
C++进阶:(三)深度解析二叉搜索树原理及实现
开发语言·数据结构·c++·二叉树·二叉搜索树·键值对
古译汉书3 小时前
Stm32江科大入门教程--各章节详细笔记---查阅传送门
数据结构·stm32·单片机·嵌入式硬件·算法
武子康5 小时前
Java-165 Neo4j 图论详解 欧拉路径与欧拉回路 10 分钟跑通:Python NetworkX 判定实战
java·数据库·性能优化·系统架构·nosql·neo4j·图论
Jasmine_llq5 小时前
《P2656 采蘑菇》
算法·强连通分量·广度优先搜索(bfs)·tarjan 算法·图的缩点操作·有向无环图(dag)·最长路径
芥子沫5 小时前
《人工智能基础》[算法篇3]:决策树
人工智能·算法·决策树
mit6.8245 小时前
dfs|位运算
算法
苏纪云5 小时前
算法<C++>——双指针 | 滑动窗口
数据结构·c++·算法·双指针·滑动窗口
保持低旋律节奏5 小时前
算法——二叉树、dfs、bfs、适配器、队列练习
算法·深度优先·宽度优先