hihocoder-1546-集合计数

hihocoder-1546-集合计数

#1546 : 集合计数

时间限制:10000ms

单点时限:1000ms

内存限制:256MB

描述

给定一个包含N个整数的集合S={A1, A2, ... AN},以及一个给定的整数K,请计算有多少个S的子集满足其中的最大值与最小值的和小于等于K。

例如对于S={4, 2, 5, 8}以及K=7,满足的条件的子集有以下4个:{2}, {2, 4}, {2, 5}, {2, 4, 5}。

输入

第一行包含两个整数N和K。

第二行包含N个整数A1, A2, ... AN。

对于30%的数据,1 <= N <= 20

对于70%的数据,1 <= N <= 1000

对于100%的数据,1 <= N <= 100000, 0 <= Ai <= 1000000000, 0 <= K <= 2000000000, A1 ~ AN 两两不同。

输出

输出满足条件的子集数目。由于答案可能非常大,你只需要输出答案除以1000000007的余数。

样例输入
4 7
4 2 5 8
样例输出
4

总结:

  (1), 详细审题,看清题意!!!! 是集合的最大值 + 最小值 <= K

  (2), 经过 sort 之后,双端往里面扫。

  假如 刚好到了 num[i] + num[j] <= K , 那么,num[i]一定在集合中,保证这个集合的最小值。所以,剩下的 i + 1, i + 2, ..... j, 是随意在不在这个集合, 则是 2 ^ (j - i) 种集合。

  (3),注意 fast_mode 的 取模操作, 同时对 ans 和 b 取模,保证结果 % MOD。

#include <cstdio>
#include <cstdlib>
const int MAXN = 1e5 + 10;
const int MOD = 1000000007;
int num[MAXN]; 

int cmp(const void *a, const void *b){
    return (*(int *)a - *(int *)b);
}
long long fast_mode(long long a, long long b){
    long long ans = 1;
    while(b > 0){
        if(b%2 == 1){
            ans = ( ans * a ) % MOD;
        }
        a = (a*a) % MOD;
        b /= 2;
    }
    return ans;
}
int main(){

    int n, k;
    long long ans;
    while(scanf("%d %d", &n, &k) != EOF){
        for(int i=0; i<n; ++i){
            scanf("%d", &num[i]);
        }
        qsort(num, n, sizeof(num[0]), cmp);
        ans = 0;
        int j = n-1;
        for(int i=0; i<n && 2*num[i] <= k; ++i){
            while(j>=i && num[j] + num[i] > k){
                --j;
            }
            if(j >= i){
                ans  = (ans + fast_mode(2, j-i)) % MOD;
            }
        }
        printf("%lld\n", ans );
    }
    return 0;
}

  

时间: 2024-10-13 18:22:37

hihocoder-1546-集合计数的相关文章

hihocoder 1546

#1546 : 集合计数 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 给定一个包含N个整数的集合S={A1, A2, ... AN},以及一个给定的整数K,请计算有多少个S的子集满足其中的最大值与最小值的和小于等于K. 例如对于S={4, 2, 5, 8}以及K=7,满足的条件的子集有以下4个:{2}, {2, 4}, {2, 5}, {2, 4, 5}. 输入 第一行包含两个整数N和K. 第二行包含N个整数A1, A2, ... AN. 对于30%的数据,1

【BZOJ2839】集合计数 组合数+容斥

