首页 > 其他 > 详细

容斥原理

时间:2018-08-17 00:22:56      阅读:158      评论:0      收藏:0      [点我收藏+]

1,交集(∩)

多个事件若需同时满足,则取交集

2,并集(∪)

满足任意一个事件即可,则取并集

3,容斥原理

即先容再斥,以集合为例,求集合并。

则可以先单独考虑每个集合,然后直接都加起来,这样的话集合间的交集就会被重复计数,所以就有了容斥原理来消除这个重复计数。 

技术分享图片

其实就是在将问题分成单个问题来考虑,原问题可以表示为单个问题的并集,且交集很好求,就能很愉快地使用容斥原理了。

德摩根定律:非(P 且 Q) = (非 P) 或 (非 Q),非(P 或 Q) = (非 P) 且 (非 Q)

有了这些知识就可以来分析杭电多校的一道题:

技术分享图片

step1,当括号里的条件只为Xi>=0时,发现答案为C(k+m-1,m-1)

step2,再考虑有限制时,设集合Ai表示Xi超出限制,则原问题可以表示为技术分享图片

这就显然可以容斥定理了。

step3,Xi超出了限制,则它的取值范围为Xi >= n,那么给Xi减去n便使它的取值范围又回到了step1,再结合下德摩根定律便能得到答案了。

代码:

#include<stdio.h>
#include<algorithm>
#include<string.h>
#include<map>
#include<queue>
#include<set>
#include<iostream>
#include<vector> 
#define MAXN 1000005 
using namespace std;
typedef long long ll;

const ll mod = 998244353;
ll fact[200005], ifact[200005], inv[200005];

void init() {
    fact[0] = 1;
    for(int i=1;i<=200000;i++) fact[i] = fact[i-1] * i % mod;
    inv[1] = 1;
    for (int i = 2; i <= 200000; i++)
        inv[i] = (mod - mod / i) * inv[mod % i] % mod;
    ifact[0] = 1;
    for(int i=1;i<=200000;i++) ifact[i] = ifact[i-1] * inv[i] % mod;
}
ll C(ll a,ll b)
{
    if(b>a||a<0||b<0) return 0;
    return (fact[a]*ifact[b]%mod)*ifact[a-b]%mod;
}
int main()
{
    int T;
    init();
    cin>>T;
    
    while(T--)
    {
        ll n,m,k;
        cin>>n>>m>>k;
        ll sum=0;
        
        for(ll i=0;i<=m;i++)
        {
            if(i%2==0)
            {
                sum=(sum+C(m,i)*C(k-i*n+m-1,k-i*n)%mod)%mod;
            }
            else sum=(sum-C(m,i)*C(k-i*n+m-1,k-i*n)%mod)%mod;
            
        }
        cout<<(sum+mod)%mod<<endl;
    }
    
}

 

容斥原理

原文:https://www.cnblogs.com/lnu161403214/p/9490857.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!