[BZOJ2839]集合计数 Description 一个有N个元素的集合有2^N个不同子集(包含空集),现在要在这2^N个集合中取出若干集合(至少一个),使得它们的交集的元素个数为K,求取法的方案数,答案模1000000007.(是质数喔~) Input 一行两个整数N,K Output 一行为答案. Sample Input 3 2 Sample Output 6 HINT [样例说明]假设原集合为{A,B,C}则满足条件的方案为:{AB,ABC},{AC,ABC},{BC,ABC},{AB

51nod 1352:集合计数

1352 集合计数 基准时间限制:1 秒 空间限制:131072 KB 分值: 20 难度:3级算法题  收藏  关注 给出N个固定集合{1,N},{2,N-1},{3,N-2},...,{N-1,2},{N,1}.求出有多少个集合满足:第一个元素是A的倍数且第二个元素是B的倍数. 提示: 对于第二组测试数据,集合分别是:{1,10},{2,9},{3,8},{4,7},{5,6},{6,5},{7,4},{8,3},{9,2},{10,1}.满足条件的是第2个和第8个. Input 第1行:1

bzoj2839 集合计数

2839: 集合计数 Time Limit: 10 Sec  Memory Limit: 128 MB Submit: 243  Solved: 129 [Submit][Status][Discuss] Description 一个有N个元素的集合有2^N个不同子集(包含空集),现在要在这2^N个集合中取出若干集合(至少一个),使得 它们的交集的元素个数为K,求取法的方案数,答案模1000000007.(是质数喔~) Input 一行两个整数N,K Output 一行为答案. Sample I

bzoj2839 集合计数(容斥)

2839: 集合计数 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 883  Solved: 490[Submit][Status][Discuss] Description 一个有N个元素的集合有2^N个不同子集(包含空集),现在要在这2^N个集合中取出若干集合(至少一个),使得 它们的交集的元素个数为K,求取法的方案数,答案模1000000007.(是质数喔~) Input 一行两个整数N,K Output 一行为答案. Sample Inp

hdu 6092 Rikka with Subset (集合计数,01背包)

Problem Description As we know, Rikka is poor at math. Yuta is worrying about this situation, so he gives Rikka some math tasks to practice. There is one of them: Yuta has n positive A1?An and their sum is m. Then for each subset S of A, Yuta calcula

51nod 1352 集合计数(扩展欧几里得)

题目链接:传送门 题意:略 分析: 非常easy能够得到一个方程 A*x + B*y = N + 1 这式子能够用扩展GCD求出gcd,x和y,然后我们求出大于0的最小x,A*x第一个满足条件的集合firstSet,剩下的N-firstSet个集合能够直接除LCM(A,B)(A和B的最小公倍数)统计出数量. 代码例如以下: #include <stdio.h> #include <string.h> #include <iostream> #define LL long

BZOJ 2839 集合计数 容斥原理

题目大意:给定n个元素,求交集大小为k的集合的集合共有多少种 考虑容斥原理 计算交集大小至少为i的集合有多少种 首先需要选出i个元素 方案为C(n,i) 其它2^(n-i)个集合每个可选可不选 一共2^[2^(n-i)]种 故答案为Σ[k<=i<=n]C(n,i)C(i,k)*2^[2^(n-i)] #include <cstdio> #include <cstring> #include <iostream> #include <algorithm&

51nod 1352 集合计数

给出N个固定集合{1,N},{2,N-1},{3,N-2},...,{N-1,2},{N,1}.求出有多少个集合满足:第一个元素是A的倍数且第二个元素是B的倍数. 提示: 对于第二组测试数据,集合分别是:{1,10},{2,9},{3,8},{4,7},{5,6},{6,5},{7,4},{8,3},{9,2},{10,1}.满足条件的是第2个和第8个. Input 第1行:1个整数T(1<=T<=50000),表示有多少组测试数据. 第2 - T+1行:每行三个整数N,A,B(1<=N

51 Nod 1352 集合计数(中国剩余定理+扩展欧几里得)

题目链接:点我点我 题意:中文题 题解:由题意我们可以构造出方程:Ax+By=N+1,用扩展欧几里得算出最小的非负整数解x(x确定,y也就确定了),然后再把剩余的数分配掉(以它们的最小公倍数去分). 1 #include <cstdio> 2 #include <iostream> 3 #include <algorithm> 4 using namespace std; 5 6 typedef long long LL; 7 8 LL e_gcd(LL a,LL b,